SMP
This commit is contained in:
parent
45bc5a721f
commit
5f83c51347
|
@ -1,3 +1,13 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2006-2024 RT-Thread Development Team
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: Apache-2.0
|
||||||
|
*
|
||||||
|
* Change Logs:
|
||||||
|
* Date Author Notes
|
||||||
|
* 2024/9/12 zhujiale the first version
|
||||||
|
*/
|
||||||
|
|
||||||
#include "smp.h"
|
#include "smp.h"
|
||||||
|
|
||||||
#define DBG_TAG "SMP"
|
#define DBG_TAG "SMP"
|
||||||
|
@ -5,18 +15,18 @@
|
||||||
#include <rtdbg.h>
|
#include <rtdbg.h>
|
||||||
|
|
||||||
struct smp_call global_work[RT_CPUS_NR];
|
struct smp_call global_work[RT_CPUS_NR];
|
||||||
|
rt_atomic_t wait;
|
||||||
rt_err_t smp_call_handler(struct smp_event *event)
|
rt_err_t smp_call_handler(struct smp_event *event)
|
||||||
{
|
{
|
||||||
switch(event->event_id)
|
switch(event->event_id)
|
||||||
{
|
{
|
||||||
case SMP_CALL_EVENT_FUNC:
|
case SMP_CALL_EVENT_FUNC:
|
||||||
event->func(event->data);
|
event->func(event->data);
|
||||||
|
rt_atomic_add(&wait,1);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
rt_kprintf("error event id\n");
|
LOG_E("error event id\n");
|
||||||
return -RT_ERROR;
|
return RT_ERROR;
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
return RT_EOK;
|
return RT_EOK;
|
||||||
}
|
}
|
||||||
|
@ -40,17 +50,33 @@ void rt_smp_call_ipi_handler(int vector, void *param)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void rt_smp_call_func_cond(int cpu_mask, smp_func func, void *data)
|
/**
|
||||||
|
* @brief call function on specified CPU ,
|
||||||
|
*
|
||||||
|
* @param cpu_mask cpu mask for call
|
||||||
|
* @param func the function pointer
|
||||||
|
* @param data the data pointer
|
||||||
|
* @param flag call flag if you set SMP_CALL_WAIT_ALL
|
||||||
|
* then it will wait all cpu call finish and return
|
||||||
|
* else it will call function on specified CPU and return immediately
|
||||||
|
* @param cond the condition function pointer,if you set it then it will call function only when cond return true
|
||||||
|
*/
|
||||||
|
void rt_smp_call_func_cond(int cpu_mask, smp_call_func_back func, void *data,rt_uint8_t flag,smp_cond cond)
|
||||||
{
|
{
|
||||||
RT_DEBUG_NOT_IN_INTERRUPT;
|
RT_DEBUG_NOT_IN_INTERRUPT;
|
||||||
struct smp_call work;
|
|
||||||
struct smp_event event;
|
struct smp_event event;
|
||||||
rt_bool_t need_call = RT_TRUE;
|
rt_bool_t need_call = RT_TRUE,need_wait = RT_FALSE;
|
||||||
int cur_cpu = rt_hw_cpu_id();
|
int cur_cpu = rt_hw_cpu_id();
|
||||||
int cpuid = 1 << cur_cpu;
|
int cpuid = 1 << cur_cpu;
|
||||||
int tmp_id = 0;
|
int tmp_id = 0,cpu_nr = 0;
|
||||||
int tmp_mask;
|
int tmp_mask;
|
||||||
|
|
||||||
|
if(flag == SMP_CALL_WAIT_ALL)
|
||||||
|
{
|
||||||
|
need_wait = RT_TRUE;
|
||||||
|
rt_atomic_store(&wait,0);
|
||||||
|
}
|
||||||
|
|
||||||
if(cpuid & cpu_mask)
|
if(cpuid & cpu_mask)
|
||||||
{
|
{
|
||||||
func(data);
|
func(data);
|
||||||
|
@ -67,6 +93,9 @@ void rt_smp_call_func_cond(int cpu_mask, smp_func func, void *data)
|
||||||
{
|
{
|
||||||
if((tmp_mask & 1) && (tmp_id < RT_CPUS_NR))
|
if((tmp_mask & 1) && (tmp_id < RT_CPUS_NR))
|
||||||
{
|
{
|
||||||
|
if(cond && !cond(tmp_id,data))
|
||||||
|
continue;
|
||||||
|
cpu_nr++;
|
||||||
event.event_id = SMP_CALL_EVENT_FUNC;
|
event.event_id = SMP_CALL_EVENT_FUNC;
|
||||||
event.func = func;
|
event.func = func;
|
||||||
event.data = data;
|
event.data = data;
|
||||||
|
@ -80,11 +109,34 @@ void rt_smp_call_func_cond(int cpu_mask, smp_func func, void *data)
|
||||||
}
|
}
|
||||||
rt_hw_ipi_send(RT_IPI_FUNC, cpu_mask);
|
rt_hw_ipi_send(RT_IPI_FUNC, cpu_mask);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(need_wait)
|
||||||
|
{
|
||||||
|
while(rt_atomic_load(&wait) != cpu_nr);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
struct rt_spinlock lock_1;
|
|
||||||
|
void rt_call_each_cpu(smp_call_func_back func, void *data,rt_uint8_t flag)
|
||||||
|
{
|
||||||
|
rt_smp_call_func_cond(RT_ALL_CPU,func,data,flag,RT_NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
void rt_call_each_cpu_cond(smp_call_func_back func, void *data,rt_uint8_t flag,smp_cond cond_func)
|
||||||
|
{
|
||||||
|
rt_smp_call_func_cond(RT_ALL_CPU,func,data,flag,cond_func);
|
||||||
|
}
|
||||||
|
void rt_call_any_cpu(int cpu_mask,smp_call_func_back func, void *data,rt_uint8_t flag)
|
||||||
|
{
|
||||||
|
rt_smp_call_func_cond(cpu_mask,func,data,flag,RT_NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
void rt_call_any_cpu_cond(int cpu_mask,smp_call_func_back func, void *data,rt_uint8_t flag,smp_cond cond_func)
|
||||||
|
{
|
||||||
|
rt_smp_call_func_cond(cpu_mask,func,data,flag,cond_func);
|
||||||
|
}
|
||||||
|
|
||||||
void smp_init(void)
|
void smp_init(void)
|
||||||
{
|
{
|
||||||
rt_spin_lock_init(&lock_1);
|
|
||||||
for(int i = 0; i < RT_CPUS_NR; i++)
|
for(int i = 0; i < RT_CPUS_NR; i++)
|
||||||
{
|
{
|
||||||
rt_memset(&global_work[i],0,sizeof(struct smp_call));
|
rt_memset(&global_work[i],0,sizeof(struct smp_call));
|
||||||
|
|
|
@ -1,28 +1,34 @@
|
||||||
#ifndef __SMP_IPI_H__
|
#ifndef __SMP_IPI_H__
|
||||||
#define __SMP_IPI_H__
|
#define __SMP_IPI_H__
|
||||||
#include <rtthread.h>
|
#include <rtthread.h>
|
||||||
typedef void (*smp_func)(void *data);
|
typedef void (*smp_call_func_back)(void *data);
|
||||||
|
typedef rt_bool_t (*smp_cond)(int cpu, void *info);
|
||||||
|
|
||||||
#define SMP_CALL_EVENT_FUNC 0x1
|
#define SMP_CALL_EVENT_FUNC 0x1
|
||||||
|
|
||||||
|
#define SMP_CALL_WAIT_ALL (1 << 0)
|
||||||
|
#define SMP_CALL_NO_WAIT (1 << 1)
|
||||||
|
|
||||||
|
#define RT_ALL_CPU ((1 << RT_CPUS_NR) - 1)
|
||||||
struct smp_event
|
struct smp_event
|
||||||
{
|
{
|
||||||
int cpu_mask;
|
int cpu_mask;
|
||||||
int event_id;
|
int event_id;
|
||||||
void *data;
|
void *data;
|
||||||
smp_func func;
|
smp_call_func_back func;
|
||||||
|
|
||||||
};
|
};
|
||||||
struct smp_call
|
struct smp_call
|
||||||
{
|
{
|
||||||
struct rt_spinlock lock;
|
struct rt_spinlock lock;
|
||||||
struct smp_event event;
|
struct smp_event event;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
void rt_smp_call_func_cond(int cpu_mask,smp_func func, void *data);
|
|
||||||
void rt_smp_call_ipi_handler(int vector, void *param);
|
void rt_smp_call_ipi_handler(int vector, void *param);
|
||||||
|
void rt_call_each_cpu(smp_call_func_back func, void *data,rt_uint8_t flag);
|
||||||
|
void rt_call_each_cpu_cond(smp_call_func_back func, void *data,rt_uint8_t flag,smp_cond cond_func);
|
||||||
|
void rt_call_any_cpu(int cpu_mask,smp_call_func_back func, void *data,rt_uint8_t flag);
|
||||||
|
void rt_call_any_cpu_cond(int cpu_mask,smp_call_func_back func, void *data,rt_uint8_t flag,smp_cond cond_func);
|
||||||
void smp_init(void);
|
void smp_init(void);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -2,39 +2,58 @@
|
||||||
#include "utest.h"
|
#include "utest.h"
|
||||||
#include "utest_assert.h"
|
#include "utest_assert.h"
|
||||||
#include "smp.h"
|
#include "smp.h"
|
||||||
int pass_count = 0;
|
int pass_count = 0;
|
||||||
int pass = 1000;
|
int pass = 1000;
|
||||||
struct rt_spinlock lock;
|
struct rt_spinlock lock;
|
||||||
|
|
||||||
void test_call(void *data)
|
void test_call(void *data)
|
||||||
{
|
{
|
||||||
rt_spin_lock(&lock);
|
rt_spin_lock(&lock);
|
||||||
int *i = (int *)data;
|
int *i = (int *)data;
|
||||||
int id = rt_hw_cpu_id();
|
int id = rt_hw_cpu_id();
|
||||||
*i &= ~(1 << id);
|
*i &= ~(1 << id);
|
||||||
if(*i == 0)
|
if (*i == 0)
|
||||||
pass_count++;
|
pass_count++;
|
||||||
rt_spin_unlock(&lock);
|
rt_spin_unlock(&lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void test()
|
void test1()
|
||||||
{
|
{
|
||||||
int cpu_mask = 0xf;
|
int cpu_mask = 0xf;
|
||||||
for(int i =0 ;i < 1000 ;i++)
|
for (int i = 0; i < 1000; i++)
|
||||||
{
|
{
|
||||||
cpu_mask = rand()% 0xf;
|
cpu_mask = rand() % 0xf;
|
||||||
if (cpu_mask == 0)
|
if (cpu_mask == 0)
|
||||||
pass--;
|
pass--;
|
||||||
rt_smp_call_func_cond(cpu_mask,test_call, &cpu_mask);
|
rt_call_each_cpu(test_call, &cpu_mask, SMP_CALL_NO_WAIT);
|
||||||
if(i % 20 == 0)
|
if (i % 20 == 0)
|
||||||
rt_kprintf("#");
|
rt_kprintf("#");
|
||||||
rt_thread_mdelay(1);
|
rt_thread_mdelay(1);
|
||||||
}
|
}
|
||||||
rt_kprintf("\n");
|
rt_kprintf("\n");
|
||||||
|
uassert_true(pass_count == pass);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void test_call2(void *data)
|
||||||
|
{
|
||||||
|
rt_spin_lock(&lock);
|
||||||
|
int a = 100000;
|
||||||
|
while (a--);
|
||||||
|
int *i = (int *)data;
|
||||||
|
(*i)++;
|
||||||
|
rt_spin_unlock(&lock);
|
||||||
|
}
|
||||||
|
void test2(void)
|
||||||
|
{
|
||||||
|
int data = 0;
|
||||||
|
rt_call_each_cpu(test_call2, &data, SMP_CALL_WAIT_ALL);
|
||||||
|
uassert_true(data == RT_CPUS_NR);
|
||||||
|
rt_thread_mdelay(10);
|
||||||
|
data = 0;
|
||||||
|
rt_call_each_cpu(test_call2, &data, SMP_CALL_NO_WAIT);
|
||||||
|
uassert_true(data != RT_CPUS_NR);
|
||||||
|
}
|
||||||
|
|
||||||
static rt_err_t utest_tc_init(void)
|
static rt_err_t utest_tc_init(void)
|
||||||
{
|
{
|
||||||
|
@ -44,12 +63,12 @@ static rt_err_t utest_tc_init(void)
|
||||||
|
|
||||||
static rt_err_t utest_tc_cleanup(void)
|
static rt_err_t utest_tc_cleanup(void)
|
||||||
{
|
{
|
||||||
uassert_true(pass_count == pass);
|
|
||||||
return RT_EOK;
|
return RT_EOK;
|
||||||
}
|
}
|
||||||
static void testcase(void)
|
static void testcase(void)
|
||||||
{
|
{
|
||||||
UTEST_UNIT_RUN(test);
|
UTEST_UNIT_RUN(test1);
|
||||||
|
UTEST_UNIT_RUN(test2);
|
||||||
}
|
}
|
||||||
|
|
||||||
UTEST_TC_EXPORT(testcase, "testcase.smp.smp", utest_tc_init, utest_tc_cleanup, 10);
|
UTEST_TC_EXPORT(testcase, "testcase.smp.smp", utest_tc_init, utest_tc_cleanup, 10);
|
||||||
|
|
Loading…
Reference in New Issue