2016-01-08 20:49:23 +00:00
|
|
|
#ifndef GUARD_GBA_MACRO_H
|
|
|
|
#define GUARD_GBA_MACRO_H
|
|
|
|
|
2023-07-24 19:04:23 +01:00
|
|
|
#define CPU_FILL_UNCHECKED(value, dest, size, bit) \
|
2016-10-31 08:14:22 +00:00
|
|
|
{ \
|
2016-01-08 20:49:23 +00:00
|
|
|
vu##bit tmp = (vu##bit)(value); \
|
|
|
|
CpuSet((void *)&tmp, \
|
|
|
|
dest, \
|
|
|
|
CPU_SET_##bit##BIT | CPU_SET_SRC_FIXED | ((size)/(bit/8) & 0x1FFFFF)); \
|
2016-10-31 08:14:22 +00:00
|
|
|
}
|
2016-01-08 20:49:23 +00:00
|
|
|
|
2023-07-24 19:04:23 +01:00
|
|
|
#if MODERN
|
|
|
|
#define CPU_FILL(value, dest, size, bit) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
_Static_assert(_Alignof(dest) >= (bit / 8), "destination potentially unaligned"); \
|
|
|
|
CPU_FILL_UNCHECKED(value, dest, size, bit); \
|
|
|
|
} while (0)
|
|
|
|
#else
|
|
|
|
#define CPU_FILL(value, dest, size, bit) CPU_FILL_UNCHECKED(value, dest, size, bit)
|
|
|
|
#endif
|
|
|
|
|
2016-10-31 08:14:22 +00:00
|
|
|
#define CpuFill16(value, dest, size) CPU_FILL(value, dest, size, 16)
|
|
|
|
#define CpuFill32(value, dest, size) CPU_FILL(value, dest, size, 32)
|
|
|
|
|
2023-07-24 19:04:23 +01:00
|
|
|
#define CPU_COPY_UNCHECKED(src, dest, size, bit) CpuSet(src, dest, CPU_SET_##bit##BIT | ((size)/(bit/8) & 0x1FFFFF))
|
|
|
|
|
|
|
|
#if MODERN
|
|
|
|
#define CPU_COPY(src, dest, size, bit) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
_Static_assert(_Alignof(src) >= (bit / 8), "source potentially unaligned"); \
|
|
|
|
_Static_assert(_Alignof(dest) >= (bit / 8), "destination potentially unaligned"); \
|
|
|
|
CPU_COPY_UNCHECKED(src, dest, size, bit); \
|
|
|
|
} while (0)
|
|
|
|
#else
|
|
|
|
#define CPU_COPY(src, dest, size, bit) CPU_COPY_UNCHECKED(src, dest, size, bit)
|
|
|
|
#endif
|
2016-10-31 08:14:22 +00:00
|
|
|
|
|
|
|
#define CpuCopy16(src, dest, size) CPU_COPY(src, dest, size, 16)
|
|
|
|
#define CpuCopy32(src, dest, size) CPU_COPY(src, dest, size, 32)
|
|
|
|
|
|
|
|
#define CpuFastFill(value, dest, size) \
|
|
|
|
{ \
|
|
|
|
vu32 tmp = (vu32)(value); \
|
|
|
|
CpuFastSet((void *)&tmp, \
|
|
|
|
dest, \
|
|
|
|
CPU_FAST_SET_SRC_FIXED | ((size)/(32/8) & 0x1FFFFF)); \
|
|
|
|
}
|
|
|
|
|
2017-01-14 01:40:03 +00:00
|
|
|
#define CpuFastFill16(value, dest, size) CpuFastFill(((value) << 16) | (value), (dest), (size))
|
|
|
|
|
2017-03-30 01:02:15 +01:00
|
|
|
#define CpuFastFill8(value, dest, size) CpuFastFill(((value) << 24) | ((value) << 16) | ((value) << 8) | (value), (dest), (size))
|
|
|
|
|
2016-10-31 08:14:22 +00:00
|
|
|
#define CpuFastCopy(src, dest, size) CpuFastSet(src, dest, ((size)/(32/8) & 0x1FFFFF))
|
|
|
|
|
2023-07-24 19:04:23 +01:00
|
|
|
#define DmaSetUnchecked(dmaNum, src, dest, control) \
|
2016-10-31 08:14:22 +00:00
|
|
|
{ \
|
|
|
|
vu32 *dmaRegs = (vu32 *)REG_ADDR_DMA##dmaNum; \
|
|
|
|
dmaRegs[0] = (vu32)(src); \
|
|
|
|
dmaRegs[1] = (vu32)(dest); \
|
|
|
|
dmaRegs[2] = (vu32)(control); \
|
|
|
|
dmaRegs[2]; \
|
|
|
|
}
|
|
|
|
|
2023-07-24 19:04:23 +01:00
|
|
|
#if MODERN
|
|
|
|
// NOTE: Assumes 16-bit DMAs.
|
|
|
|
#define DmaSet(dmaNum, src, dest, control) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
_Static_assert(_Alignof(src) >= __builtin_choose_expr(__builtin_constant_p(control), ((control) & (DMA_32BIT << 16)) ? 4 : 2, 2), "source potentially unaligned"); \
|
|
|
|
_Static_assert(_Alignof(dest) >= __builtin_choose_expr(__builtin_constant_p(control), ((control) & (DMA_32BIT << 16)) ? 4 : 2, 2), "destination potentially unaligned"); \
|
|
|
|
DmaSetUnchecked(dmaNum, src, dest, control); \
|
|
|
|
} while (0)
|
|
|
|
#else
|
|
|
|
#define DmaSet(dmaNum, src, dest, control) \
|
|
|
|
DmaSetUnchecked(dmaNum, src, dest, control)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define DMA_FILL_UNCHECKED(dmaNum, value, dest, size, bit) \
|
2016-10-31 08:14:22 +00:00
|
|
|
{ \
|
|
|
|
vu##bit tmp = (vu##bit)(value); \
|
|
|
|
DmaSet(dmaNum, \
|
|
|
|
&tmp, \
|
|
|
|
dest, \
|
|
|
|
(DMA_ENABLE | DMA_START_NOW | DMA_##bit##BIT | DMA_SRC_FIXED | DMA_DEST_INC) << 16 \
|
|
|
|
| ((size)/(bit/8))); \
|
|
|
|
}
|
|
|
|
|
2023-07-24 19:04:23 +01:00
|
|
|
#if MODERN
|
|
|
|
#define DMA_FILL(dmaNum, value, dest, size, bit) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
_Static_assert(_Alignof(dest) >= (bit / 8), "destination potentially unaligned"); \
|
|
|
|
DMA_FILL_UNCHECKED(dmaNum, value, dest, size, bit); \
|
|
|
|
} while (0)
|
|
|
|
#else
|
|
|
|
#define DMA_FILL(dmaNum, value, dest, size, bit) DMA_FILL_UNCHECKED(dmaNum, value, dest, size, bit)
|
|
|
|
#endif
|
|
|
|
|
2016-10-31 08:14:22 +00:00
|
|
|
#define DmaFill16(dmaNum, value, dest, size) DMA_FILL(dmaNum, value, dest, size, 16)
|
|
|
|
#define DmaFill32(dmaNum, value, dest, size) DMA_FILL(dmaNum, value, dest, size, 32)
|
|
|
|
|
|
|
|
// Note that the DMA clear macros cause the DMA control value to be calculated
|
|
|
|
// at runtime rather than compile time. The size is divided by the DMA transfer
|
|
|
|
// unit size (2 or 4 bytes) and then combined with the DMA control flags using a
|
|
|
|
// bitwise OR operation.
|
|
|
|
|
2023-07-24 19:04:23 +01:00
|
|
|
#define DMA_CLEAR_UNCHECKED(dmaNum, dest, size, bit) \
|
2016-10-31 08:14:22 +00:00
|
|
|
{ \
|
|
|
|
vu##bit *_dest = (vu##bit *)(dest); \
|
|
|
|
u32 _size = size; \
|
|
|
|
DmaFill##bit(dmaNum, 0, _dest, _size); \
|
|
|
|
}
|
|
|
|
|
2023-07-24 19:04:23 +01:00
|
|
|
#if MODERN
|
|
|
|
#define DMA_CLEAR(dmaNum, dest, size, bit) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
_Static_assert(_Alignof(dest) >= (bit / 8), "destination potentially unaligned"); \
|
|
|
|
DMA_CLEAR_UNCHECKED(dmaNum, dest, size, bit); \
|
|
|
|
} while (0)
|
|
|
|
#else
|
|
|
|
#define DMA_CLEAR(dmaNum, dest, size, bit) DMA_CLEAR_UNCHECKED(dmaNum, dest, size, bit)
|
|
|
|
#endif
|
|
|
|
|
2016-10-31 08:14:22 +00:00
|
|
|
#define DmaClear16(dmaNum, dest, size) DMA_CLEAR(dmaNum, dest, size, 16)
|
|
|
|
#define DmaClear32(dmaNum, dest, size) DMA_CLEAR(dmaNum, dest, size, 32)
|
|
|
|
|
2023-07-24 19:04:23 +01:00
|
|
|
#define DMA_COPY_UNCHECKED(dmaNum, src, dest, size, bit) \
|
2016-10-31 08:14:22 +00:00
|
|
|
DmaSet(dmaNum, \
|
|
|
|
src, \
|
|
|
|
dest, \
|
|
|
|
(DMA_ENABLE | DMA_START_NOW | DMA_##bit##BIT | DMA_SRC_INC | DMA_DEST_INC) << 16 \
|
|
|
|
| ((size)/(bit/8)))
|
|
|
|
|
2023-07-24 19:04:23 +01:00
|
|
|
#if MODERN
|
|
|
|
#define DMA_COPY(dmaNum, src, dest, size, bit) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
_Static_assert(_Alignof(src) >= (bit / 8), "source potentially unaligned"); \
|
|
|
|
_Static_assert(_Alignof(dest) >= (bit / 8), "destination potentially unaligned"); \
|
|
|
|
DMA_COPY_UNCHECKED(dmaNum, src, dest, size, bit); \
|
|
|
|
} while (0)
|
|
|
|
#else
|
|
|
|
#define DMA_COPY(dmaNum, src, dest, size, bit) DMA_COPY_UNCHECKED(dmaNum, src, dest, size, bit)
|
|
|
|
#endif
|
|
|
|
|
2016-10-31 08:14:22 +00:00
|
|
|
#define DmaCopy16(dmaNum, src, dest, size) DMA_COPY(dmaNum, src, dest, size, 16)
|
|
|
|
#define DmaCopy32(dmaNum, src, dest, size) DMA_COPY(dmaNum, src, dest, size, 32)
|
2017-11-24 23:00:26 +00:00
|
|
|
|
|
|
|
#define DmaCopyLarge(dmaNum, src, dest, size, block, bit) \
|
|
|
|
{ \
|
|
|
|
const void *_src = src; \
|
|
|
|
void *_dest = dest; \
|
|
|
|
u32 _size = size; \
|
|
|
|
while (1) \
|
|
|
|
{ \
|
|
|
|
DmaCopy##bit(dmaNum, _src, _dest, (block)); \
|
|
|
|
_src += (block); \
|
|
|
|
_dest += (block); \
|
|
|
|
_size -= (block); \
|
|
|
|
if (_size <= (block)) \
|
|
|
|
{ \
|
|
|
|
DmaCopy##bit(dmaNum, _src, _dest, _size); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define DmaCopyLarge16(dmaNum, src, dest, size, block) DmaCopyLarge(dmaNum, src, dest, size, block, 16)
|
|
|
|
|
|
|
|
#define DmaCopyLarge32(dmaNum, src, dest, size, block) DmaCopyLarge(dmaNum, src, dest, size, block, 32)
|
|
|
|
|
|
|
|
#define DmaFillLarge(dmaNum, value, dest, size, block, bit) \
|
|
|
|
{ \
|
|
|
|
void *_dest = dest; \
|
|
|
|
u32 _size = size; \
|
|
|
|
while (1) \
|
|
|
|
{ \
|
|
|
|
DmaFill##bit(dmaNum, value, _dest, (block)); \
|
|
|
|
_dest += (block); \
|
|
|
|
_size -= (block); \
|
|
|
|
if (_size <= (block)) \
|
|
|
|
{ \
|
|
|
|
DmaFill##bit(dmaNum, value, _dest, _size); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define DmaFillLarge16(dmaNum, value, dest, size, block) DmaFillLarge(dmaNum, value, dest, size, block, 16)
|
|
|
|
|
|
|
|
#define DmaFillLarge32(dmaNum, value, dest, size, block) DmaFillLarge(dmaNum, value, dest, size, block, 32)
|
|
|
|
|
2018-01-24 03:30:13 +00:00
|
|
|
#define DmaClearLarge(dmaNum, dest, size, block, bit) \
|
|
|
|
{ \
|
|
|
|
void *_dest = dest; \
|
|
|
|
u32 _size = size; \
|
|
|
|
while (1) \
|
|
|
|
{ \
|
|
|
|
DmaFill##bit(dmaNum, 0, _dest, (block)); \
|
|
|
|
_dest += (block); \
|
|
|
|
_size -= (block); \
|
|
|
|
if (_size <= (block)) \
|
|
|
|
{ \
|
|
|
|
DmaFill##bit(dmaNum, 0, _dest, _size); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define DmaClearLarge16(dmaNum, dest, size, block) DmaClearLarge(dmaNum, dest, size, block, 16)
|
|
|
|
|
|
|
|
#define DmaClearLarge32(dmaNum, dest, size, block) DmaClearLarge(dmaNum, dest, size, block, 32)
|
|
|
|
|
2017-11-24 23:00:26 +00:00
|
|
|
#define DmaCopyDefvars(dmaNum, src, dest, size, bit) \
|
|
|
|
{ \
|
|
|
|
const void *_src = src; \
|
|
|
|
void *_dest = dest; \
|
|
|
|
u32 _size = size; \
|
|
|
|
DmaCopy##bit(dmaNum, _src, _dest, _size); \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define DmaCopy16Defvars(dmaNum, src, dest, size) DmaCopyDefvars(dmaNum, src, dest, size, 16)
|
|
|
|
#define DmaCopy32Defvars(dmaNum, src, dest, size) DmaCopyDefvars(dmaNum, src, dest, size, 32)
|
|
|
|
|
|
|
|
#define DmaFillDefvars(dmaNum, value, dest, size, bit) \
|
|
|
|
{ \
|
|
|
|
void *_dest = dest; \
|
|
|
|
u32 _size = size; \
|
|
|
|
DmaFill##bit(dmaNum, value, _dest, _size); \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define DmaFill16Defvars(dmaNum, value, dest, size) DmaFillDefvars(dmaNum, value, dest, size, 16)
|
|
|
|
#define DmaFill32Defvars(dmaNum, value, dest, size) DmaFillDefvars(dmaNum, value, dest, size, 32)
|
|
|
|
|
|
|
|
#define DmaClearDefvars(dmaNum, dest, size, bit) \
|
|
|
|
{ \
|
|
|
|
void *_dest = dest; \
|
|
|
|
u32 _size = size; \
|
|
|
|
DmaClear##bit(dmaNum, _dest, _size); \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define DmaClear16Defvars(dmaNum, dest, size) DmaClearDefvars(dmaNum, dest, size, 16)
|
|
|
|
#define DmaClear32Defvars(dmaNum, dest, size) DmaClearDefvars(dmaNum, dest, size, 32)
|
2016-10-31 08:14:22 +00:00
|
|
|
|
|
|
|
#define DmaStop(dmaNum) \
|
|
|
|
{ \
|
|
|
|
vu16 *dmaRegs = (vu16 *)REG_ADDR_DMA##dmaNum; \
|
|
|
|
dmaRegs[5] &= ~(DMA_START_MASK | DMA_DREQ_ON | DMA_REPEAT); \
|
|
|
|
dmaRegs[5] &= ~DMA_ENABLE; \
|
|
|
|
dmaRegs[5]; \
|
|
|
|
}
|
2016-01-08 20:49:23 +00:00
|
|
|
|
2017-09-07 07:51:59 +01:00
|
|
|
#define IntrEnable(flags) \
|
|
|
|
{ \
|
|
|
|
u16 imeTemp; \
|
|
|
|
\
|
|
|
|
imeTemp = REG_IME; \
|
|
|
|
REG_IME = 0; \
|
|
|
|
REG_IE |= flags; \
|
|
|
|
REG_IME = imeTemp; \
|
|
|
|
} \
|
|
|
|
|
2016-01-08 20:49:23 +00:00
|
|
|
#endif // GUARD_GBA_MACRO_H
|