|
|
@ -33,11 +33,10 @@ |
|
|
|
****************************************************************************/ |
|
|
|
****************************************************************************/ |
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* @file px4_qurt_tasks.c |
|
|
|
* @file px4_linux_tasks.c |
|
|
|
* Implementation of existing task API for Linux |
|
|
|
* Implementation of existing task API for Linux |
|
|
|
*/ |
|
|
|
*/ |
|
|
|
|
|
|
|
|
|
|
|
#include <px4_debug.h> |
|
|
|
|
|
|
|
#include <unistd.h> |
|
|
|
#include <unistd.h> |
|
|
|
#include <stdio.h> |
|
|
|
#include <stdio.h> |
|
|
|
#include <stdlib.h> |
|
|
|
#include <stdlib.h> |
|
|
@ -61,10 +60,11 @@ |
|
|
|
#define PX4_MAX_TASKS 100 |
|
|
|
#define PX4_MAX_TASKS 100 |
|
|
|
struct task_entry |
|
|
|
struct task_entry |
|
|
|
{ |
|
|
|
{ |
|
|
|
pthread_t pid; |
|
|
|
int pid; |
|
|
|
std::string name; |
|
|
|
std::string name; |
|
|
|
bool isused; |
|
|
|
bool isused; |
|
|
|
task_entry() : isused(false) {} |
|
|
|
task_entry() : isused(false) {} |
|
|
|
|
|
|
|
void *sp; |
|
|
|
}; |
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
static task_entry taskmap[PX4_MAX_TASKS]; |
|
|
|
static task_entry taskmap[PX4_MAX_TASKS]; |
|
|
@ -77,28 +77,28 @@ typedef struct |
|
|
|
// strings are allocated after the
|
|
|
|
// strings are allocated after the
|
|
|
|
} pthdata_t; |
|
|
|
} pthdata_t; |
|
|
|
|
|
|
|
|
|
|
|
static void *entry_adapter ( void *ptr ) |
|
|
|
static void entry_adapter ( void *ptr ) |
|
|
|
{ |
|
|
|
{ |
|
|
|
PX4_DBG("entry_adapter\n"); |
|
|
|
pthdata_t *data = (pthdata_t *) ptr; |
|
|
|
pthdata_t *data; |
|
|
|
|
|
|
|
data = (pthdata_t *) ptr; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
data->entry(data->argc, data->argv); |
|
|
|
printf("TEST3\n"); |
|
|
|
free(ptr); |
|
|
|
#if 0 |
|
|
|
PX4_DBG("Before px4_task_exit"); |
|
|
|
//data->entry(data->argc, data->argv);
|
|
|
|
|
|
|
|
printf("TEST4\n"); |
|
|
|
|
|
|
|
printf("Before px4_task_exit\n"); |
|
|
|
px4_task_exit(0);
|
|
|
|
px4_task_exit(0);
|
|
|
|
PX4_DBG("After px4_task_exit"); |
|
|
|
//free(ptr);
|
|
|
|
|
|
|
|
printf("After px4_task_exit\n"); |
|
|
|
return NULL; |
|
|
|
#endif |
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void |
|
|
|
void |
|
|
|
px4_systemreset(bool to_bootloader) |
|
|
|
px4_systemreset(bool to_bootloader) |
|
|
|
{ |
|
|
|
{ |
|
|
|
PX4_WARN("Called px4_system_reset"); |
|
|
|
printf("Called px4_system_reset\n"); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
px4_task_t px4_task_spawn_cmd(const char *name, int scheduler, int priority, int stack_size, px4_main_t entry, char * const argv[]) |
|
|
|
px4_task_t px4_task_spawn_cmd(const char *name, int scheduler, int priority, int stack_size, px4_main_t entry, char * const *argv) |
|
|
|
{ |
|
|
|
{ |
|
|
|
int rv; |
|
|
|
int rv; |
|
|
|
int argc = 0; |
|
|
|
int argc = 0; |
|
|
@ -106,20 +106,21 @@ px4_task_t px4_task_spawn_cmd(const char *name, int scheduler, int priority, int |
|
|
|
unsigned int len = 0; |
|
|
|
unsigned int len = 0; |
|
|
|
unsigned long offset; |
|
|
|
unsigned long offset; |
|
|
|
unsigned long structsize; |
|
|
|
unsigned long structsize; |
|
|
|
char * p = (char *)argv; |
|
|
|
char * p; |
|
|
|
|
|
|
|
|
|
|
|
pthread_t task; |
|
|
|
|
|
|
|
pthread_attr_t attr; |
|
|
|
|
|
|
|
struct sched_param param; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//printf("arg %d %s %s %p\n", argc, argv[0], argv[1], argv[2]);
|
|
|
|
|
|
|
|
printf("arg %d %p\n", argc, argv); |
|
|
|
// Calculate argc
|
|
|
|
// Calculate argc
|
|
|
|
while (p != (char *)0) { |
|
|
|
if (argv) { |
|
|
|
|
|
|
|
for(;;) { |
|
|
|
p = argv[argc]; |
|
|
|
p = argv[argc]; |
|
|
|
if (p == (char *)0) |
|
|
|
if (p == (char *)0) |
|
|
|
break; |
|
|
|
break; |
|
|
|
|
|
|
|
printf("arg %d %s\n", argc, argv[argc]); |
|
|
|
++argc; |
|
|
|
++argc; |
|
|
|
len += strlen(p)+1; |
|
|
|
len += strlen(p)+1; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
structsize = sizeof(pthdata_t)+(argc+1)*sizeof(char *); |
|
|
|
structsize = sizeof(pthdata_t)+(argc+1)*sizeof(char *); |
|
|
|
pthdata_t *taskdata; |
|
|
|
pthdata_t *taskdata; |
|
|
|
|
|
|
|
|
|
|
@ -131,6 +132,7 @@ px4_task_t px4_task_spawn_cmd(const char *name, int scheduler, int priority, int |
|
|
|
taskdata->argc = argc; |
|
|
|
taskdata->argc = argc; |
|
|
|
|
|
|
|
|
|
|
|
for (i=0; i<argc; i++) { |
|
|
|
for (i=0; i<argc; i++) { |
|
|
|
|
|
|
|
printf("TEST\n"); |
|
|
|
printf("arg %d %s\n", i, argv[i]); |
|
|
|
printf("arg %d %s\n", i, argv[i]); |
|
|
|
taskdata->argv[i] = (char *)offset; |
|
|
|
taskdata->argv[i] = (char *)offset; |
|
|
|
strcpy((char *)offset, argv[i]); |
|
|
|
strcpy((char *)offset, argv[i]); |
|
|
@ -139,124 +141,38 @@ px4_task_t px4_task_spawn_cmd(const char *name, int scheduler, int priority, int |
|
|
|
// Must add NULL at end of argv
|
|
|
|
// Must add NULL at end of argv
|
|
|
|
taskdata->argv[argc] = (char *)0; |
|
|
|
taskdata->argv[argc] = (char *)0; |
|
|
|
|
|
|
|
|
|
|
|
rv = pthread_attr_init(&attr); |
|
|
|
|
|
|
|
if (rv != 0) { |
|
|
|
|
|
|
|
PX4_WARN("px4_task_spawn_cmd: failed to init thread attrs"); |
|
|
|
|
|
|
|
return (rv < 0) ? rv : -rv; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
#if 0 |
|
|
|
|
|
|
|
rv = pthread_attr_setinheritsched(&attr, PTHREAD_EXPLICIT_SCHED); |
|
|
|
|
|
|
|
if (rv != 0) { |
|
|
|
|
|
|
|
PX4_WARN("px4_task_spawn_cmd: failed to set inherit sched"); |
|
|
|
|
|
|
|
return (rv < 0) ? rv : -rv; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
rv = pthread_attr_setschedpolicy(&attr, scheduler); |
|
|
|
|
|
|
|
if (rv != 0) { |
|
|
|
|
|
|
|
PX4_WARN("px4_task_spawn_cmd: failed to set sched policy"); |
|
|
|
|
|
|
|
return (rv < 0) ? rv : -rv; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
param.sched_priority = priority; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rv = pthread_attr_setschedparam(&attr, ¶m); |
|
|
|
|
|
|
|
if (rv != 0) { |
|
|
|
|
|
|
|
PX4_WARN("px4_task_spawn_cmd: failed to set sched param"); |
|
|
|
|
|
|
|
return (rv < 0) ? rv : -rv; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rv = pthread_create (&task, &attr, &entry_adapter, (void *) taskdata); |
|
|
|
|
|
|
|
if (rv != 0) { |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (rv == EPERM) { |
|
|
|
|
|
|
|
//printf("WARNING: NOT RUNING AS ROOT, UNABLE TO RUN REALTIME THREADS\n");
|
|
|
|
|
|
|
|
rv = pthread_create (&task, NULL, &entry_adapter, (void *) taskdata); |
|
|
|
|
|
|
|
if (rv != 0) { |
|
|
|
|
|
|
|
PX4_ERR("px4_task_spawn_cmd: failed to create thread %d %d\n", rv, errno); |
|
|
|
|
|
|
|
return (rv < 0) ? rv : -rv; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
else { |
|
|
|
|
|
|
|
return (rv < 0) ? rv : -rv; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for (i=0; i<PX4_MAX_TASKS; ++i) { |
|
|
|
for (i=0; i<PX4_MAX_TASKS; ++i) { |
|
|
|
if (taskmap[i].isused == false) { |
|
|
|
if (taskmap[i].isused == false) { |
|
|
|
taskmap[i].pid = task; |
|
|
|
taskmap[i].pid = i+1; |
|
|
|
taskmap[i].name = name; |
|
|
|
taskmap[i].name = name; |
|
|
|
taskmap[i].isused = true; |
|
|
|
taskmap[i].isused = true; |
|
|
|
//taskmap[i].sp = malloc(stack_size);
|
|
|
|
taskmap[i].sp = malloc(stack_size); |
|
|
|
break; |
|
|
|
break; |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
if (i>=PX4_MAX_TASKS) { |
|
|
|
printf("TEST2\n"); |
|
|
|
return -ENOSPC; |
|
|
|
thread_create(entry_adapter, taskmap[i].sp, i+1, (void *) taskdata); |
|
|
|
} |
|
|
|
|
|
|
|
return i; |
|
|
|
return i+1; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
int px4_task_delete(px4_task_t id) |
|
|
|
int px4_task_delete(px4_task_t id) |
|
|
|
{ |
|
|
|
{ |
|
|
|
int rv = 0; |
|
|
|
printf("Called px4_task_delete\n"); |
|
|
|
pthread_t pid; |
|
|
|
|
|
|
|
PX4_WARN("Called px4_task_delete"); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (id < PX4_MAX_TASKS && taskmap[id].isused) |
|
|
|
|
|
|
|
pid = taskmap[id].pid; |
|
|
|
|
|
|
|
else |
|
|
|
|
|
|
|
return -EINVAL; |
|
|
|
return -EINVAL; |
|
|
|
|
|
|
|
|
|
|
|
// If current thread then exit, otherwise cancel
|
|
|
|
|
|
|
|
if (pthread_self() == pid) { |
|
|
|
|
|
|
|
taskmap[id].isused = false; |
|
|
|
|
|
|
|
pthread_exit(0); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
rv = pthread_cancel(pid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
taskmap[id].isused = false; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return rv; |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void px4_task_exit(int ret) |
|
|
|
void px4_task_exit(int ret) |
|
|
|
{ |
|
|
|
{ |
|
|
|
int i;
|
|
|
|
thread_stop(); |
|
|
|
pthread_t pid = pthread_self(); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Get pthread ID from the opaque ID
|
|
|
|
// Free stack
|
|
|
|
for (i=0; i<PX4_MAX_TASKS; ++i) { |
|
|
|
|
|
|
|
if (taskmap[i].pid == pid) { |
|
|
|
|
|
|
|
taskmap[i].isused = false; |
|
|
|
|
|
|
|
break; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if (i>=PX4_MAX_TASKS) { |
|
|
|
|
|
|
|
PX4_ERR("px4_task_exit: self task not found!"); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
else { |
|
|
|
|
|
|
|
PX4_DBG("px4_task_exit: %s", taskmap[i].name.c_str()); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
pthread_exit((void *)(unsigned long)ret); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
int px4_task_kill(px4_task_t id, int sig) |
|
|
|
int px4_task_kill(px4_task_t id, int sig) |
|
|
|
{ |
|
|
|
{ |
|
|
|
int rv = 0; |
|
|
|
printf("Called px4_task_kill\n"); |
|
|
|
pthread_t pid; |
|
|
|
|
|
|
|
PX4_DBG("Called px4_task_kill %d", sig); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (id < PX4_MAX_TASKS && taskmap[id].pid != 0) |
|
|
|
|
|
|
|
pid = taskmap[id].pid; |
|
|
|
|
|
|
|
else |
|
|
|
|
|
|
|
return -EINVAL; |
|
|
|
return -EINVAL; |
|
|
|
|
|
|
|
|
|
|
|
// If current thread then exit, otherwise cancel
|
|
|
|
|
|
|
|
rv = pthread_kill(pid, sig); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return rv; |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void px4_show_tasks() |
|
|
|
void px4_show_tasks() |
|
|
@ -264,16 +180,17 @@ void px4_show_tasks() |
|
|
|
int idx; |
|
|
|
int idx; |
|
|
|
int count = 0; |
|
|
|
int count = 0; |
|
|
|
|
|
|
|
|
|
|
|
PX4_INFO("Active Tasks:"); |
|
|
|
printf("Active Tasks:\n"); |
|
|
|
for (idx=0; idx < PX4_MAX_TASKS; idx++) |
|
|
|
for (idx=0; idx < PX4_MAX_TASKS; idx++) |
|
|
|
{ |
|
|
|
{ |
|
|
|
if (taskmap[idx].isused) { |
|
|
|
if (taskmap[idx].isused) { |
|
|
|
PX4_INFO(" %-10s", taskmap[idx].name.c_str()); |
|
|
|
printf(" %-10s %d\n", taskmap[idx].name.c_str(), taskmap[idx].pid); |
|
|
|
count++; |
|
|
|
count++; |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
if (count == 0) |
|
|
|
if (count == 0) |
|
|
|
PX4_INFO(" No running tasks"); |
|
|
|
printf(" No running tasks\n"); |
|
|
|
|
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// STUBS
|
|
|
|
// STUBS
|
|
|
|