Если у нас есть:
__int32 some_var = 0;
Каков наилучший (если есть) способ вызова InterlockedExchange
, InterlockedIncrement
и других взаимосвязанных функций, для которых требуется LONG*
для some_var
?
Поскольку существует гарантия того, что LONG
является 32-битным в любой Windows, возможно, просто пропустить (long*) some_var
.Тем не менее, мне это кажется довольно уродливым, и я не могу найти подтверждение, что это безопасно.
Обратите внимание, я не могу изменить тип на long
, потому что он не переносимый.Мне нужен ровно 32-битный тип.
Обновление: Некоторые исследования библиотек, которые предоставляют переносимые атомарные операции, показали, что никто не заботится о приведении типов.Некоторые примеры:
Apache Portable Runtime (APR) :
typedef WINBASEAPI apr_uint32_t (WINAPI * apr_atomic_win32_ptr_val_fn)
(apr_uint32_t volatile *,
apr_uint32_t);
APR_DECLARE(apr_uint32_t) apr_atomic_add32(volatile apr_uint32_t *mem, apr_uint32_t val)
{
#if (defined(_M_IA64) || defined(_M_AMD64))
return InterlockedExchangeAdd(mem, val);
#elif defined(__MINGW32__)
return InterlockedExchangeAdd((long *)mem, val);
#else
return ((apr_atomic_win32_ptr_val_fn)InterlockedExchangeAdd)(mem, val);
#endif
}
atomic_ops :
AO_INLINE AO_t
AO_fetch_and_sub1_full (volatile AO_t *p)
{
return _InterlockedDecrement64((LONGLONG volatile *)p) + 1;
}