1570af302Sopenharmony_ci/*
2570af302Sopenharmony_ci * Copyright (c) 2022 Huawei Device Co., Ltd.
3570af302Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
4570af302Sopenharmony_ci * you may not use this file except in compliance with the License.
5570af302Sopenharmony_ci * You may obtain a copy of the License at
6570af302Sopenharmony_ci *
7570af302Sopenharmony_ci *   http://www.apache.org/licenses/LICENSE-2.0
8570af302Sopenharmony_ci *
9570af302Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software
10570af302Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
11570af302Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12570af302Sopenharmony_ci * See the License for the specific language governing permissions and
13570af302Sopenharmony_ci * limitations under the License.
14570af302Sopenharmony_ci */
15570af302Sopenharmony_ci
16570af302Sopenharmony_ci#include <stdbool.h>
17570af302Sopenharmony_ci#include <threads.h>
18570af302Sopenharmony_ci#include "test.h"
19570af302Sopenharmony_ci
20570af302Sopenharmony_cistatic cnd_t cond;
21570af302Sopenharmony_cistatic mtx_t mutex;
22570af302Sopenharmony_cistatic unsigned int waiting_threads;
23570af302Sopenharmony_ci
24570af302Sopenharmony_ci#define N 3
25570af302Sopenharmony_ci
26570af302Sopenharmony_cistatic int child_wait(void *data)
27570af302Sopenharmony_ci{
28570af302Sopenharmony_ci    mtx_lock(&mutex);
29570af302Sopenharmony_ci    ++waiting_threads;
30570af302Sopenharmony_ci    cnd_wait(&cond, &mutex);
31570af302Sopenharmony_ci    mtx_unlock(&mutex);
32570af302Sopenharmony_ci
33570af302Sopenharmony_ci    thrd_exit(thrd_success);
34570af302Sopenharmony_ci}
35570af302Sopenharmony_ci
36570af302Sopenharmony_ci/**
37570af302Sopenharmony_ci * @tc.name      : cnd_broadcast_0100
38570af302Sopenharmony_ci * @tc.desc      :
39570af302Sopenharmony_ci * @tc.level     : Level 0
40570af302Sopenharmony_ci */
41570af302Sopenharmony_civoid cnd_broadcast_0100(void)
42570af302Sopenharmony_ci{
43570af302Sopenharmony_ci    thrd_t ids[N];
44570af302Sopenharmony_ci    unsigned char i;
45570af302Sopenharmony_ci
46570af302Sopenharmony_ci    if (cnd_init(&cond) != thrd_success) {
47570af302Sopenharmony_ci        t_error("%s cnd_init failed\n", __func__);
48570af302Sopenharmony_ci    }
49570af302Sopenharmony_ci
50570af302Sopenharmony_ci    if (mtx_init(&mutex, mtx_plain) != thrd_success) {
51570af302Sopenharmony_ci        t_error("%s mtx_init failed\n", __func__);
52570af302Sopenharmony_ci    }
53570af302Sopenharmony_ci
54570af302Sopenharmony_ci    for (i = 0; i < N; ++i) {
55570af302Sopenharmony_ci        if (thrd_create(&ids[i], child_wait, NULL) != thrd_success) {
56570af302Sopenharmony_ci            t_error("%s thrd_create failed\n", __func__);
57570af302Sopenharmony_ci        }
58570af302Sopenharmony_ci    }
59570af302Sopenharmony_ci
60570af302Sopenharmony_ci    while (true) {
61570af302Sopenharmony_ci        mtx_lock(&mutex);
62570af302Sopenharmony_ci        if (waiting_threads > N) {
63570af302Sopenharmony_ci            t_error("%s ", __func__);
64570af302Sopenharmony_ci        }
65570af302Sopenharmony_ci        bool done_waiting = waiting_threads == N;
66570af302Sopenharmony_ci        mtx_unlock(&mutex);
67570af302Sopenharmony_ci        if (done_waiting) {
68570af302Sopenharmony_ci            break;
69570af302Sopenharmony_ci        }
70570af302Sopenharmony_ci        thrd_sleep(&((struct timespec){.tv_nsec = 100 * 1000 * 1000}), NULL);
71570af302Sopenharmony_ci    }
72570af302Sopenharmony_ci
73570af302Sopenharmony_ci    mtx_lock(&mutex);
74570af302Sopenharmony_ci    if (cnd_broadcast(&cond) != thrd_success) {
75570af302Sopenharmony_ci        t_error("%s cnd_broadcast failed\n", __func__);
76570af302Sopenharmony_ci    }
77570af302Sopenharmony_ci    mtx_unlock(&mutex);
78570af302Sopenharmony_ci
79570af302Sopenharmony_ci    for (i = 0; i < N; ++i) {
80570af302Sopenharmony_ci        if (thrd_join(ids[i], NULL) != thrd_success) {
81570af302Sopenharmony_ci            t_error("%s thrd_join failed\n", __func__);
82570af302Sopenharmony_ci        }
83570af302Sopenharmony_ci    }
84570af302Sopenharmony_ci
85570af302Sopenharmony_ci    mtx_destroy(&mutex);
86570af302Sopenharmony_ci    cnd_destroy(&cond);
87570af302Sopenharmony_ci}
88570af302Sopenharmony_ci
89570af302Sopenharmony_ciint main(int argc, char *argv[])
90570af302Sopenharmony_ci{
91570af302Sopenharmony_ci    cnd_broadcast_0100();
92570af302Sopenharmony_ci    return t_status;
93570af302Sopenharmony_ci}