nuttx-apps/testing/ostest/pthread_exit.c
fangxinyong ba075747be ostest: Add initial support for CONFIG_BUILD_KERNEL
task_* APIs are unavailable in kernel build mode, replacing with
posix_spawn or pthread_* API to pass the case. But posix_spawn requires
some dependency, for example CONFIG_BUILTIN and CONFIG_LIBC_EXECFUNCS,
so pthread_* APIs are used in most scenarios. Some tests should be
re-visited because the intent is to user another task (in this case
another process) to e.g. receive signals.
That will require quite a bit of extra work.

Tests that had to be disabled:
- restart: task_restart() does not work at all with kernel mode so it is
  disabled entirely
- fpu: make sure the FPU test is not even attempted, because it will cause
  ASSERT() and stop the test
- vfork: vfork() does not work for some reason in CONFIG_BUILD_KERNEL,
  there is something missing on the kernel side, so just disable the test for now

Tests that should be re-visited:
- The signal tests, now they signal the process itself while before the
  signal was sent to another task. This will require building the part
  that receives the signal as a separate process
- waitpid: Like stated above, waitpid does not work for pthreads
- suspend: kill to send signal does not work for pthreads

Signed-off-by: fangxinyong <fangxinyong@xiaomi.com>
Co-authored-by: Ville Juven <ville.juven@unikie.com>
2023-12-27 10:01:05 -08:00

126 lines
3.7 KiB
C

/****************************************************************************
* apps/testing/ostest/pthread_exit.c
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you 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.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <sys/wait.h>
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
#include <unistd.h>
#include <errno.h>
#include "ostest.h"
#ifdef CONFIG_SCHED_WAITPID
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
#define PRIORITY 100
/****************************************************************************
* Private Functions
****************************************************************************/
static FAR void *pthread_exit_thread(FAR void *parameter)
{
unsigned me = (unsigned)pthread_self();
printf("pthread_exit_thread %u: Sleeping for 10 second\n", me);
sleep(5);
printf("pthread_exit_thread %u: Still running...\n", me);
sleep(5);
printf("pthread_exit_thread %u: Exiting\n", me);
return NULL;
}
static FAR void *pthread_exit_main(FAR void *arg)
{
pthread_t child;
#ifdef SDCC
pthread_attr_t attr;
#endif
unsigned me = (unsigned)pthread_self();
int ret;
printf("pthread_exit_main %u: Starting pthread_exit_thread\n", me);
#ifdef SDCC
pthread_attr_init(&attr);
ret = pthread_create(&child, &attr, pthread_exit_thread, NULL);
#else
ret = pthread_create(&child, NULL, pthread_exit_thread, NULL);
#endif
if (ret != 0)
{
printf("Thread creation failed, return code %d", ret);
}
printf("pthread_exit_main %u: Sleeping for 5 seconds\n", me);
fflush(stdout);
sleep(5);
printf("pthread_exit_main %u: Calling pthread_exit()\n", me);
pthread_exit(NULL);
printf("pthread_exit_main %u: ERROR: Still running\n", me);
exit(0);
return NULL;
}
/****************************************************************************
* Public Functions
****************************************************************************/
void pthread_exit_test(void)
{
struct sched_param param;
pthread_attr_t attr;
pid_t pid;
int ret;
pthread_attr_init(&attr);
param.sched_priority = PRIORITY;
pthread_attr_setschedparam(&attr, &param);
pthread_attr_setstacksize(&attr, STACKSIZE);
ret = pthread_create(&pid, &attr, pthread_exit_main, NULL);
if (ret < 0)
{
printf("pthread_exit_test: ERROR pthread_create Failed\n");
}
else
{
printf("pthread_exit_test: Started pthread_exit_main at PID=%d\n",
pid);
if (pthread_join(pid, NULL) != 0)
{
printf("pthread_exit_test: ERROR Failed to join to terminate\n");
ASSERT(false);
};
}
}
#endif /* CONFIG_SCHED_WAITPID */