planecnt = BltBitMap(SrcBitMap, SrcX, SrcY, DstBitMap,
D0 A0 D0:16 D1:16 A1
DstX, DstY, SizeX, SizeY, Minterm, Mask, TempA)
D2:16 D3:16 D4:16 D5:16 D6:8 D7:8 A2
ULONG BltBitMap(struct BitMap *, WORD, WORD, struct BitMap *,
WORD, WORD, WORD, WORD, UBYTE, UBYTE, UWORD *);
Perform a blit operation which copies a rectangle from one area in a BitMap to a different BitMap, or within the same BitMap. Depending upon the Minterm used, the operation may ignore the source BitMap and change the destination BitMap.
The BltBitMap() function performs no sanity checking on its parameters which places the burden on you to make sure that the blit operation is safe and will not result in data corruption or undefined behaviour.
This means that you need to watch the position and size of the area which may not overstep the indicated ranges. Of particular note is that the Blitter of the original Amiga chipset (OCS) is limited to an area size of 1024 x 1024 pixels. Furthermore, the OCS Blitter cannot be used with BitMaps wider than 1024 pixels (i.e. BitMap.BytesPerRow > 128) even if the area size never exceeds 1024 x 1024 pixels. Exceeding any of these limitations may result in data corruption and undefined behaviour.
All the BltBitMap() function position and size parameters are signed 16 bit integers. If called from 'C' code, the parameters will be filled with 32 bit integer values, though, because the <clib/graphics_protos.h> header file defines them like this for historic reasons. Do not make the mistake of actually passing position and size information which makes use of the 32 bit integer range when running on an Amiga which solely uses the Blitter.
The graphics.library functions (such as
ClipBlit(),
BltBitMapRastPort(),
BltMaskBitMapRastPort()) all build upon the same Blitter code and share the same limitations. The intuition.library functions which make use of the Blitter (such as DrawImage()) will use the same graphics.library functions.
When BltBitMap() returns you must assume that the Blitter has been started and is still running. If your code depends upon it that the Blitter is no longer active, such as when releasing chip memory safely, you must first call
WaitBlit() to be sure.
This function may use the blitter.
MINTERMS AND HOW TO USE THEM
The BltBitMap() function is intended to make use of the Amiga Blitter custom chip hardware which uses a Minterm value to determine how to combine the Blitter sources into a result which will be written to the destination.
The Minterm value is a combination of control flags which determine which logic operations are to be performed. You can find these flags in the <hardware/blit.h> header file:
/* definitions for blitter control register 0 */
#define ABC 0x80
#define ABNC 0x40
#define ANBC 0x20
#define ANBNC 0x10
#define NABC 0x8
#define NABNC 0x4
#define NANBC 0x2
#define NANBNC 0x1
These macro names are shorthand for the respective logical operation, e.g. ABC stands for "A & B & C" in 'C' language terms. The "N" is a logical "not" operation, e.g. NABNC stands for "~A & B & ~C".
Each flag stands for a logical operation performed on up to three Blitter sources. The BltBitMap() function uses two sources only, these being B (SrcBitMap) and C (DstBitMap; also the destination).
Source A is never enabled by BltBitMap(), which means that only four operations may be used, and it does not matter if source A is present or negated:
#define BC 0x8 /* B & C */
#define BNC 0x4 /* B & ~C */
#define NBC 0x2 /* ~B & C */
#define NBNC 0x1 /* ~B & ~C */
Each Minterm combines between 0 and 4 of these flags. Here are all the 16 Minterms which can be used with BltBitMap() and, by extension, the
ClipBlit(),
BltBitMapRastPort() and
BltMaskBitMapRastPort() functions which make use of them:
Clear the bit and ignore the source:
Minterm = 0 (0x00)
B | C | 0
--|--|----------------------
0 | 0 | 0
0 | 1 | 0
1 | 0 | 0
1 | 1 | 0
Set the bit if B and C are both clear, otherwise clear the bit:
Minterm = ANBNC (0x10)
B | C | (~B & ~C)
--|--|----------------------
0 | 0 | 1
0 | 1 | 0
1 | 0 | 0
1 | 1 | 0
Clear the bit if there is a bit set in the source (mask):
Minterm = ANBC (0x20)
B | C | (~B & C)
--|--|----------------------
0 | 0 | 0
0 | 1 | 1
1 | 0 | 0
1 | 1 | 0
Copy an inverted source to the destination:
Minterm = ANBNC|ANBC (0x30)
B | C | (~B & ~C) | (~B & C)
--|--|----------------------
0 | 0 | 1
0 | 1 | 1
1 | 0 | 0
1 | 1 | 0
Minterm = ABNC (0x40)
B | C | (B & ~C)
--|--|----------------------
0 | 0 | 0
0 | 1 | 0
1 | 0 | 1
1 | 1 | 0
Invert the destination and ignore the source:
Minterm = ANBNC|ABNC (0x50)
B | C | (~B & ~C) | (B & ~C)
--|--|----------------------
0 | 0 | 1
0 | 1 | 0
1 | 0 | 1
1 | 1 | 0
Clear the bit if source and destination are identical, otherwise
set the bit:
Minterm = ANBC|ABNC (0x60)
B | C | (~B & C) | (B & ~C)
--|--|----------------------
0 | 0 | 0
0 | 1 | 1
1 | 0 | 1
1 | 1 | 0
Minterm = ANBNC|ANBC|ABNC (0x70)
B | C | (~B & ~C) | (~B & C) | (B & ~C)
--|--|----------------------
0 | 0 | 1
0 | 1 | 1
1 | 0 | 1
1 | 1 | 0
Set the bit if both source and destination are set, otherwise
clear the bit (sieve):
Minterm = ABNC (0x80)
B | C | (B & C)
--|--|----------------------
0 | 0 | 0
0 | 1 | 0
1 | 0 | 0
1 | 1 | 1
Set the bit if source and destination are identical, otherwise
clear the bit:
Minterm = ANBNC|ABNC (0x90)
B | C | (~B & ~C) | (B & C)
--|--|----------------------
0 | 0 | 1
0 | 1 | 0
1 | 0 | 0
1 | 1 | 1
Minterm = ANBC|ABNC (0xA0)
B | C | (~B & C) | (B & C)
--|--|----------------------
0 | 0 | 0
0 | 1 | 1
1 | 0 | 0
1 | 1 | 1
Minterm = ANBNC|ANBC|ABNC (0xB0)
B | C | (~B & ~C) | (~B & C) | (B & C)
--|--|----------------------
0 | 0 | 1
0 | 1 | 1
1 | 0 | 0
1 | 1 | 1
Copy the source to the destination:
Minterm = ABNC|ABC (0xC0)
B | C | (B & ~C) | (B & C)
--|--|----------------------
0 | 0 | 0
0 | 1 | 0
1 | 0 | 1
1 | 1 | 1
Minterm = ANBNC|ABNC|ABNC (0xD0)
B | C | (~B & ~C) | (B & ~C) | (B & C)
--|--|----------------------
0 | 0 | 1
0 | 1 | 0
1 | 0 | 1
1 | 1 | 1
Minterm = ANBC|ABNC|ABNC (0xE0)
B | C | (~B & C) | (B & ~C) | (B & C)
--|--|----------------------
0 | 0 | 0
0 | 1 | 1
1 | 0 | 1
1 | 1 | 1
Set the bit and ignore the source:
Minterm = ANBNC|ANBC|ABNC|ABNC (0xF0)
B | C | (~B & ~C) | (~B & C) | (B & ~C) | (B & C)
--|--|----------------------
0 | 0 | 1
0 | 1 | 1
1 | 0 | 1
1 | 1 | 1
WHEN AND HOW TO ALLOCATE MEMORY FOR THE TEMPA PARAMETER
If SrcBitMap and DstBitMap are the same and the source and destination area overlaps then the copy operation will have to buffer the source row data. Unless you provide a chip memory buffer for this purpose, BltBitMap() will attempt to allocate such memory for you and release it again before it returns. Should this attempt fail, BltBitMap() will try to use a special private memory buffer set aside for such an eventuality.
Both allocating temporary memory and using the private memory buffer come with a price. It takes time to allocate memory, especially if the attempt fails (due to fragmentation issues, for example). Accessing the private memory buffer requires waiting for other clients which compete for the same resource to do their blitting first. All of this can become burdensome if you must perform your blit operations with as little delay and overhead as possible, such as when you have to perform multiple overlapping blits in sequence.
In these cases you might want to allocate chip memory prior to calling BltBitMap(). Here is how you would go about it:
#include <hardware/blit.h>
PLANEPTR tempa;
tempa = AllocMem(MAXBYTESPERROW, MEMF_CHIP);
if (tempa != NULL)
{
/* Perform all overlapping blit operations */
BltBitMap(source, x, y, destination, x, y, minterm, 0xff, tempa)
...
/* Wait for the Blitter to finish before you release the
* memory as otherwise a program may succeed in reallocating
* it while the Blitter still modifies the memory contents!
*/
WaitBlit();
FreeMem(tempa, MAXBYTESPERROW);
}