|
@@ -0,0 +1,216 @@
|
|
|
+/*
|
|
|
+ * Copyright (c) 2006-2021, RT-Thread Development Team
|
|
|
+ *
|
|
|
+ * SPDX-License-Identifier: Apache-2.0
|
|
|
+ *
|
|
|
+ * Change Logs:
|
|
|
+ * Date Author Notes
|
|
|
+ * 2021-06-04 WillianChan first version
|
|
|
+ * 2021-06-08 WillianChan support to MS VC++ compiler
|
|
|
+ */
|
|
|
+
|
|
|
+#include <var_export.h>
|
|
|
+
|
|
|
+static const ve_exporter_t *ve_exporter_table = RT_NULL;
|
|
|
+static rt_size_t ve_exporter_num = 0;
|
|
|
+
|
|
|
+/* for IAR compiler */
|
|
|
+#if defined(__ICCARM__) || defined(__ICCRX__)
|
|
|
+#pragma section="VarExpTab"
|
|
|
+#endif
|
|
|
+
|
|
|
+/* for ARM C and IAR Compiler */
|
|
|
+#if defined(__CC_ARM) || defined(__CLANG_ARM) || defined (__ICCARM__) || defined(__ICCRX__)
|
|
|
+static RT_USED const struct ve_exporter __ve_table_start
|
|
|
+RT_SECTION("0.""VarExpTab") = {"ve_start", "ve_start", 0};
|
|
|
+
|
|
|
+static RT_USED const struct ve_exporter __ve_table_end
|
|
|
+RT_SECTION("2.""VarExpTab") = {"ve_end", "ve_end", 2};
|
|
|
+#endif
|
|
|
+
|
|
|
+/* for MS VC++ compiler */
|
|
|
+#if defined(_MSC_VER)
|
|
|
+static ve_exporter_t ve_exporter_tab[2048];
|
|
|
+
|
|
|
+#pragma section("VarExpTab$a", read)
|
|
|
+__declspec(allocate("VarExpTab$a"))
|
|
|
+RT_USED const struct ve_exporter __ve_table_start = { "ve_start", "ve_start", 0};
|
|
|
+
|
|
|
+#pragma section("VarExpTab$z", read)
|
|
|
+__declspec(allocate("VarExpTab$z"))
|
|
|
+RT_USED const struct ve_exporter __ve_table_end = { "ve_end", "ve_end", 2};
|
|
|
+#endif
|
|
|
+
|
|
|
+/* initialize var export */
|
|
|
+int var_export_init(void)
|
|
|
+{
|
|
|
+ /* initialize the var export table.*/
|
|
|
+#if defined(__CC_ARM) || defined(__CLANG_ARM) /* for ARM C Compiler */
|
|
|
+ ve_exporter_table = &__ve_table_start + 1;
|
|
|
+ ve_exporter_num = &__ve_table_end - &__ve_table_start;
|
|
|
+#elif defined (__GNUC__) /* for GCC Compiler */
|
|
|
+ extern const int __ve_table_start;
|
|
|
+ extern const int __ve_table_end;
|
|
|
+ ve_exporter_table = (const ve_exporter_t *)&__ve_table_start;
|
|
|
+ ve_exporter_num = (const ve_exporter_t *)&__ve_table_end - ve_exporter_table;
|
|
|
+#elif defined (__ICCARM__) || defined(__ICCRX__) /* for IAR Compiler */
|
|
|
+ ve_exporter_table = &__ve_table_start + 1;
|
|
|
+ ve_exporter_num = &__ve_table_end - &__ve_table_start - 1;
|
|
|
+#elif defined (_MSC_VER) /* for MS VC++ compiler */
|
|
|
+ unsigned int *ptr_begin = (unsigned int *)&__ve_table_start;
|
|
|
+ unsigned int *ptr_end = (unsigned int *)&__ve_table_end;
|
|
|
+
|
|
|
+ /* past the three members in first ptr_begin */
|
|
|
+ ptr_begin += (sizeof(struct ve_exporter) / sizeof(unsigned int));
|
|
|
+ while (*ptr_begin == 0) ptr_begin++;
|
|
|
+ do ptr_end--; while (*ptr_end == 0);
|
|
|
+
|
|
|
+ ve_exporter_table = (const ve_exporter_t *)ptr_begin;
|
|
|
+ ve_exporter_num = (ptr_end - ptr_begin) / (sizeof(struct ve_exporter) / sizeof(unsigned int)) + 1;
|
|
|
+
|
|
|
+ int index_i, index_j, index_min;
|
|
|
+
|
|
|
+ for (index_i = 0; index_i < ve_exporter_num; index_i++)
|
|
|
+ {
|
|
|
+ ve_exporter_tab[index_i] = ve_exporter_table[index_i];
|
|
|
+ }
|
|
|
+
|
|
|
+ for (index_i = 0; index_i < (ve_exporter_num - 1); index_i++)
|
|
|
+ {
|
|
|
+ index_min = index_i;
|
|
|
+
|
|
|
+ for (index_j = index_i + 1; index_j < ve_exporter_num; index_j++)
|
|
|
+ {
|
|
|
+ if (rt_strcmp(ve_exporter_tab[index_j].module, ve_exporter_tab[index_min].module) < 0)
|
|
|
+ {
|
|
|
+ index_min = index_j;
|
|
|
+ ve_exporter_t ve_exporter_temp[1] = {ve_exporter_tab[index_min]};
|
|
|
+ ve_exporter_tab[index_min] = ve_exporter_tab[index_i];
|
|
|
+ ve_exporter_tab[index_i] = ve_exporter_temp[0];
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+#endif
|
|
|
+
|
|
|
+ return ve_exporter_num;
|
|
|
+}
|
|
|
+INIT_PREV_EXPORT(var_export_init);
|
|
|
+
|
|
|
+/* initialize module */
|
|
|
+int ve_module_init(ve_module_t *mod, const char *module)
|
|
|
+{
|
|
|
+#if defined(_MSC_VER)
|
|
|
+ const ve_exporter_t *exporter = ve_exporter_tab;
|
|
|
+#else
|
|
|
+ const ve_exporter_t *exporter = ve_exporter_table;
|
|
|
+#endif
|
|
|
+ rt_bool_t first_exist = RT_FALSE;
|
|
|
+ rt_size_t found_index;
|
|
|
+
|
|
|
+ for (found_index = 0; found_index < ve_exporter_num; found_index++)
|
|
|
+ {
|
|
|
+ if (!rt_strcmp(exporter->module, module))
|
|
|
+ {
|
|
|
+ if (first_exist == RT_FALSE)
|
|
|
+ {
|
|
|
+ mod->begin = exporter;
|
|
|
+ first_exist = RT_TRUE;
|
|
|
+ }
|
|
|
+ mod->end = exporter;
|
|
|
+ }
|
|
|
+ exporter++;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (first_exist == RT_FALSE)
|
|
|
+ {
|
|
|
+ return -RT_ERROR;
|
|
|
+ }
|
|
|
+
|
|
|
+ return RT_EOK;
|
|
|
+}
|
|
|
+
|
|
|
+/* initialize iterator */
|
|
|
+void ve_iter_init(ve_module_t *mod, ve_iterator_t *iter)
|
|
|
+{
|
|
|
+ if (iter)
|
|
|
+ {
|
|
|
+ iter->exp_index = mod->begin;
|
|
|
+ iter->exp_end = mod->end;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/* iterate backward */
|
|
|
+const ve_exporter_t *ve_iter_next(ve_iterator_t *iter)
|
|
|
+{
|
|
|
+ if (iter->exp_index <= iter->exp_end)
|
|
|
+ {
|
|
|
+ return iter->exp_index++;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ return RT_NULL;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/* binary search based on identifier */
|
|
|
+static const ve_exporter_t *ve_binary_search(ve_module_t *mod, const char *identifier)
|
|
|
+{
|
|
|
+#if defined(_MSC_VER)
|
|
|
+ rt_size_t ve_low_num = mod->begin - ve_exporter_tab;
|
|
|
+ rt_size_t ve_high_num = mod->end - ve_exporter_tab;
|
|
|
+#else
|
|
|
+ rt_size_t ve_low_num = mod->begin - ve_exporter_table;
|
|
|
+ rt_size_t ve_high_num = mod->end - ve_exporter_table;
|
|
|
+#endif
|
|
|
+ rt_size_t ve_mid_num;
|
|
|
+ int strcmp_rst;
|
|
|
+
|
|
|
+ while (ve_low_num <= ve_high_num)
|
|
|
+ {
|
|
|
+ ve_mid_num = (ve_high_num - ve_low_num) / 2;
|
|
|
+ strcmp_rst = rt_strcmp(mod->begin[ve_mid_num].identifier, identifier);
|
|
|
+
|
|
|
+ if (strcmp_rst == 0)
|
|
|
+ {
|
|
|
+ return &mod->begin[ve_mid_num];
|
|
|
+ }
|
|
|
+ else if (strcmp_rst > 0)
|
|
|
+ {
|
|
|
+ ve_high_num = ve_mid_num + 1;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ ve_low_num = ve_mid_num - 1;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ return RT_NULL;
|
|
|
+}
|
|
|
+
|
|
|
+/* get the value by identifier */
|
|
|
+rt_base_t ve_value_get(ve_module_t *mod, const char *identifier)
|
|
|
+{
|
|
|
+ const ve_exporter_t *exporter = ve_binary_search(mod, identifier);
|
|
|
+
|
|
|
+ if (exporter)
|
|
|
+ {
|
|
|
+ return exporter->value;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ return VE_NOT_FOUND;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/* check if this value exists in the module*/
|
|
|
+rt_bool_t ve_value_exist(ve_module_t *mod, const char *identifier)
|
|
|
+{
|
|
|
+ if (ve_binary_search(mod, identifier))
|
|
|
+ {
|
|
|
+ return RT_TRUE;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ return RT_FALSE;
|
|
|
+ }
|
|
|
+}
|