bitmap=AllocBitMap(sizex,sizey,depth, flags, friend_bitmap)
struct BitMap *AllocBitMap(ULONG,ULONG,ULONG,ULONG, struct BitMap *);
Allocates and initializes a bitmap structure. Allocates and initializes bitplane data, and sets the bitmap's planes to point to it.
Returns a pointer to a 'struct BitMap' or NULL in case of failure.
NULL will be returned if there is not enough contiguous RAM available to allocate an interleaved BitMap. Even if there is still nominally enough RAM available, the constraints of an interleaved BitMap render it unusable.
NULL will be returned if the requested BitMap width exceeds 32760 pixels on an ECS or AGA system.
Interleaved BitMaps are a crucial building block of the Amiga graphics system and its hardware. You are bound to make use of them, but you should be aware of their limitations and the constraints which govern their proper use.
Exceeding these limitations will lead to software errors and data corruption, whose causes are difficult to track down. The graphics.library Blitter functions do not provide protection against accidental misuse.
- Note:
-
These constraints are subtle because interleaved BitMaps are not merely an extension of the BitMaps which you are already familiar with. There are side-effects which must be accounted for.
The AllocBitMap() function was introduced in Kickstart 3.0. Prior to this the creation of a BitMap suitable for both display purposes and Blitter operations went along the following steps:
struct BitMap bm;
int width = 320, height = 200, depth = 4;
int i;
/* This will yield bm.BytesPerRow == 2 * ((320 + 15)) / 16 == 40.
* You need to add 40 bytes to the address of a bit plane row
* to access the next row.
*/
InitBitMap(&bm, width, height, depth);
for (i = 0 ; i < depth ; i++)
bm.Planes[i] =
AllocRaster(width, height);
The memory bandwidth requirements of the AGA chipset made it necessary for bit planes not to be allocated separately, as illustrated by this example.
Preferably, the memory contents representing each line of graphics should be close by so that the display hardware could fetch all of it in a single sweep. This is what the interleaved bitmap format enables and why the AllocBitMap() function exists which creates this special type of BitMap.
-
For the BitMap "interleaving" means that all the planes of a single BitMap row directly follow one another in memory. For an interleaved BitMap the BitMap.BytesPerRow value will be affected by the BitMap depth. For the example above which constructed a BitMap using InitBitMap() and AllocRaster() you would find that invoking AllocBitMap() with width=320, depth=4 and requesting the BitMap to be interleaved, the BitMap.BytesPerRow value will be 4 * 40 = 160 bytes instead of 40.
- CAUTION:
-
The number of bytes per row are no longer both the row modulus and an indication of the width of each row for an interleaved BitMap.
-
AllocBitMap() allocates the interleaved plane data, taking alignment requirements into account (e.g. AGA requires that each row of image data must be aligned to a 64 bit address, i.e. a multiple of 8 bytes), in a single consecutive chunk.
- CAUTION:
-
Memory fragmentation constraints may cause the allocation to fail entirely.
-
The ECS and AGA Blitter cannot handle BitMaps with BytesPerRow values exceeding 4095 bytes. Which means that interleaved BitMaps on the ECS and AGA chipsets are constrained to width * depth <= 4095 (approximately).
The maximum BitMap height which the Blitter can access is reduced, too. For the ECS and AGA chipsets the height is limited to 32768 / depth pixels (approximately).
- CAUTION:
-
AllocBitMap() permits you to allocate BitMaps which exceed these dimensions. But you should be aware that you may be unable to use them with the Blitter when running on a system which exclusively uses the Amiga custom chipset.
For Amigas which use the original chipset (OCS) no sanity check is performed when BitMaps wider than 1024 pixels are requested. The operating system will try to satisfy your request even though the resulting BitMap will be unsafe to use. You should catch such cases before the call to AllocBitMap() is made!
If you still try, the results may lead to software errors and data corruption.
When allocating using a friend bitmap, it is not safe to assume anything about the structure of the bitmap data if that friend BitMap might not be a standard amiga bitmap.
For instance, if the workbench is running on a non-amiga display device, its Screen->RastPort->BitMap won't be in standard Amiga format. The only safe operations to perform on a non-standard BitMap are:
-
blitting it to another bitmap, which must be either a standard Amiga bitmap, or a friend of this bitmap.
-
blitting from this bitmap to a friend bitmap or to a standard Amiga bitmap.
-
attaching it to a rastport and making rendering calls.
Good arguments to pass for the friend_bitmap are your window's RPort->BitMap, and your screen's RastPort->BitMap. Do NOT pass &(screenptr->BitMap)!
BitMaps not allocated with BMF_DISPLAYABLE may not be used as Intuition Custom BitMaps or as RasInfo->BitMaps. They may be blitted to a BMF_DISPLAYABLE BitMap, using one of the
BltBitMap() family of functions.
The taglist which AllocBitMap() accepts with the flag combination of BMF_RTGTAGS|BMF_RTGCHECK|BMF_FRIENDISTAG uses tag IDs that are intentionally identical to those of OpenScreenTagList().
It is good practise, which is in fact followed by intuition V47, to pass over the taglist used for creating a screen to this function to receive a bitmap that is suitable for the specific mode ID and screen dimensions.