|
|
|
@ -0,0 +1,141 @@
|
|
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2013-2019 Huawei Technologies Co., Ltd. All rights reserved.
|
|
|
|
|
* Copyright (c) 2020-2021 Huawei Device Co., Ltd. All rights reserved.
|
|
|
|
|
*
|
|
|
|
|
* Redistribution and use in source and binary forms, with or without modification,
|
|
|
|
|
* are permitted provided that the following conditions are met:
|
|
|
|
|
*
|
|
|
|
|
* 1. Redistributions of source code must retain the above copyright notice, this list of
|
|
|
|
|
* conditions and the following disclaimer.
|
|
|
|
|
*
|
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice, this list
|
|
|
|
|
* of conditions and the following disclaimer in the documentation and/or other materials
|
|
|
|
|
* provided with the distribution.
|
|
|
|
|
*
|
|
|
|
|
* 3. Neither the name of the copyright holder nor the names of its contributors may be used
|
|
|
|
|
* to endorse or promote products derived from this software without specific prior written
|
|
|
|
|
* permission.
|
|
|
|
|
*
|
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
|
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
|
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
|
|
|
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
|
|
|
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
|
|
|
|
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
|
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
|
|
|
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
|
|
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
*/
|
|
|
|
|
#include "it_pthread_test.h" // 引入测试头文件
|
|
|
|
|
|
|
|
|
|
// 全局变量定义
|
|
|
|
|
static pthread_mutex_t g_pthreadMuxTest1; // 互斥锁
|
|
|
|
|
static pthread_cond_t g_pthrdCondTest1; // 条件变量
|
|
|
|
|
static unsigned int g_pthreadExit = 0; // 控制线程退出的标志
|
|
|
|
|
|
|
|
|
|
// 线程函数1,模拟一个消费者
|
|
|
|
|
static void *PthreadF01(void *t)
|
|
|
|
|
{
|
|
|
|
|
int rc;
|
|
|
|
|
unsigned int count = 0;
|
|
|
|
|
const int testLoop = 2000; // 循环次数为2000
|
|
|
|
|
|
|
|
|
|
// 循环2000次进行条件变量等待和互斥锁操作
|
|
|
|
|
while (count < testLoop) {
|
|
|
|
|
// 加锁
|
|
|
|
|
rc = pthread_mutex_lock(&g_pthreadMuxTest1);
|
|
|
|
|
ICUNIT_GOTO_EQUAL(rc, 0, rc, EXIT); // 判断加锁是否成功,失败则跳转到EXIT
|
|
|
|
|
|
|
|
|
|
// 等待条件变量,释放互斥锁并进入阻塞状态,直到条件变量通知
|
|
|
|
|
rc = pthread_cond_wait(&g_pthrdCondTest1, &g_pthreadMuxTest1);
|
|
|
|
|
ICUNIT_GOTO_EQUAL(rc, 0, rc, EXIT);
|
|
|
|
|
|
|
|
|
|
// 解锁
|
|
|
|
|
rc = pthread_mutex_unlock(&g_pthreadMuxTest1);
|
|
|
|
|
ICUNIT_GOTO_EQUAL(rc, 0, rc, EXIT);
|
|
|
|
|
count++; // 增加计数器
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 设置退出标志
|
|
|
|
|
g_pthreadExit = 1;
|
|
|
|
|
|
|
|
|
|
EXIT:
|
|
|
|
|
return NULL; // 线程函数结束
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 线程函数2,模拟一个生产者
|
|
|
|
|
static void *PthreadF02(void *t)
|
|
|
|
|
{
|
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
|
|
// 当g_pthreadExit不为1时,继续循环发送信号
|
|
|
|
|
while (g_pthreadExit != 1) {
|
|
|
|
|
// 加锁
|
|
|
|
|
rc = pthread_mutex_lock(&g_pthreadMuxTest1);
|
|
|
|
|
ICUNIT_GOTO_EQUAL(rc, 0, rc, EXIT); // 判断加锁是否成功,失败则跳转到EXIT
|
|
|
|
|
|
|
|
|
|
// 发送信号,通知一个等待的线程
|
|
|
|
|
rc = pthread_cond_signal(&g_pthrdCondTest1);
|
|
|
|
|
ICUNIT_GOTO_EQUAL(rc, 0, rc, EXIT);
|
|
|
|
|
|
|
|
|
|
// 解锁
|
|
|
|
|
rc = pthread_mutex_unlock(&g_pthreadMuxTest1);
|
|
|
|
|
ICUNIT_GOTO_EQUAL(rc, 0, rc, EXIT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
EXIT:
|
|
|
|
|
pthread_exit(NULL); // 线程退出
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 测试用例
|
|
|
|
|
static unsigned int TestCase(void)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
long t1 = 1; // 线程1的标识
|
|
|
|
|
long t2 = 2; // 线程2的标识
|
|
|
|
|
int rc;
|
|
|
|
|
pthread_t threads[3]; // 创建3个线程
|
|
|
|
|
pthread_attr_t attr; // 线程属性
|
|
|
|
|
const int loopNum = 2; // 循环创建和等待线程次数
|
|
|
|
|
|
|
|
|
|
g_pthreadExit = 0; // 初始化退出标志
|
|
|
|
|
g_testCount = 0; // 初始化测试计数器
|
|
|
|
|
|
|
|
|
|
// 初始化互斥锁和条件变量
|
|
|
|
|
pthread_mutex_init(&g_pthreadMuxTest1, NULL);
|
|
|
|
|
pthread_cond_init(&g_pthrdCondTest1, NULL);
|
|
|
|
|
|
|
|
|
|
// 创建线程1 (消费者)
|
|
|
|
|
rc = pthread_create(&threads[0], NULL, PthreadF01, (void *)t1);
|
|
|
|
|
ICUNIT_ASSERT_EQUAL(rc, 0, rc); // 确认线程创建成功
|
|
|
|
|
|
|
|
|
|
// 创建线程2 (生产者)
|
|
|
|
|
rc = pthread_create(&threads[1], NULL, PthreadF02, (void *)t2);
|
|
|
|
|
ICUNIT_ASSERT_EQUAL(rc, 0, rc); // 确认线程创建成功
|
|
|
|
|
|
|
|
|
|
// 等待线程1和线程2执行完毕
|
|
|
|
|
for (i = 0; i < loopNum; i++) {
|
|
|
|
|
rc = pthread_join(threads[i], NULL);
|
|
|
|
|
ICUNIT_ASSERT_EQUAL(rc, 0, rc); // 确认线程结束
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 销毁线程属性、互斥锁和条件变量
|
|
|
|
|
rc = pthread_attr_destroy(&attr);
|
|
|
|
|
ICUNIT_ASSERT_EQUAL(rc, 0, rc); // 确认销毁线程属性成功
|
|
|
|
|
|
|
|
|
|
rc = pthread_mutex_destroy(&g_pthreadMuxTest1);
|
|
|
|
|
ICUNIT_ASSERT_EQUAL(rc, 0, rc); // 确认销毁互斥锁成功
|
|
|
|
|
|
|
|
|
|
rc = pthread_cond_destroy(&g_pthrdCondTest1);
|
|
|
|
|
ICUNIT_ASSERT_EQUAL(rc, 0, rc); // 确认销毁条件变量成功
|
|
|
|
|
|
|
|
|
|
return 0; // 返回0表示测试通过
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 测试入口函数
|
|
|
|
|
void ItTestPthreadCond002(void)
|
|
|
|
|
{
|
|
|
|
|
TEST_ADD_CASE("IT_POSIX_PTHREAD_COND_002", TestCase, TEST_POSIX, TEST_PTHREAD, TEST_LEVEL2, TEST_FUNCTION);
|
|
|
|
|
}
|