Какова цель операции Shift в REGSITER_TM_CLONES?
Глядя на ассемблерный код, я не совсем понимаю, что означает сдвигать операции в register_tm_clones. Какова роль сменной операции?
Сравнивая код сборки с исходным кодом, я подумал, что понял назначение SUB RAX,6020F8H. Операция связана с размером в исходном коде. Поскольку значение SIZE равно нулю, я предположил, что код сборки является реализацией (TMC_END - TMC_LIST). Во время отладки я обнаружил, что FOR может быть выполнен только один раз или даже пропущен size++ для непосредственного выполнения оператора IF.
исходный код:
register_tm_clones (void)
{
void (*fn) (void *, size_t);
size_t size;
#ifdef HAVE_GAS_HIDDEN
size = (__TMC_END__ - __TMC_LIST__) / 2;
#else
for (size = 0; __TMC_LIST__[size * 2] != NULL; size++)
continue;
#endif
if (size == 0)
return;
fn = _ITM_registerTMCloneTable;
__asm ("" : "+r" (fn));
if (fn)
fn (__TMC_LIST__, size);
}
#endif /* USE_TM_CLONE_REGISTRY */
код сборки:
register_tm_clones proc near
mov eax, 6020F8h ;.bss
push rbp
sub rax, 6020F8h ;.bss
sar rax, 3
mov rbp, rsp ; return address
mov rdx, rax
shr rdx, 3Fh
add rax, rdx
sar rax, 1
jnz short loc_400C24
loc_400C22:
pop rbp
retn
loc_400C24:
mov edx, 0
test rdx, rdx
jz short loc_400C22
pop rbp
mov rsi, rax
mov edi, 6020F8h
jmp rdx
register_tm_clones endp
Я хотел бы знать роль этого кода сборки для конкретной операции. Конечно, главное все же захотеть узнать цель операции смены здесь. Если я смогу это понять, мне будет легче решить другие проблемы. (Возможно, у меня были некоторые ошибки в моем предыдущем понимании кода, и я надеюсь, что вы можете указать на это)
1 ответ
for
нет в ассемблерном коде. Он никогда не видится компилятором, потому что он удаляется препроцессором, если HAVE_GAS_HIDDEN
устанавливается, что случается здесь.
sar rax, 3
связано с арифметикой указателя, так как размер элементов в __TMC_LIST__
составляет 8 байтов, будучи 2^3.
Два других сдвига - это подписанное деление на два. Он реализован таким образом, чтобы следовать правилам округления отрицательных чисел. Конечно, это не нужно, но плохой компилятор не знает, что __TMC_END__ - __TMC_LIST__ >= 0
,