--- /dev/null
+/*
+ * Copyright (C) 2010 The Android Open Source Project
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* ThreadSync implementation */
+
+#include "sles_allinclusive.h"
+
+
+static SLresult IThreadSync_EnterCriticalSection(SLThreadSyncItf self)
+{
+ SL_ENTER_INTERFACE
+
+ IThreadSync *this = (IThreadSync *) self;
+ interface_lock_exclusive(this);
+ for (;;) {
+ if (this->mInCriticalSection) {
+ if (!pthread_equal(this->mOwner, pthread_self())) {
+ ++this->mWaiting;
+ interface_cond_wait(this);
+ continue;
+ }
+ // nested locks are not allowed
+ result = SL_RESULT_PRECONDITIONS_VIOLATED;
+ break;
+ }
+ this->mInCriticalSection = SL_BOOLEAN_TRUE;
+ this->mOwner = pthread_self();
+ result = SL_RESULT_SUCCESS;
+ break;
+ }
+ interface_unlock_exclusive(this);
+
+ SL_LEAVE_INTERFACE
+}
+
+
+static SLresult IThreadSync_ExitCriticalSection(SLThreadSyncItf self)
+{
+ SL_ENTER_INTERFACE
+
+ IThreadSync *this = (IThreadSync *) self;
+ interface_lock_exclusive(this);
+ if (!this->mInCriticalSection || !pthread_equal(this->mOwner, pthread_self())) {
+ result = SL_RESULT_PRECONDITIONS_VIOLATED;
+ } else {
+ this->mInCriticalSection = SL_BOOLEAN_FALSE;
+ memset(&this->mOwner, 0, sizeof(pthread_t));
+ if (this->mWaiting) {
+ --this->mWaiting;
+ interface_cond_signal(this);
+ }
+ result = SL_RESULT_SUCCESS;
+ }
+ interface_unlock_exclusive(this);
+
+ SL_LEAVE_INTERFACE
+}
+
+
+static const struct SLThreadSyncItf_ IThreadSync_Itf = {
+ IThreadSync_EnterCriticalSection,
+ IThreadSync_ExitCriticalSection
+};
+
+void IThreadSync_init(void *self)
+{
+ IThreadSync *this = (IThreadSync *) self;
+ this->mItf = &IThreadSync_Itf;
+ this->mInCriticalSection = SL_BOOLEAN_FALSE;
+ this->mWaiting = 0;
+ memset(&this->mOwner, 0, sizeof(pthread_t));
+}
+
+void IThreadSync_deinit(void *self)
+{
+ IThreadSync *this = (IThreadSync *) self;
+ if (this->mInCriticalSection) {
+ SL_LOGW("ThreadSync::EnterCriticalSection was active at Engine::Destroy");
+ }
+}