c - C 预处理器中指向 void 的指针

标签 c gcc goto pointer-arithmetic

我阅读了这个来源 ( https://github.com/lattera/glibc/blob/master/stdio-common/vfprintf.c ) 并找到了一些我不完全理解的有趣的行:

#ifdef SHARED
/* 'int' is enough and it saves some space on 64 bit systems.  */
# define JUMP_TABLE_TYPE const int
# define JUMP_TABLE_BASE_LABEL do_form_unknown
# define REF(Name) &&do_##Name - &&JUMP_TABLE_BASE_LABEL
# define JUMP(ChExpr, table)                              \
  do                                      \
{                                     \
  int offset;                                 \
  void *ptr;                                  \
  spec = (ChExpr);                            \
  offset = NOT_IN_JUMP_RANGE (spec) ? REF (form_unknown)          \
    : table[CHAR_CLASS (spec)];                       \
  ptr = &&JUMP_TABLE_BASE_LABEL + offset;                 \
  goto *ptr;                                  \
}                                     \
  while (0)

 ...

#define STEP0_3_TABLE                                 \
/* Step 0: at the beginning.  */                          \
static JUMP_TABLE_TYPE step0_jumps[30] =                      \
{                                         \
  REF (form_unknown),                             \
  REF (flag_space),     /* for ' ' */                     \
  REF (flag_plus),      /* for '+' */                     \
  REF (flag_minus),     /* for '-' */                     \
  REF (flag_hash),      /* for '<hash>' */                \
  REF (flag_zero),      /* for '0' */                     \
  REF (flag_quote),     /* for '\'' */                    \
  REF (width_asterics), /* for '*' */                     \
  REF (width),      /* for '1'...'9' */               \
  REF (precision),      /* for '.' */                     \
  REF (mod_half),       /* for 'h' */                     \
  ...

我写了一个简单的例子,明白这一行 &&do_##Namedo_##Name 转换为指向 void 的指针。但我不明白在这种情况下指针算法是如何工作的:或者写一些指向 Internet 资源的链接,我可以在其中阅读有关此技术的信息。

最佳答案

大概是为了保证具有线性复杂性,代码使用了一个由用作值的标签组成的跳转表。

Labels-as-values是一个 GNU C 扩展,它允许您使用 && 获取标签的地址。输入地址 void *,然后您可以使用 goto *address; 跳转到它。

基本标签的小变化是,代码不是在表中存储绝对标签,而是存储 do_uknown_form 标签的偏移量。

这可以节省表中的空间(偏移量可以是 4 字节 int 而不是 8 字节指针)并有助于为共享库生成更好的代码(因此 #ifdef SHARED)甚至static const 当代码加载到可重定位共享库中时,需要修补绝对标签的跳转表,但偏移量保持不变,因此不需要修补,表可以存储在只读存储器。

该技术在 How to Write Shared Libraries 中有所描述。 Ulrich Drepper 的文章。

关于c - C 预处理器中指向 void 的指针,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56609886/

相关文章:

c - 这个定义语句是什么意思?

c - 使用 Unix C 将数据从父级传递给子级并返回

c - 使用__libc_init_array调用STM32

c++ - SFINAE for cast 运算符

java - 要求 Java 返回到特定语句

lua - Lua中双冒号的作用是什么?

c - 使用strcpy & buffer overflow执行shell代码

c - 将值添加到 char* [][]

c++ - <random> 在 Linux 中生成相同的数字,但在 Windows 中不生成

python - GoTo(基本)程序