zhujiale 10 months ago
parent
commit
d1865d0d93

+ 39 - 55
components/smp/smp.c

@@ -1,7 +1,12 @@
 #include "smp.h"
 #include "smp.h"
-struct smp_call **global_work;
 
 
-rt_err_t smp_call_handler(struct smp_event * event)
+#define DBG_TAG "SMP"
+#define DBG_LVL DBG_INFO
+#include <rtdbg.h>
+
+struct smp_call global_work[RT_CPUS_NR];
+
+rt_err_t smp_call_handler(struct smp_event *event)
 {
 {
     switch(event->event_id)
     switch(event->event_id)
     {
     {
@@ -18,64 +23,57 @@ rt_err_t smp_call_handler(struct smp_event * event)
 void rt_smp_call_ipi_handler(int vector, void *param)
 void rt_smp_call_ipi_handler(int vector, void *param)
 {
 {
     int err;
     int err;
-    struct smp_call *work,*tmp;
     int cur_cpu = rt_hw_cpu_id();
     int cur_cpu = rt_hw_cpu_id();
-    rt_spin_lock(&global_work[cur_cpu]->lock);
-    rt_list_for_each_entry_safe(work,tmp,&global_work[cur_cpu]->node,node)
+    rt_spin_lock(&global_work[cur_cpu].lock);
+    if(global_work[cur_cpu].event.event_id)
     {
     {
-        if(work->event)
+        err = smp_call_handler(&global_work[cur_cpu].event);
+        if(err)
         {
         {
-            err = smp_call_handler(work->event);
-            if(err)
-                break;
-            rt_list_remove(&work->node);
-            rt_free(work);
+            LOG_E("Have no event\n");
+            rt_memset(&global_work[cur_cpu].event,0,sizeof(struct smp_event));
+            rt_spin_unlock(&global_work[cur_cpu].lock);
         }
         }
-            
+        rt_memset(&global_work[cur_cpu].event,0,sizeof(struct smp_event));
     }
     }
-    rt_spin_unlock(&global_work[cur_cpu]->lock);
+    rt_spin_unlock(&global_work[cur_cpu].lock);
     
     
 }
 }
 
 
 void rt_smp_call_func_cond(int cpu_mask, smp_func func, void *data)
 void rt_smp_call_func_cond(int cpu_mask, smp_func func, void *data)
 {
 {
-    rt_bool_t run_cur_cpu = RT_TRUE;
+    RT_DEBUG_NOT_IN_INTERRUPT;
+    struct smp_call work;
+    struct smp_event event;
     rt_bool_t need_call = RT_TRUE;
     rt_bool_t need_call = RT_TRUE;
     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;
-    int tmp_mask = cpu_mask;
+    int tmp_mask;
 
 
-    if(cpuid & ~cpu_mask)
-        run_cur_cpu = RT_FALSE;
-
-    if(run_cur_cpu)
+    if(cpuid & cpu_mask)
+    {
         func(data);
         func(data);
-
-    if(!(cpu_mask & cpuid))
-        need_call = RT_FALSE;
-    else
         cpu_mask = cpu_mask & (~cpuid);
         cpu_mask = cpu_mask & (~cpuid);
+    }
+
+    if(!cpu_mask)
+        need_call = RT_FALSE;   
 
 
+    tmp_mask = cpu_mask;
     if(need_call)
     if(need_call)
     {
     {
         while(tmp_mask)
         while(tmp_mask)
         {
         {
-            if((tmp_mask & 1) && tmp_id < RT_CPUS_NR)
+            if((tmp_mask & 1) && (tmp_id < RT_CPUS_NR))
             {
             {
-                struct smp_event *event = rt_calloc(1, sizeof(struct smp_event));
-                event->event_id = SMP_CALL_EVENT_FUNC;
-                event->func = func;
-                event->data = data;
-                event->cpu_mask = cpu_mask;
-                struct smp_call *work = rt_calloc(1, sizeof(struct smp_call));
-                if(work == RT_NULL)
-                    break;
-                
-                work->event = event;
-                rt_spin_lock(&global_work[tmp_id]->lock);
-                rt_list_insert_before(&global_work[tmp_id]->node, &work->node);
-                rt_spin_unlock(&global_work[tmp_id]->lock);
+                event.event_id = SMP_CALL_EVENT_FUNC;
+                event.func = func;
+                event.data = data;
+                event.cpu_mask = cpu_mask;
+                rt_spin_lock(&global_work[tmp_id].lock);
+                global_work[tmp_id].event = event;
+                rt_spin_unlock(&global_work[tmp_id].lock);
             }
             }
             tmp_id++;
             tmp_id++;
             tmp_mask = tmp_mask >> 1;
             tmp_mask = tmp_mask >> 1;
@@ -83,28 +81,14 @@ 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);
     }
     }
 }
 }
-
+struct rt_spinlock lock_1;
 void smp_init(void)
 void smp_init(void)
 {
 {
-    struct smp_call **work_list = (struct smp_call **)rt_malloc(sizeof(struct smp_call *));
+    rt_spin_lock_init(&lock_1);
     for(int i = 0; i < RT_CPUS_NR; i++)
     for(int i = 0; i < RT_CPUS_NR; i++)
     {        
     {        
-        work_list[i] = rt_calloc(1, sizeof(struct smp_call));
-        if(work_list[i] == RT_NULL)
-            break;
-        rt_list_init(&work_list[i]->node);
-        rt_spin_lock_init(&work_list[i]->lock);
+        rt_memset(&global_work[i],0,sizeof(struct smp_call));
+        rt_spin_lock_init(&global_work[i].lock);
     }        
     }        
-    global_work = work_list;
-}
-
-void test_call(void *data)
-{
-    rt_kprintf("call cpu id  = %d \n",rt_hw_cpu_id());
 }
 }
-void test()
-{
-    int cpu_mask = 0xf;
-    rt_smp_call_func_cond(cpu_mask,test_call, RT_NULL);
 
 
-}

+ 8 - 9
components/smp/smp.h

@@ -5,14 +5,6 @@ typedef void (*smp_func)(void *data);
 
 
 #define SMP_CALL_EVENT_FUNC 0x1
 #define SMP_CALL_EVENT_FUNC 0x1
 
 
-struct smp_call
-{
-    struct rt_spinlock lock;
-    struct rt_list_node node;
-    struct smp_event *event;
-
-};
-
 struct smp_event
 struct smp_event
 {
 {
     int cpu_mask;
     int cpu_mask;
@@ -21,8 +13,15 @@ struct smp_event
     smp_func func;
     smp_func func;
 
 
 };
 };
+struct smp_call
+{
+    struct rt_spinlock lock;
+    struct rt_list_node node;
+    struct smp_event event;
+
+};
+
 
 
-void test();
 void rt_smp_call_func_cond(int cpu_mask,smp_func func, void *data);
 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 smp_init(void);
 void smp_init(void);

+ 1 - 0
examples/utest/testcases/Kconfig

@@ -15,6 +15,7 @@ rsource "drivers/ipc/Kconfig"
 rsource "posix/Kconfig"
 rsource "posix/Kconfig"
 rsource "mm/Kconfig"
 rsource "mm/Kconfig"
 rsource "tmpfs/Kconfig"
 rsource "tmpfs/Kconfig"
+rsource "smp/Kconfig"
 endif
 endif
 
 
 endmenu
 endmenu

+ 6 - 0
examples/utest/testcases/smp/Kconfig

@@ -0,0 +1,6 @@
+menu "SMP Testcase"
+
+config UTEST_SMP_CALL_FUNC
+    bool "Call random cpu to run func"
+    default n
+endmenu

+ 13 - 0
examples/utest/testcases/smp/SConscript

@@ -0,0 +1,13 @@
+Import('rtconfig')
+from building import *
+
+cwd     = GetCurrentDir()
+src     = []
+CPPPATH = [cwd]
+
+if GetDepend(['RT_USING_SMP','UTEST_SMP_CALL_FUNC']):
+    src += ['smp.c']
+
+group = DefineGroup('utestcases', src, depend = ['RT_USING_UTESTCASES'], CPPPATH = CPPPATH)
+
+Return('group')

+ 55 - 0
examples/utest/testcases/smp/smp.c

@@ -0,0 +1,55 @@
+#include <rtdevice.h>
+#include "utest.h"
+#include "utest_assert.h"
+#include "smp.h"
+int pass_count = 0;
+int pass = 1000;
+struct rt_spinlock lock;
+
+void test_call(void *data)
+{
+    rt_spin_lock(&lock);
+    int *i = (int *)data;
+    int id = rt_hw_cpu_id();
+    *i &= ~(1 << id);
+    if(*i == 0)
+        pass_count++;
+    rt_spin_unlock(&lock);
+}
+
+
+void test()
+{
+    int cpu_mask = 0xf;
+    for(int i =0 ;i < 1000 ;i++)
+    {
+        cpu_mask = rand()% 0xf;
+        if (cpu_mask == 0) 
+        pass--;
+        rt_smp_call_func_cond(cpu_mask,test_call, &cpu_mask);
+        if(i % 20 == 0)
+            rt_kprintf("#");
+        rt_thread_mdelay(1);
+    }    
+    rt_kprintf("\n");
+}
+
+
+
+static rt_err_t utest_tc_init(void)
+{
+    rt_spin_lock_init(&lock);
+    return RT_EOK;
+}
+
+static rt_err_t utest_tc_cleanup(void)
+{
+    uassert_true(pass_count == pass);
+    return RT_EOK;
+}
+static void testcase(void)
+{
+    UTEST_UNIT_RUN(test);
+}
+
+UTEST_TC_EXPORT(testcase, "testcase.smp.smp", utest_tc_init, utest_tc_cleanup, 10);