windows-nt/Source/XPSP1/NT/multimedia/media/avi/mciavi32/palettes.txt

136 lines
5.8 KiB
Plaintext
Raw Normal View History

2020-09-26 03:20:57 -05:00
AVI and Palettes
----------------
This is an overview of AVI's use of palettes: there are some small
complexities beyond this that are not generally significant.
Palette Sources
---------------
When playing on a palettised display, AVI must use a palette for
drawing the frames. It gets these palettes from one of these places:
- from the AVI file header. Attached to the DIB header at the
start of an 8-bit file will be a colour table (an array of 4-byte
RGBQUAD values). The frame DIBs will consist of indices into this
table. In addition, new palettes can appear throughout the movie
to be used for frames from that point onwards.
- from the dither proc. If the movie is a higher colour
resolution, drawdib will dither it to the display colour resolution.
In the case of 8-bit displays, 16 and 24-bit videos will be dithered
to a standard 256-colour palette created in drawdib\dith775.c which
has 7 levels of red and green and 5 levels of blue.
- from the codec. If the video is a higher colour resolution
and being decompressed to 8-bit palettised, the codec will supply the
palette. for example, the Video 1 codec will decompress 16-bit CRAM
videos directly to 8-bit pal using the standard 7-7-5 palette (same
as the dither code's palette).
- 4-bit displays will always use the 16-colour VGA palette.
In all of the cases, once we have a frame fully decompressed/dithered
and ready to draw, it is a DIB that consists of a series of indices
into one of these palettes. We may also have:
- A client palette: a palette handle can be sent us from the
client. This represents the palette used by the window we are playing
in. We need to draw using this palette to avoid redraws of the
surrounding window when playing in place. In this case we need to
deal with two palettes: the palette containing the colours actually
used in the DIBs (from the avi file or supplied by dither/codec), and
the palette we need to draw with, supplied by the client.
Using The Palette
-----------------
We draw the frame using SetDIBitsToDevice (or possibly
StretchDIBits). This will interpret the DIB bits in one of two ways
depending on the fuColorUse flag (in fact there are more than two,
but we only use these)
DIB_PAL_INDICES: this means that the bytes in the DIB are
indices directly into the system palette.
DIB_PAL_COLORS the bytes in the DIB are indices into a colour table
supplied as the bmiColors field of the bitmapinfoheader supplied.
This colour table must contain indices into the logical palette
selected into the DC (the header including colour table, and the DC
are supplied as arguments).
The DIB_PAL_COLORS case is slower since GDI has to translate the DIB
bytes once using the bmiColors table, and then once again from the
logical palette of the DC to the system palette. However, since we
can't realistically change the DIB data, we can only use
DIB_PAL_INDICES if the system palette is the same as our palette,
which depends on the RealizePalette behaviour:
Realizing Palettes:
-------------------
Before drawing the DIB, we select the palette we want to draw with
into the DC and then call RealizePalette. This has one of two
behaviours:
- if we are the foreground app, RealizePalette assigns entries in the
system palette to contain the colours in our logical palette. The
system keeps some entries in the palette with standard colours in
(the first 10 and last 10 entries) - these are the 'static' colours;
it will allow us to use the remaining entries.
- if not, RealizePalette creates a mapping between our logical
palette and the nearest colours in the current system palette.
When a foreground app realizes a palette, it changes entries in the
system palette, and all other apps need to re-Realize their palettes
in order to get a mapping to the new system palette (unless they are
only using the default, static colours). Thus when we start playing a
video, all other palette-aware apps change colours (when the system
palette changes) and then redraw with the mapping to the new system
palette.
Support for DIB_PAL_INDICES.
---------------------------
Using DIB_PAL_INDICES is only possible if the system palette not only
contains all our colours, but in exactly the same order. To ensure
that this is possible, AVI palettes are generally created containing
the static colours as the first and last 10 colours, and
movie-specific colours in between. Thus when we are foreground and
realize the palette, the system palette will end up exactly the same
as the movie palette, and we can use DIB_PAL_INDICES with its
significant performance benefits.
Whenever the system palette changes, we compare all the colours in
the system palette with the colours in our DIBs. If the two colour
tables are the same, we know it is safe to use DIB_PAL_INDICES. If
not, then we use DIB_PAL_COLORS with a colour table. This colour
table maps the contents of the DIB to the logical palette: this is a
no-op, since the logical palette for our dc is the same as the DIB
palette. The real translation is being done between the logical
palette and the current system palette, and this is set up by GDI
when we issue a 'realizepalette' call.
Support for client Palettes
---------------------------
We are sometimes given a client palette to use (mplayer does this if
playing in place, in an app that is using a non-default palette). The
reason for this is to avoid the flash and re-draw of the client app
that would be necessary if we realized our palette and changed the
system palette. In this case, we select the client palette as our dc
palette, and we build a colour table that maps entries in our palette
to the nearest entry in the client palette. When we then draw - using
DIB_PAL_COLORS - GDI will use this colour table to translate between
the DIB palette and the DC's logical palette ( which is the client
app's palette), and then will translate between the logical palette
and the system palette (probably the same).