os2threads.h 4.68 KB
Newer Older
1
/*
2
 * Copyright (c) 2011-2017 KO Myung-Hun <komh@chollian.net>
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
 *
 * This file is part of FFmpeg.
 *
 * FFmpeg is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * FFmpeg 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with FFmpeg; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 */

/**
 * @file
 * os2threads to pthreads wrapper
 */

26 27
#ifndef COMPAT_OS2THREADS_H
#define COMPAT_OS2THREADS_H
28 29 30 31 32 33 34

#define INCL_DOS
#include <os2.h>

#undef __STRICT_ANSI__          /* for _beginthread() */
#include <stdlib.h>

35
#include <sys/builtin.h>
36 37
#include <sys/fmutex.h>

38 39 40 41 42 43 44 45
#include "libavutil/attributes.h"

typedef struct {
    TID tid;
    void *(*start_routine)(void *);
    void *arg;
    void *result;
} pthread_t;
46

47 48
typedef void pthread_attr_t;

49
typedef _fmutex pthread_mutex_t;
50 51
typedef void pthread_mutexattr_t;

52 53
#define PTHREAD_MUTEX_INITIALIZER _FMUTEX_INITIALIZER

54
typedef struct {
55 56 57
    HEV event_sem;
    HEV ack_sem;
    volatile unsigned  wait_count;
58 59 60 61
} pthread_cond_t;

typedef void pthread_condattr_t;

62 63 64 65 66 67 68
typedef struct {
    volatile int done;
    _fmutex mtx;
} pthread_once_t;

#define PTHREAD_ONCE_INIT {0, _FMUTEX_INITIALIZER}

69 70
static void thread_entry(void *arg)
{
71
    pthread_t *thread = arg;
72

73
    thread->result = thread->start_routine(thread->arg);
74 75
}

76 77 78 79
static av_always_inline int pthread_create(pthread_t *thread,
                                           const pthread_attr_t *attr,
                                           void *(*start_routine)(void*),
                                           void *arg)
80
{
81 82 83
    thread->start_routine = start_routine;
    thread->arg = arg;
    thread->result = NULL;
84

85
    thread->tid = _beginthread(thread_entry, NULL, 1024 * 1024, thread);
86 87 88 89 90 91

    return 0;
}

static av_always_inline int pthread_join(pthread_t thread, void **value_ptr)
{
92 93 94 95
    DosWaitThread(&thread.tid, DCWW_WAIT);

    if (value_ptr)
        *value_ptr = thread.result;
96 97 98 99

    return 0;
}

100 101
static av_always_inline int pthread_mutex_init(pthread_mutex_t *mutex,
                                               const pthread_mutexattr_t *attr)
102
{
103
    _fmutex_create(mutex, 0);
104 105 106 107 108 109

    return 0;
}

static av_always_inline int pthread_mutex_destroy(pthread_mutex_t *mutex)
{
110
    _fmutex_close(mutex);
111 112 113 114 115 116

    return 0;
}

static av_always_inline int pthread_mutex_lock(pthread_mutex_t *mutex)
{
117
    _fmutex_request(mutex, 0);
118 119 120 121 122 123

    return 0;
}

static av_always_inline int pthread_mutex_unlock(pthread_mutex_t *mutex)
{
124
    _fmutex_release(mutex);
125 126 127 128

    return 0;
}

129 130
static av_always_inline int pthread_cond_init(pthread_cond_t *cond,
                                              const pthread_condattr_t *attr)
131 132
{
    DosCreateEventSem(NULL, &cond->event_sem, DCE_POSTONE, FALSE);
133
    DosCreateEventSem(NULL, &cond->ack_sem, DCE_POSTONE, FALSE);
134 135 136 137 138 139 140 141 142

    cond->wait_count = 0;

    return 0;
}

static av_always_inline int pthread_cond_destroy(pthread_cond_t *cond)
{
    DosCloseEventSem(cond->event_sem);
143
    DosCloseEventSem(cond->ack_sem);
144 145 146 147 148 149

    return 0;
}

static av_always_inline int pthread_cond_signal(pthread_cond_t *cond)
{
150
    if (!__atomic_cmpxchg32(&cond->wait_count, 0, 0)) {
151
        DosPostEventSem(cond->event_sem);
152
        DosWaitEventSem(cond->ack_sem, SEM_INDEFINITE_WAIT);
153 154 155 156 157 158 159
    }

    return 0;
}

static av_always_inline int pthread_cond_broadcast(pthread_cond_t *cond)
{
160 161
    while (!__atomic_cmpxchg32(&cond->wait_count, 0, 0))
        pthread_cond_signal(cond);
162 163 164 165

    return 0;
}

166 167
static av_always_inline int pthread_cond_wait(pthread_cond_t *cond,
                                              pthread_mutex_t *mutex)
168
{
169
    __atomic_increment(&cond->wait_count);
170 171 172 173 174

    pthread_mutex_unlock(mutex);

    DosWaitEventSem(cond->event_sem, SEM_INDEFINITE_WAIT);

175 176 177 178
    __atomic_decrement(&cond->wait_count);

    DosPostEventSem(cond->ack_sem);

179 180 181 182 183
    pthread_mutex_lock(mutex);

    return 0;
}

184 185
static av_always_inline int pthread_once(pthread_once_t *once_control,
                                         void (*init_routine)(void))
186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202
{
    if (!once_control->done)
    {
        _fmutex_request(&once_control->mtx, 0);

        if (!once_control->done)
        {
            init_routine();

            once_control->done = 1;
        }

        _fmutex_release(&once_control->mtx);
    }

    return 0;
}
203
#endif /* COMPAT_OS2THREADS_H */