| /****************************************************************************** |
| * |
| * Copyright (C) 2014 Google, Inc. |
| * |
| * 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. |
| * |
| ******************************************************************************/ |
| |
| #define LOG_TAG "bt_osi_future" |
| |
| #include "osi/include/future.h" |
| |
| #include <base/logging.h> |
| |
| #include "osi/include/allocator.h" |
| #include "osi/include/log.h" |
| #include "osi/include/osi.h" |
| #include "osi/include/semaphore.h" |
| |
| struct future_t { |
| bool ready_can_be_called; |
| semaphore_t* semaphore; // NULL semaphore means immediate future |
| void* result; |
| }; |
| |
| static void future_free(future_t* future); |
| |
| future_t* future_new(void) { |
| future_t* ret = static_cast<future_t*>(osi_calloc(sizeof(future_t))); |
| |
| ret->semaphore = semaphore_new(0); |
| if (!ret->semaphore) { |
| LOG_ERROR(LOG_TAG, "%s unable to allocate memory for the semaphore.", |
| __func__); |
| goto error; |
| } |
| |
| ret->ready_can_be_called = true; |
| return ret; |
| error:; |
| future_free(ret); |
| return NULL; |
| } |
| |
| future_t* future_new_immediate(void* value) { |
| future_t* ret = static_cast<future_t*>(osi_calloc(sizeof(future_t))); |
| |
| ret->result = value; |
| ret->ready_can_be_called = false; |
| return ret; |
| } |
| |
| void future_ready(future_t* future, void* value) { |
| CHECK(future != NULL); |
| CHECK(future->ready_can_be_called); |
| |
| future->ready_can_be_called = false; |
| future->result = value; |
| semaphore_post(future->semaphore); |
| } |
| |
| void* future_await(future_t* future) { |
| CHECK(future != NULL); |
| |
| // If the future is immediate, it will not have a semaphore |
| if (future->semaphore) semaphore_wait(future->semaphore); |
| |
| void* result = future->result; |
| future_free(future); |
| return result; |
| } |
| |
| static void future_free(future_t* future) { |
| if (!future) return; |
| |
| semaphore_free(future->semaphore); |
| osi_free(future); |
| } |