2020-09-30 17:12:29 +02:00

182 lines
8.9 KiB
Plaintext

AVIFile: it's not just an API set, it's a way of life.
So, people are confused about how to approach the AVIFile APIs.
Let's try to help by making some things clear.
1. AVIFile isn't just for AVI files.
Perhaps this would be clearer if these functions had some different
name, but let's live with AVIFile for now. The important thing to
keep in mind is that we aren't talking about .AVI files, but about
some ideal time-based file format which supports the operations we'd
like it to support, like reading and writing from multiple
time-stamped streams. If the APIs conform closely to what's actually
in .AVI files, it's just because I made up both the .AVI format and
the AVIFile APIs.
The guiding principle is that any other type of file containing this
time-stamped sort of data, be it a QuickTime file, an MPEG file, a
.WAV file, or whatever, should be accessible in one simple way.
2. AVIFile isn't just for files.
Having a uniform API to read these different sorts of files is
convenient, but we can go beyond that. Once you have the abstract
concept of a "file" which is a bundle of "streams" (each of which has
a particular data type and format, along with a start time and
duration, and can be read from or written to at various points in
time), it is clear that not every "file" needs to correspond to an
actual disk file, and not every "stream" has to come from a "file" at
all.
Useful kinds of streams to think about:
Entirely synthetic streams, like the bouncing ball example; in this
case, a "stream" reduces to a function GiveMeVideoFrame(x).
Streams which are essentially filters applied to other streams; for
example, given a stream of video (that is, some abstract object that
can hand you frame 4 if you want it), you can build from that a
"compressed" stream, which is simply a stream that when asked for
frame 4 goes and asks the original stream for its frame 4, then
compresses it and hands it back to you, saying "here's frame 4, all
compressed like you wanted it."
3. AVIFile tries to conform to the Component Object model.
Since AVIFile is intended to be extensible to support additional file
formats, it has to have some way of transparently linking to DLLs
containing routines that understand those file formats. In the past,
we in MMSys have done this sort of thing using installable drivers
and a message-based scheme. (See MCI, ICM, and more....) With the
release of OLE 2.0, we have a new standard for this sort of thing.
It's a little scary at first, requires either C++ or some C++-like
thinking, and involves header files you haven't seen before. In any
case, we're all going to have to live with it.
4. Don't be intimidated by talk of C++ and "objects".
From a C point of view, all an "object pointer" like a PAVISTREAM is
is a pointer to a structure whose first member happens to be another
pointer to a table of functions. This means that in addition to
carrying around data like "how long is this video sequence", the
structure also contains pointers to code that knows how to actually
get things done. All C++ does for you is provide a nicer syntax for
doing this sort of thing.
You can make yourself a PAVISTREAM by hand. All you have to do
is allocate room for a structure big enough to contain the pointer
to the function table and any other data you need to keep around.
Then, you make a function table with the Read, Write, and other
functions to operate on your type of stream, and make sure your
structure points to your table. Just like that, you have a bona fide
PAVISTREAM which you can pass to AVISave, AVIStreamRead, and so on.
5. Luckily, you can usually ignore all of the "Component Object model"
stuff.
Unless you're doing something complicated, you should be able to just
call the various AVIFileXXXX and AVIStreamXXXX APIs and pretty much
ignore all of the talk of ISomethingOrOther and CLSID_Confusing.
If you are trying to make a DLL that will add support for reading and
writing a new file format, you will in fact have to learn something
about what a "class factory" is, but cross that bridge when you come
to it.
6. AVISave() is just a helper function.
All the AVISave function does is copy some streams into a new file.
It doesn't do anything that you couldn't do yourself by calling the
AVIFileXXXX and AVIStreamXXXX APIs. It does, however, make your life
easier by: calling AVIMakeCompressedStream for you according to the
options you pass in; calling AVIFileOpen and AVIFileCreateStream
appropriately to make the new streams in the new file you're making;
and finally, looping through all of the streams from start to end
copying the data from the old streams into the new file, and doing it
in the right order so that things come out nicely interleaved.
7. AVIStreamReadData, AVIStreamWriteData, AVIFileReadData, and
AVIFileWriteData are not actually for reading and writing real data.
I know, these routines have stupid names and they're just confusing
everybody.
What they're not for: These routines are not what you use to read and
write audio samples and video frames and that sort of thing. For
that, you must use AVIStreamRead and AVIStreamWrite. (If you have a
PAVIFile, call AVIFileGetStream and then call AVIStreamRead.)
What they're for: Reading and writing copyright information, the
author's name, and other stuff like that that's kept in INFO chunks
in RIFF files.
8. AVIStreamGetFrame is just another helper function.
AVIStreamGetFrame (and AVIStreamGetFrameOpen/Close) are just
functions which handle the relatively simple task of getting a
decompressed video frame out of a stream. This involves finding the
right ICM decompressor for the task at hand, figuring out where the
last key frame was, and decompressing frames as necessary.
Perhaps this should all be done automatically for you; perhaps you
should do this instead by opening a "decompressed" stream based on
the compressed stream you want to read. It's not either of those
ways now, so use AVIStreamGetFrame.
8a. What in heck is AVIMakeCompressedStream() for?
AVIMakeCompressedStream makes a new stream pointer for you that acts
just like the old stream you already had, but is compressed or
decompressed. That isn't very clear, so I'll try again: If you have
a PAVISTREAM which, when you read it, gives you back 8-bit RGB frames
of "Gone With the Wind", you can use AVIMakeCompressedStream to make
a stream that will return the same pictures, but compressed in, say,
CRAM format. The reverse is also true: given a compressed stream,
AVIMakeCompressedStream can make you an uncompressed version of that
stream.
To use it, just fill out an AVICOMPRESSOPTIONS structure. (Or pass
NULL, which will decompress.)
In theory, you can either read from the compressed stream or write to
it. (Right now, this still doesn't work for audio compression....)
However, you can only do one at a time.
9. What in the world is an HRESULT?
For some reason, all OLE 2.0 functions return HRESULTs which could
potentially contain extra information about an error that occurred.
For now, they are essentially just the same as a plain error code.
To convert one of the AVIERR_XXXX codes to an HRESULT, call
ResultFromScode() on it. To convert back, use GetScode()....
10. In its current state, AVIFile is not finished.
Some things which are bad about the current AVIFile APIs:
AVISaveOptions() puts up a nasty big nested dialog which isn't
user-friendly.
AVISave() doesn't handle palette changes correctly.
Almost nothing handles the concept of, say, a wave format changing
halfway through a stream.
T here is no way to find when the next palette change happens, short
of asking for the video format on every frame until you find a place
it changes.
Dealing with compressed and uncompressed streams is kind of screwy.
If you know exactly what you have and what you want, you can call
AVIMakeCompressedStream() to do what you need.
Right now, the handler for compressing streams only supports reading,
not writing. There is no reason this should be true, and maybe I'll
fix it.
There are no status callbacks for anything. If something takes a
really long time, you wait.
Error returns are generally stupid, if they're there at all.
There is no way to find what a specific handler can do short of
trying to do it and seeing what works.
The documentation is still primitive.
11. Miscellaneous hints:
Be sure to call AVIStreamInit() and AVIStreamExit(). If you don't,
the component object DLL won't get initialized and nothing will work.
Be sure the right information is in your registration database. It
should all get put there automatically, but isn't too resistant to
tampering.
Some functions take pointers to long variables which need to be initialized
to the size of your buffer before you call them. You need to use code like:
cbSize = cbBuffer; AVIStreamReadFormat(pstream, 0, lpBuffer, &cbSize);
After this, cbSize will contain the actual correct size of the format, which
you can then compare to the size you passed in to see if your buffer
was large enough.