mirror of
				https://github.com/open5gs/open5gs.git
				synced 2025-11-03 21:43:25 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			143 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			143 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (C) 2019 by Sukchan Lee <acetcom@gmail.com>
 | 
						|
 *
 | 
						|
 * This file is part of Open5GS.
 | 
						|
 *
 | 
						|
 * This program is free software: you can redistribute it and/or modify
 | 
						|
 * it under the terms of the GNU Affero General Public License as published by
 | 
						|
 * the Free Software Foundation, either version 3 of the License, or
 | 
						|
 * (at your option) any later version.
 | 
						|
 *
 | 
						|
 * This program is distributed in the hope that it will be useful,
 | 
						|
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
 * GNU General Public License for more details.
 | 
						|
 *
 | 
						|
 * You should have received a copy of the GNU General Public License
 | 
						|
 * along with this program.  If not, see <https://www.gnu.org/licenses/>.
 | 
						|
 */
 | 
						|
 | 
						|
#include "ogs-core.h"
 | 
						|
 | 
						|
#undef OGS_LOG_DOMAIN
 | 
						|
#define OGS_LOG_DOMAIN __ogs_thread_domain
 | 
						|
 | 
						|
#if !defined(_WIN32)
 | 
						|
#define ogs_thread_id_t pthread_t
 | 
						|
#define ogs_thread_join(_n) pthread_join((_n), NULL)
 | 
						|
#else
 | 
						|
#define ogs_thread_id_t HANDLE
 | 
						|
static ogs_inline DWORD ogs_thread_join(ogs_thread_id_t thread)
 | 
						|
{
 | 
						|
    DWORD ret = WaitForSingleObject(thread, INFINITE);
 | 
						|
    if (!CloseHandle (thread)) {
 | 
						|
        ogs_log_message(
 | 
						|
            OGS_LOG_ERROR, ogs_errno, "Couldn't close thread handle");
 | 
						|
    }
 | 
						|
 | 
						|
    return ret;
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
typedef struct ogs_thread_s {
 | 
						|
    ogs_thread_id_t id;
 | 
						|
 | 
						|
    ogs_thread_mutex_t mutex;
 | 
						|
    ogs_thread_cond_t cond;
 | 
						|
 | 
						|
    bool running;
 | 
						|
 | 
						|
    void (*func)(void *);
 | 
						|
    void *data;
 | 
						|
} ogs_thread_t;
 | 
						|
 | 
						|
static void *thread_worker(void *arg)
 | 
						|
{
 | 
						|
    ogs_thread_t *thread = arg;
 | 
						|
    ogs_assert(thread);
 | 
						|
 | 
						|
    ogs_thread_mutex_lock(&thread->mutex);
 | 
						|
 | 
						|
    thread->running = true;
 | 
						|
    ogs_thread_cond_signal(&thread->cond);
 | 
						|
 | 
						|
    ogs_thread_mutex_unlock(&thread->mutex);
 | 
						|
 | 
						|
    ogs_debug("[%p] worker signal", thread);
 | 
						|
    thread->func(thread->data);
 | 
						|
 | 
						|
    ogs_thread_mutex_lock(&thread->mutex);
 | 
						|
    thread->running = false;
 | 
						|
    ogs_thread_mutex_unlock(&thread->mutex);
 | 
						|
    ogs_debug("[%p] worker done", thread);
 | 
						|
 | 
						|
    return NULL;
 | 
						|
}
 | 
						|
 | 
						|
ogs_thread_t *ogs_thread_create(void (*func)(void *), void *data)
 | 
						|
{
 | 
						|
    ogs_thread_t *thread = ogs_calloc(1, sizeof *thread);
 | 
						|
    if (!thread) {
 | 
						|
        ogs_error("ogs_calloc() failed");
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
 | 
						|
    ogs_thread_mutex_init(&thread->mutex);
 | 
						|
    ogs_thread_cond_init(&thread->cond);
 | 
						|
 | 
						|
    ogs_thread_mutex_lock(&thread->mutex);
 | 
						|
 | 
						|
    thread->running = false;
 | 
						|
 | 
						|
    thread->func = func;
 | 
						|
    thread->data = data;
 | 
						|
 | 
						|
#if !defined(_WIN32)
 | 
						|
    pthread_create(&thread->id, NULL, thread_worker, thread);
 | 
						|
#else
 | 
						|
    thread->id = CreateThread(NULL, 0,
 | 
						|
                    (LPTHREAD_START_ROUTINE)thread_worker, thread, 0, NULL);
 | 
						|
#endif
 | 
						|
 | 
						|
    ogs_thread_cond_wait(&thread->cond, &thread->mutex);
 | 
						|
    ogs_thread_mutex_unlock(&thread->mutex);
 | 
						|
    ogs_debug("[%p] thread started", thread);
 | 
						|
 | 
						|
    return thread;
 | 
						|
}
 | 
						|
 | 
						|
void ogs_thread_destroy(ogs_thread_t *thread)
 | 
						|
{
 | 
						|
    const ogs_time_t deadline = ogs_get_monotonic_time() + 5 * 1000 * 1000;
 | 
						|
    ogs_assert(thread);
 | 
						|
 | 
						|
    ogs_debug("[%p] thread running(%d)", thread, thread->running);
 | 
						|
    while(ogs_get_monotonic_time() <= deadline) {
 | 
						|
        /* wait 5 seconds */
 | 
						|
        ogs_thread_mutex_lock(&thread->mutex);
 | 
						|
        if (!thread->running) {
 | 
						|
            ogs_thread_mutex_unlock(&thread->mutex);
 | 
						|
            break;
 | 
						|
        }
 | 
						|
        ogs_thread_mutex_unlock(&thread->mutex);
 | 
						|
        ogs_usleep(1000);
 | 
						|
    }
 | 
						|
 | 
						|
    ogs_debug("[%p] thread destroy", thread);
 | 
						|
    ogs_thread_mutex_lock(&thread->mutex);
 | 
						|
    if (thread->running) {
 | 
						|
        ogs_fatal("thread still running after 3 seconds");
 | 
						|
        ogs_assert_if_reached();
 | 
						|
    }
 | 
						|
    ogs_thread_mutex_unlock(&thread->mutex);
 | 
						|
 | 
						|
    ogs_thread_join(thread->id);
 | 
						|
    ogs_debug("[%p] thread join", thread);
 | 
						|
 | 
						|
    ogs_thread_cond_destroy(&thread->cond);
 | 
						|
    ogs_thread_mutex_destroy(&thread->mutex);
 | 
						|
 | 
						|
    ogs_free(thread);
 | 
						|
    ogs_debug("[%p] thread done", thread);
 | 
						|
}
 |