count = ReadPixelArray8(rp,xstart,ystart,xstop,ystop,array,temprp)
D0 A0 D0:16 D1:16 D2:16 D3:16 A2 A1
LONG ReadPixelArray8(struct RastPort *, UWORD, UWORD, UWORD, UWORD,
UBYTE *, struct RastPort *);
Read pixel color data from the RastPort, beginning at position (xstart, ystart), stopping at position (xstop, ystart). This process will continue for each following row, reading from position xstart to xstop, until the ystop row has been read.
When finished, a total of (xstop - xstart + 1) * (ystop - ystart + 1) pixels will have been read in sequence.
The color data is read in bulk, which is significantly faster than calling
ReadPixel() for each single pixel covered.
For each pixel in the array:
- Pen
-
Pixel color (0..255) at that position
- count
-
The number of pixels read; this will be (xstop - xstart + 1) * (ystop - ystart + 1)
READPIXELARRAY8 MAY STORE MORE DATA THAN YOU ASKED FOR
ReadPixelArray8() works like the
ReadPixelLine8() function, reading one row at a time. As such it has the same limitations as
ReadPixelLine8() in that each row read will copy (xstop - xstart + 1) number of pixels to the temporary RastPort by means of
ClipBlit() and then process the entire row. This means that the number of bytes written to the array will be rounded up to match the horizontal size of the temprp.BitMap. The number of bytes will be (xstop - xstart + 1), rounded up to a multiple of 16.
If you ask for n = (xstop - xstart + 1) pixels to be read, data for 16 * ((n + 15) / 16) pixels will be read and stored in each row of the array. Even with xstop == xstart a total of 16 pixels will be read and their data will be stored in the row of the array.
This means that the array you need to allocate will not necessarily feature (xstop - xstart + 1) bytes per row. There may be padding of of up to 15 bytes at the end of each single row. To calculate the complete size of each row, the RASSIZE() macro found in <graphics/gfx.h> may be used. The row size in bytes can be calculated as RASSIZE(xstop - xstart + 1, 1).
The easiest way to ensure that the array is large enough is to make use of the RASSIZE() macro. Make sure that your array has room for at least RASSIZE(xstop - xstart + 1, ystop - ystart + 1) number of bytes.
READPIXELARRAY8 DOES NOT RETURN THE TOTAL NUMBER OF PIXELS READ
Every call to ReadPixelArray8() returns the number of pixels you requested to be read. In practice, ReadPixelArray8() may read and process more than these pixels. Do not expect the pixel count returned to accurately reflect the number of bytes changed in the array.
The correct use of ReadPixelArray8() requires the temporary RastPort and the array to store the pixel color data in to be set up in a very specific manner. Deviating from these requirements can lead to data corruption and undefined behavior.
How you would set up the temporary RastPort and allocate a memory buffer of the proper size is detailed with ready to use example code in the EXAMPLE section.
For ReadPixelArray8() you would use the example code as follows:
#include <proto/exec.h>
#include <proto/graphics.h>
struct RastPort temprp;
UBYTE * array;
if (setup_temp_rastport(&temprp, rp, xstop - xstart + 1))
{
array = allocate_pixel_array(xstop - xstart + 1, ystop - ystart + 1);
if (array != NULL)
{
/* Call ReadPixelArray8(rp,xstart,ystart,xstop,ystop,array,&temprp)
FreeVec(array);
}
cleanup_temp_rastport(&temprp, xstop - xstart + 1);
}
To initialize a temporary RastPort and allocate memory for its BitMap
you could use the following functions. Both require a temporary RastPort
which first has to be initialized by setup_temp_rastport(). This involves
allocating memory for a BitMap. In order to release the allocated memory
call cleanup_temp_rastport().
The temporary RastPort can be allocated separately but a declaring a
local 'struct RastPort temprp;' is fine, too.
The
ReadPixelLine8(), ReadPixelArray8(),
WritePixelLine8() and
WritePixelArray8() functions transfer pixel color data to/from a
memory buffer with very specific size requirements. The
allocate_pixel_array() function below will accomplish this.
/* Initialize the temporary RastPort, then create a BitMap
* which is suitable for use with
ReadPixelLine8(), ReadPixelArray8(),
*
WritePixelLine8() and
WritePixelArray8().
*
* This function will return FALSE if the BitMap could not be
* created and TRUE otherwise.
*
* Call cleanup_temp_rastport() when the BitMap is no longer needed,
* so that the memory allocated for it will be released again.
*/
BOOL
setup_temp_rastport(
struct RastPort * temp_rp,
CONST struct RastPort * rp,
UWORD width)
{
struct BitMap * temp_rp_bm;
BOOL success = FALSE;
int i, j;
/* Use a default RastPort configuration to avoid side-effects
* resulting from copying the source RastPort data.
*/
InitRastPort(temp_rp);
/* For Kickstart 3.0 and beyond use the designated BitMap
* allocation function in graphics.library.
*/
if (GfxBase->LibNode.lib_Version >= 39)
{
temp_rp->BitMap =
AllocBitMap(width, 1,
GetBitMapAttr(rp->BitMap, BMA_DEPTH),
GetBitMapAttr(rp->BitMap, BMA_FLAGS) & BMF_INTERLEAVED,
NULL);
}
/* For Kickstart 2.x set up a BitMap via
InitBitMap() and
*
AllocRaster() for each single bitplane.
*/
else
{
temp_rp_bm = AllocVec(sizeof(*temp_rp_bm), MEMF_PUBLIC);
if (temp_rp_bm != NULL)
{
InitBitMap(temp_rp_bm, width, 1, rp->BitMap->Depth);
temp_rp->BitMap = temp_rp_bm;
for (i = 0 ; i < temp_rp_bm->Depth ; i++)
{
temp_rp_bm->Planes[i] =
AllocRaster(width, 1);
if (temp_rp_bm->Planes[i] == NULL)
{
/* Free all the bit planes allocated so far. */
for (j = 0 ; j < i ; j++)
{
FreeRaster(temp_rp_bm->Planes[j], width, 1);
temp_rp_bm->Planes[j] = NULL;
}
temp_rp->BitMap = NULL;
FreeVec(temp_rp_bm);
break;
}
}
}
}
success = (BOOL)(temp_rp->BitMap != NULL);
return success;
}
/* Release the BitMap allocated for the temporary RastPort
* when no longer needed.
*/
VOID
cleanup_temp_rastport(
struct RastPort * temp_rp,
UWORD width)
{
int i;
if (temp_rp->BitMap != NULL)
{
/* Wait for the Blitter to finish, in case it is still
* modifying the BitMap created.
*/
WaitBlit();
if (GfxBase->LibNode.lib_Version >= 39)
{
FreeBitMap(temp_rp->BitMap);
}
else
{
for (i = 0 ; i < temp_rp->BitMap->Depth ; i++)
{
if (temp_rp->BitMap->Planes[i] != NULL)
{
FreeRaster(temp_rp->BitMap->Planes[i],
}
}
}
temp_rp->BitMap = NULL;
}
}
/* Allocate a memory buffer suitable for use with the
ReadPixelLine8(),
* ReadPixelArray8(),
WritePixelLine8() and
WritePixelArray8() functions.
*
* Returns the address of the buffer allocated or NULL for failure. The
* memory buffer allocated can be freed with the FreeVec() function.
*
* For
ReadPixelLine8() and
WritePixelLine8() you would use height = 1.
*/
UBYTE *
allocate_pixel_array(UWORD width, UWORD height)
{
UBYTE * array;
array = AllocVec(RASSIZE(width, height), MEMF_PUBLIC);
return array;
}
Previous documentation suggested copying the contents of the rp parameter when setting up the temprp (temporary RastPort). This is not recommended because the rp.Mask settings could interfere with the
ClipBlit() operation implied by
ReadPixelLine8(). This could yield corrupted color data.
ReadPixel(),
ReadPixelLine8(), <graphics/gfx.h>, <graphics/rastport.h>