foundationdb/flow/ThreadPrimitives.cpp

111 lines
2.8 KiB
C++

/*
* ThreadPrimitives.cpp
*
* This source file is part of the FoundationDB open source project
*
* Copyright 2013-2018 Apple Inc. and the FoundationDB project authors
*
* 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.
*/
#include "flow/ThreadPrimitives.h"
#include "flow/Trace.h"
#include <stdint.h>
#include <iostream>
#include <errno.h>
#include <string.h>
#include <stdio.h>
#ifdef _WIN32
#include <windows.h>
#undef min
#undef max
#endif
extern std::string format( const char *form, ... );
Event::Event() {
#ifdef _WIN32
ev = CreateEvent(nullptr, FALSE, FALSE, nullptr);
#elif defined(__linux__) || defined(__FreeBSD__)
int result = sem_init(&sem, 0, 0);
if (result)
criticalError(FDB_EXIT_INIT_SEMAPHORE, "UnableToInitializeSemaphore", format("Could not initialize semaphore - %s", strerror(errno)).c_str());
#elif defined(__APPLE__)
self = mach_task_self();
kern_return_t ret = semaphore_create(self, &sem, SYNC_POLICY_FIFO, 0);
if(ret != KERN_SUCCESS)
criticalError(FDB_EXIT_INIT_SEMAPHORE, "UnableToInitializeSemaphore", format("Could not initialize semaphore - %s", strerror(errno)).c_str());
#else
#error Port me!
#endif
}
Event::~Event() {
#ifdef _WIN32
CloseHandle(ev);
#elif defined(__linux__) || defined(__FreeBSD__)
sem_destroy(&sem);
#elif defined(__APPLE__)
semaphore_destroy(self, sem);
#else
#error Port me!
#endif
}
void Event::set() {
#ifdef _WIN32
SetEvent(ev);
#elif defined(__linux__) || defined(__FreeBSD__)
sem_post(&sem);
#elif defined(__APPLE__)
semaphore_signal(sem);
#else
#error Port me!
#endif
}
void Event::block() {
#ifdef _WIN32
WaitForSingleObject(ev, INFINITE);
#elif defined(__linux__) || defined(__FreeBSD__)
int ret;
do {
ret = sem_wait(&sem);
} while (ret != 0 && errno == EINTR);
#elif defined(__APPLE__)
kern_return_t ret;
do {
ret = semaphore_wait(sem);
} while ((ret != KERN_SUCCESS) && (ret != KERN_TERMINATED));
#else
#error Port me!
#endif
}
// Mutex::impl is allocated from the heap so that its size doesn't need to be in the header. Is there a better way?
Mutex::Mutex() {
impl = new CRITICAL_SECTION;
InitializeCriticalSection((CRITICAL_SECTION*)impl);
}
Mutex::~Mutex() {
DeleteCriticalSection((CRITICAL_SECTION*)impl);
delete (CRITICAL_SECTION*)impl;
}
void Mutex::enter() {
EnterCriticalSection((CRITICAL_SECTION*)impl);
}
void Mutex::leave() {
LeaveCriticalSection((CRITICAL_SECTION*)impl);
}