blob: b8dddcf2b7970af881f9a49a2ebee942c821a9ff [file] [log] [blame]
Sharvil Nanavatia3164c92014-06-23 12:07:05 -07001/******************************************************************************
2 *
3 * Copyright (C) 2014 Google, Inc.
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at:
8 *
9 * http://www.apache.org/licenses/LICENSE-2.0
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 *
17 ******************************************************************************/
18
Chris Mantonf8027002015-03-12 09:22:48 -070019#define LOG_TAG "bt_osi_semaphore"
Sharvil Nanavatic11b4072014-05-02 23:55:09 -070020
21#include <assert.h>
22#include <errno.h>
Sharvil Nanavatia3164c92014-06-23 12:07:05 -070023#include <fcntl.h>
Sharvil Nanavatic11b4072014-05-02 23:55:09 -070024#include <string.h>
25#include <sys/eventfd.h>
Sharvil Nanavatic11b4072014-05-02 23:55:09 -070026
Sharvil Nanavati0f9b91e2015-03-12 15:42:50 -070027#include "osi/include/allocator.h"
28#include "osi/include/osi.h"
Sharvil Nanavati44802762014-12-23 23:08:58 -080029#include "osi/include/log.h"
Sharvil Nanavati0f9b91e2015-03-12 15:42:50 -070030#include "osi/include/semaphore.h"
Sharvil Nanavatic11b4072014-05-02 23:55:09 -070031
32#if !defined(EFD_SEMAPHORE)
33# define EFD_SEMAPHORE (1 << 0)
34#endif
35
36struct semaphore_t {
37 int fd;
38};
39
40semaphore_t *semaphore_new(unsigned int value) {
Zach Johnson384f8a92014-08-25 23:22:24 -070041 semaphore_t *ret = osi_malloc(sizeof(semaphore_t));
Sharvil Nanavatic11b4072014-05-02 23:55:09 -070042 if (ret) {
43 ret->fd = eventfd(value, EFD_SEMAPHORE);
Sharvil Nanavati2cb29982014-08-01 17:00:12 -070044 if (ret->fd == INVALID_FD) {
Sharvil Nanavati44802762014-12-23 23:08:58 -080045 LOG_ERROR("%s unable to allocate semaphore: %s", __func__, strerror(errno));
Sharvil Nanavatic0745da2014-11-13 01:04:19 -080046 osi_free(ret);
Sharvil Nanavatic11b4072014-05-02 23:55:09 -070047 ret = NULL;
48 }
49 }
50 return ret;
51}
52
53void semaphore_free(semaphore_t *semaphore) {
Zach Johnson207fa232014-09-18 17:29:54 -070054 if (!semaphore)
55 return;
56
Sharvil Nanavati2cb29982014-08-01 17:00:12 -070057 if (semaphore->fd != INVALID_FD)
Sharvil Nanavatic11b4072014-05-02 23:55:09 -070058 close(semaphore->fd);
Zach Johnson384f8a92014-08-25 23:22:24 -070059 osi_free(semaphore);
Sharvil Nanavatic11b4072014-05-02 23:55:09 -070060}
61
62void semaphore_wait(semaphore_t *semaphore) {
63 assert(semaphore != NULL);
Sharvil Nanavati2cb29982014-08-01 17:00:12 -070064 assert(semaphore->fd != INVALID_FD);
Sharvil Nanavatic11b4072014-05-02 23:55:09 -070065
66 uint64_t value;
67 if (eventfd_read(semaphore->fd, &value) == -1)
Sharvil Nanavati44802762014-12-23 23:08:58 -080068 LOG_ERROR("%s unable to wait on semaphore: %s", __func__, strerror(errno));
Sharvil Nanavatic11b4072014-05-02 23:55:09 -070069}
70
Sharvil Nanavatia3164c92014-06-23 12:07:05 -070071bool semaphore_try_wait(semaphore_t *semaphore) {
72 assert(semaphore != NULL);
Sharvil Nanavati2cb29982014-08-01 17:00:12 -070073 assert(semaphore->fd != INVALID_FD);
Sharvil Nanavatia3164c92014-06-23 12:07:05 -070074
75 int flags = fcntl(semaphore->fd, F_GETFL);
76 if (flags == -1) {
Sharvil Nanavati44802762014-12-23 23:08:58 -080077 LOG_ERROR("%s unable to get flags for semaphore fd: %s", __func__, strerror(errno));
Sharvil Nanavatia3164c92014-06-23 12:07:05 -070078 return false;
79 }
80 if (fcntl(semaphore->fd, F_SETFL, flags | O_NONBLOCK) == -1) {
Sharvil Nanavati44802762014-12-23 23:08:58 -080081 LOG_ERROR("%s unable to set O_NONBLOCK for semaphore fd: %s", __func__, strerror(errno));
Sharvil Nanavatia3164c92014-06-23 12:07:05 -070082 return false;
83 }
84
85 eventfd_t value;
86 if (eventfd_read(semaphore->fd, &value) == -1)
87 return false;
88
89 if (fcntl(semaphore->fd, F_SETFL, flags) == -1)
Sharvil Nanavati44802762014-12-23 23:08:58 -080090 LOG_ERROR("%s unable to resetore flags for semaphore fd: %s", __func__, strerror(errno));
Sharvil Nanavatia3164c92014-06-23 12:07:05 -070091 return true;
92}
93
Sharvil Nanavatic11b4072014-05-02 23:55:09 -070094void semaphore_post(semaphore_t *semaphore) {
95 assert(semaphore != NULL);
Sharvil Nanavati2cb29982014-08-01 17:00:12 -070096 assert(semaphore->fd != INVALID_FD);
Sharvil Nanavatic11b4072014-05-02 23:55:09 -070097
98 if (eventfd_write(semaphore->fd, 1ULL) == -1)
Sharvil Nanavati44802762014-12-23 23:08:58 -080099 LOG_ERROR("%s unable to post to semaphore: %s", __func__, strerror(errno));
Sharvil Nanavatic11b4072014-05-02 23:55:09 -0700100}
Sharvil Nanavatif4013f22014-07-05 20:42:07 -0700101
102int semaphore_get_fd(const semaphore_t *semaphore) {
103 assert(semaphore != NULL);
Sharvil Nanavati2cb29982014-08-01 17:00:12 -0700104 assert(semaphore->fd != INVALID_FD);
Sharvil Nanavatif4013f22014-07-05 20:42:07 -0700105 return semaphore->fd;
106}