Windows2003-3790/enduser/netmeeting/av/codecs/intel/h263/exbase.cpp
2020-09-30 16:53:55 +02:00

1117 lines
36 KiB
C++

/* *************************************************************************
** INTEL Corporation Proprietary Information
**
** This listing is supplied under the terms of a license
** agreement with INTEL Corporation and may not be copied
** nor disclosed except in accordance with the terms of
** that agreement.
**
** Copyright (c) 1996 Intel Corporation.
** All Rights Reserved.
**
** *************************************************************************
*/
//
////////////////////////////////////////////////////////////////////////////
//
// $Author: mbodart $
// $Date: 17 Mar 1997 08:22:08 $
// $Archive: S:\h26x\src\enc\exbase.cpv $
// $Header: S:\h26x\src\enc\exbase.cpv 1.73 17 Mar 1997 08:22:08 mbodart $
// $Log: S:\h26x\src\enc\exbase.cpv $
//
// Rev 1.73 17 Mar 1997 08:22:08 mbodart
// Minor fixes.
//
// Rev 1.72 11 Mar 1997 13:46:46 JMCVEIGH
// Allow input = 320x240 and output = 320x240 for YUV12. This is
// for snapshot mode.
//
// Rev 1.71 10 Mar 1997 17:34:34 MDUDA
// Put in a check for 9-bit YUV12 and adjusted the internal compress
// structure instead of the input bitmap header info.
//
// Rev 1.70 10 Mar 1997 10:41:20 MDUDA
// Treating inconsistent format/bitwidth as a debug warning. Changing
// bit count to match format.
//
// Rev 1.69 07 Mar 1997 16:00:32 JMCVEIGH
// Added checks for non-NULL lpInst before getting H263PlusState.
// Two separate "suggestions" for image sizes if input size is not
// supported in GetFormat.
//
// Rev 1.68 07 Mar 1997 11:55:44 JMCVEIGH
// Moved query in GetFormat to after we have filled out the output
// format. This is because some apps. will ask for the format and
// then use the returned data, regardless if there was an error.
// Silly apps!
//
// Rev 1.67 07 Mar 1997 09:53:08 mbodart
// Added a call to _clearfp() in the Compress exception handler, so that
// the exception will not reoccur in the caller's code.
//
// Rev 1.66 06 Mar 1997 15:39:26 KLILLEVO
//
// CompressQuery now checks for input/output formats regardless
// of configuration status. Also put in trace support for lparam1 and lparam2.
//
// Rev 1.65 22 Jan 1997 12:17:14 MDUDA
//
// Put in more checking for H263+ option in CompressQuery
// and CompressBegin.
//
// Rev 1.64 22 Jan 1997 08:11:22 JMCVEIGH
// Backward compatibility with crop/stretch for 160x120 and 240x180
// in CompressGetFormat(). Do old way unless we have received the
// H263Plus custom message.
//
// Rev 1.63 13 Jan 1997 10:52:14 JMCVEIGH
//
// Added NULL pointer checks in all functions that interface with
// application.
//
// Rev 1.62 09 Jan 1997 13:50:50 MDUDA
// Removed some _CODEC_STATS stuff.
//
// Rev 1.61 06 Jan 1997 17:42:30 JMCVEIGH
// If H263Plus message is not sent, encoder only supports standard
// frame sizes (sub-QCIF, QCIF, or CIF along with special cases),
// as before.
//
// Rev 1.60 30 Dec 1996 19:57:04 MDUDA
// Making sure that input formats agree with the bit count field.
//
// Rev 1.59 20 Dec 1996 15:25:28 MDUDA
// Fixed problem where YUV12 was enabled for crop and stretch.
// This feature is only allowed for RGB, YVU9 and YUY2.
//
// Rev 1.58 16 Dec 1996 13:36:08 MDUDA
//
// Modified Compress Instance info for input color convertors.
//
// Rev 1.57 11 Dec 1996 16:01:20 MBODART
// In Compress, catch any exceptions and return an error code. This gives
// upstream active movie filters a chance to recover gracefully.
//
// Rev 1.56 09 Dec 1996 17:59:36 JMCVEIGH
// Added support for arbitrary frame size support.
// 4 <= width <= 352, 4 <= height <= 288, both multiples of 4.
// Normally, application will pass identical (arbitrary) frame
// sizes in lParam1 and lParam2 of CompressBegin(). If
// cropping/stretching desired to convert to standard frame sizes,
// application should pass the desired output size in lParam2 and
// the input size in lParam1.
//
// Rev 1.55 09 Dec 1996 09:50:12 MDUDA
//
// Allowing 240x180 and 160x120 (crop and stretch) for YUY2.
// Modified _CODEC_STATS stuff.
//
// Rev 1.54 07 Nov 1996 14:45:16 RHAZRA
// Added buffer size adjustment to H.261 CompressGetSize() function
//
// Rev 1.53 31 Oct 1996 22:33:32 BECHOLS
// Decided buffer arbitration must be done in cxq_main.cpp for RTP.
//
// Rev 1.52 31 Oct 1996 21:55:50 BECHOLS
// Added fudge factor for RTP waiting for Raj to decide what he wants to do.
//
// Rev 1.51 31 Oct 1996 10:05:46 KLILLEVO
// changed from DBOUT to DbgLog
//
// Rev 1.50 18 Oct 1996 14:35:46 MDUDA
//
// Separated CompressGetSize and CompressQuery for H261 and H263 cases.
//
// Rev 1.49 11 Oct 1996 16:05:16 MDUDA
//
// Added initial _CODEC_STATS stuff.
//
// Rev 1.48 16 Sep 1996 16:50:52 CZHU
// Return larger size for GetCompressedSize when RTP is enabled.
//
// Rev 1.47 13 Aug 1996 10:36:46 MDUDA
//
// Now allowing RGB4 input format.
//
// Rev 1.46 09 Aug 1996 09:43:30 MDUDA
// Now allowing RGB16 format on input. This is generated by the color Quick Ca
//
// Rev 1.45 02 Aug 1996 13:45:58 MDUDA
//
// Went back to previous version that allows RGB8 and RGB24 in
// 240x180 and 160x120 frames.
//
// Rev 1.44 01 Aug 1996 11:54:58 BECHOLS
// Cut & Paste Error.
//
// Rev 1.43 01 Aug 1996 11:20:28 BECHOLS
// Fixed handling of RGB 24 bit stuff so that it doesn't allow sizes other
// than QCIF, SQCIF, or CIF. I broke this earlier when I added the RGB 8
// bit support. ...
//
// Rev 1.42 22 Jul 1996 13:31:16 BECHOLS
//
// Added code to allow a CLUT8 input providing that the input resolutions
// are either 240x180 or 160x120.
//
// Rev 1.41 11 Jul 1996 15:43:58 MDUDA
// Added support for YVU9 240 x 180 and 160 x 120 for H263 only.
// We now produce subQCIF for 160x120 and QCIF for 240x180.
//
// Rev 1.40 05 Jun 1996 10:57:54 AKASAI
// Added #ifndef H261 in CompressQuery to make sure that H.261 will
// only support FCIF and QCIF input image sizes. All other input sizes
// should return ICERR_BADFORMAT.
//
// Rev 1.39 30 May 1996 17:02:34 RHAZRA
// Added SQCIF support for H.263 in CompressGetSize()
//
// Rev 1.38 06 May 1996 12:47:40 BECHOLS
// Changed the structure element to unBytesPerSecond.
//
// Rev 1.37 06 May 1996 00:09:44 BECHOLS
// Changed the handling of the CompressFramesInfo message to get DataRate
// from the configuration data if the configuration has the data, and
// we haven't received a CompressBegin message yet.
//
// Rev 1.36 23 Apr 1996 16:51:20 KLILLEVO
// moved paranthesis to fix format check in CompressQuery()
//
// Rev 1.35 18 Apr 1996 16:07:10 RHAZRA
// Fixed CompressQuery to keep compiler happy for the non-MICROSOFT version
//
// Rev 1.34 18 Apr 1996 15:57:46 BECHOLS
// RAJ- Changed the query logic to correctly filter the allowable resolutions
// for compression.
//
// Rev 1.33 12 Apr 1996 14:15:40 RHAZRA
// Added paranthesis in CompressGetSize() to make the ifdef case work
//
// Rev 1.32 12 Apr 1996 13:31:02 RHAZRA
// Added SQCIF support in CompressGetSize() with #ifdef SUPPORT_SQCIF;
// changed CompressGetSize() to return 0 if the input format is not
// supported.
//
// Rev 1.31 10 Apr 1996 16:53:08 RHAZRA
// Added a error return in CompressGetSize() to keep complier smiling...
//
// Rev 1.30 10 Apr 1996 16:39:56 RHAZRA
// Added a check for the 320x240 size in CompressGetSize() function;
// added a ifndef to disable certain sizes and compression formats.
//
// Rev 1.29 04 Apr 1996 13:35:00 RHAZRA
// Changed CompressGetSize() to return spec-compliant buffer sizes.
//
// Rev 1.28 03 Apr 1996 08:39:52 SCDAY
// Added H261 specific code to CompressGetSize to limit buffer size
// as defined in H261 spec
//
// Rev 1.27 21 Feb 1996 11:43:12 SCDAY
// cleaned up compiler build warning by changing conversion frlDataRate to (U3
//
// Rev 1.26 15 Feb 1996 16:03:36 RHAZRA
//
// Added a check for NULL lpInst pointer in CompressGetFormat()
//
// Rev 1.25 02 Feb 1996 18:53:46 TRGARDOS
// Changed code to read frame rate from Compressor Instance
// instead of the hack from Quality field.
//
// Rev 1.24 26 Jan 1996 09:35:32 TRGARDOS
// Added #ifndef H261 for 160x120,320x240 support.
//
// Rev 1.23 04 Jan 1996 18:36:54 TRGARDOS
// Added code to permit 320x240 input and then set a boolean
// bIs320x240.
//
// Rev 1.22 27 Dec 1995 15:32:50 RMCKENZX
// Added copyright notice
//
///////////////////////////////////////////////////////////////////////////
#include "precomp.h"
#ifdef YUV9FROMFILE
PAVIFILE paviFile;
PAVISTREAM paviStream;
U8 huge * glpTmp;
HGLOBAL hgMem;
#endif
;////////////////////////////////////////////////////////////////////////////
;// Function: DWORD PASCAL CompressGetFormat(LPCODINST, LPBITMAPINFOHEADER, LPBITMAPINFOHEADER);
;//
;// Description: Added header. This function returns a format that
;// we can deliver back to the caller.
;//
;// History: 05/11/94 -BEN-
;////////////////////////////////////////////////////////////////////////////
#ifdef USE_BILINEAR_MSH26X
DWORD PASCAL CompressGetFormat(LPINST pi, LPBITMAPINFOHEADER lParam1, LPBITMAPINFOHEADER lParam2)
#else
DWORD PASCAL CompressGetFormat(LPCODINST lpInst, LPBITMAPINFOHEADER lParam1, LPBITMAPINFOHEADER lParam2)
#endif
{
DWORD dwQuery;
#ifdef USE_BILINEAR_MSH26X
LPCODINST lpInst = (LPCODINST)pi->CompPtr;
#endif
FX_ENTRY("CompressGetFormat")
// lpInst == NULL is OK
// this is what you get on ICOpen(...,ICMODE_QUERY)
#if 0
if (lpInst == NULL) {
ERRORMESSAGE(("%s: got a NULL lpInst pointer\r\n", _fx_));
return ((DWORD) ICERR_ERROR);
}
#endif
#ifdef USE_BILINEAR_MSH26X
if(dwQuery = CompressQuery(pi, lParam1, NULL)) {
#else
if(dwQuery = CompressQuery(lpInst, lParam1, NULL)) {
#endif
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return(dwQuery);
}
if(lParam2 == NULL) {
// he just want me to return the output buffer size.
return ((DWORD)sizeof(BITMAPINFOHEADER));
}
// Check pointer
if (!lParam1)
return ICERR_ERROR;
// give him back what he passed with our stuff in it
#ifndef WIN32
(void)_fmemcpy(lParam2, lParam1,sizeof(BITMAPINFOHEADER));
#else
(void)memcpy(lParam2, lParam1,sizeof(BITMAPINFOHEADER));
#endif
lParam2->biBitCount = 24;
#ifdef USE_BILINEAR_MSH26X
lParam2->biCompression = pi->fccHandler;
#else
lParam2->biCompression = FOURCC_H263;
#endif
#if defined(H263P)
BOOL bH263PlusState = FALSE;
if (lpInst)
CustomGetH263PlusState(lpInst, (DWORD FAR *)&bH263PlusState);
if (!bH263PlusState) {
// For backward compatibility, make sure the crop and stretch cases are covered.
if ( (lParam1->biCompression == FOURCC_YVU9) ||
(lParam1->biCompression == FOURCC_YUY2) ||
(lParam1->biCompression == FOURCC_UYVY) ||
(lParam1->biCompression == FOURCC_YUV12) ||
(lParam1->biCompression == FOURCC_IYUV) ||
(lParam1->biCompression == BI_RGB) )
{
if ( (lParam1->biWidth == 240) && (lParam1->biHeight == 180) )
{
lParam2->biWidth = 176;
lParam2->biHeight = 144;
}
if ( (lParam1->biWidth == 160) && (lParam1->biHeight == 120) )
{
lParam2->biWidth = 128;
lParam2->biHeight = 96;
}
}
}
#else
if ( (lParam1->biCompression == FOURCC_YVU9) ||
(lParam1->biCompression == FOURCC_YUY2) ||
(lParam1->biCompression == FOURCC_UYVY) ||
(lParam1->biCompression == FOURCC_YUV12) ||
(lParam1->biCompression == FOURCC_IYUV) ||
(lParam1->biCompression == BI_RGB) )
{
if ( (lParam1->biWidth == 240) && (lParam1->biHeight == 180) )
{
lParam2->biWidth = 176;
lParam2->biHeight = 144;
}
if ( (lParam1->biWidth == 160) && (lParam1->biHeight == 120) )
{
lParam2->biWidth = 128;
lParam2->biHeight = 96;
}
}
else
{
lParam2->biWidth = MOD4(lParam1->biWidth);
lParam2->biHeight = MOD4(lParam1->biHeight);
}
#endif
lParam2->biClrUsed = 0;
lParam2->biClrImportant = 0;
lParam2->biPlanes = 1;
#ifdef USE_BILINEAR_MSH26X
lParam2->biSizeImage = CompressGetSize(pi, lParam1, lParam2);
#else
lParam2->biSizeImage = CompressGetSize(lpInst, lParam1, lParam2);
#endif
return(ICERR_OK);
}
;////////////////////////////////////////////////////////////////////////////
;// Function: DWORD PASCAL CompressGetSize(LPCODINST, LPBITMAPINFOHEADER, LPBITMAPINFOHEADER);
;//
;// Description: Added header. This function returns the maximum
;// size that a compressed buffer can be. This size is
;// guaranteed in encoder design.
;//
;// History: 05/11/94 -BEN-
;////////////////////////////////////////////////////////////////////////////
#if defined(H261)
DWORD PASCAL CompressGetSize(LPCODINST lpInst, LPBITMAPINFOHEADER lParam1, LPBITMAPINFOHEADER lParam2)
{
// RH: For QCIF and CIF, the maximum buffer sizes for 261 & 263 are identical.
DWORD dwRet = 0;
DWORD dwExtSize=0;
FX_ENTRY("CompressGetSize")
if ((lParam1->biWidth == 176) && (lParam1->biHeight == 144)) {
dwRet = 8192L;
} else {
if ((lParam1->biWidth == 352) && (lParam1->biHeight == 288)) {
dwRet = 32768L;
}
else // unsupported frame size; should not happen
{
ERRORMESSAGE(("%s: ICERR_BADIMAGESIZE\r\n", _fx_));
dwRet = 0;
}
}
#if 0
// Adjust the buffer size for RTP. Note that this adjustment will be performed
// only if the codec has been told previously to use RTP and the RTP-related
// information has been initialized. Therefore, the current (11/7) AM interface
// will not take advantage of this routine.
if (dwRet && lpInst && lpInst->Configuration.bRTPHeader && lpInst->Configuration.bInitialized)
{
dwRet += H261EstimateRTPOverhead(lpInst, lParam1);
}
#endif
return dwRet;
}
#else
/* H.263 case */
#ifdef USE_BILINEAR_MSH26X
DWORD PASCAL CompressGetSize(LPINST pi, LPBITMAPINFOHEADER lParam1, LPBITMAPINFOHEADER lParam2)
#else
DWORD PASCAL CompressGetSize(LPCODINST lpInst, LPBITMAPINFOHEADER lParam1, LPBITMAPINFOHEADER lParam2)
#endif
{
// RH: For QCIF and CIF, the maximum buffer sizes for 261 & 263 are identical.
#ifdef USE_BILINEAR_MSH26X
LPCODINST lpInst = (LPCODINST)pi->CompPtr;
#endif
DWORD dwRet = 0;
DWORD dwExtSize=0;
FX_ENTRY("CompressGetSize")
if (lParam1 == NULL) {
// We will use a size of zero to indicate an error for CompressGetSize
ERRORMESSAGE(("%s: got a NULL lParam1 pointer\r\n", _fx_));
dwRet = 0;
return dwRet;
}
#ifndef H263P
#ifdef USE_BILINEAR_MSH26X
if (pi->fccHandler == FOURCC_H26X)
{
// H.263+
U32 unPaddedWidth;
U32 unPaddedHeight;
U32 unSourceFormatSize;
// Base buffer size on frame dimensions padded to multiples of 16
if (lParam2 == NULL)
{
// In case an old application passed in a NULL pointer in lParam2,
// we use the input frame dimensions to calculate the format size
unPaddedWidth = (lParam1->biWidth + 0xf) & ~0xf;
unPaddedHeight = (lParam1->biHeight + 0xf) & ~0xf;
}
else
{
unPaddedWidth = (lParam2->biWidth + 0xf) & ~0xf;
unPaddedHeight = (lParam2->biHeight + 0xf) & ~0xf;
}
unSourceFormatSize = unPaddedWidth * unPaddedHeight;
// See Table 1/H.263, document LBC-96-358
if (unSourceFormatSize < 25348)
dwRet = 8192L;
else if (unSourceFormatSize < 101380)
dwRet = 32768L;
else if (unSourceFormatSize < 405508)
dwRet = 65536L;
else
dwRet = 131072L;
}
else
{
#endif
if (((lParam1->biWidth == 128) && (lParam1->biHeight == 96)) ||
#ifdef USE_BILINEAR_MSH26X
((lParam1->biWidth == 80) && (lParam1->biHeight == 64)) ||
#endif
((lParam1->biWidth == 176) && (lParam1->biHeight == 144)) ||
((lParam1->biWidth == 240) && (lParam1->biHeight == 180)) ||
((lParam1->biWidth == 160) && (lParam1->biHeight == 120)))
{
dwRet = 8192L;
}
else if (((lParam1->biWidth == 352) && (lParam1->biHeight == 288)) ||
((lParam1->biWidth == 320) && (lParam1->biHeight == 240)))
{
dwRet = 32768L;
}
else // unsupported frame size; should not happen
{
ERRORMESSAGE(("%s: ICERR_BADIMAGESIZE\r\n", _fx_));
dwRet = 0;
}
#ifdef USE_BILINEAR_MSH26X
}
#endif
#else
// H.263+
U32 unPaddedWidth;
U32 unPaddedHeight;
U32 unSourceFormatSize;
// Base buffer size on frame dimensions padded to multiples of 16
if (lParam2 == NULL)
{
// In case an old application passed in a NULL pointer in lParam2,
// we use the input frame dimensions to calculate the format size
unPaddedWidth = (lParam1->biWidth + 0xf) & ~0xf;
unPaddedHeight = (lParam1->biHeight + 0xf) & ~0xf;
}
else
{
unPaddedWidth = (lParam2->biWidth + 0xf) & ~0xf;
unPaddedHeight = (lParam2->biHeight + 0xf) & ~0xf;
}
unSourceFormatSize = unPaddedWidth * unPaddedHeight;
// See Table 1/H.263, document LBC-96-358
if (unSourceFormatSize < 25348)
dwRet = 8192L;
else if (unSourceFormatSize < 101380)
dwRet = 32768L;
else if (unSourceFormatSize < 405508)
dwRet = 65536L;
else
dwRet = 131072L;
#endif
#if 0
//adjust if RTP is enabled, based on information in Configuration
//Size calculated using DataRate, FrameRate in lpInst,
//and lpInst->Configuration.unPacketSize;
//Chad, 9/12/96
if (dwRet && lpInst &&
lpInst->Configuration.bRTPHeader && lpInst->Configuration.bInitialized)
{
dwRet += getRTPBsInfoSize(lpInst);
}
#endif
return dwRet;
}
#endif
;////////////////////////////////////////////////////////////////////////////
;// Function: DWORD PASCAL CompressQuery(LPCODINST, LPBITMAPINFOHEADER, LPBITMAPINFOHEADER);
;//
;// Description:
;//
;// History: 05/11/94 -BEN-
;////////////////////////////////////////////////////////////////////////////
#if defined(H261)
DWORD PASCAL CompressQuery(LPCODINST lpInst, LPBITMAPINFOHEADER lParam1, LPBITMAPINFOHEADER lParam2)
{
// Check for good input format
FX_ENTRY("CompressQuery")
if(NULL == lParam1)
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
if( (lParam1->biCompression != BI_RGB) &&
(lParam1->biCompression != FOURCC_YUV12) &&
(lParam1->biCompression != FOURCC_IYUV) )
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
if( (lParam1->biCompression == BI_RGB) && (lParam1->biBitCount != 24))
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
if(! ( ((lParam1->biWidth == 176) && (lParam1->biHeight == 144)) ||
((lParam1->biWidth == 352) && (lParam1->biHeight == 288)) ))
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
if( lParam1->biPlanes != 1 )
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
if(0 == lParam2) // Checking input only
return(ICERR_OK);
// TODO: Do we want to check frame dimensions of output?
if( lParam2->biCompression != FOURCC_H263 )
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return ((DWORD)ICERR_BADFORMAT);
}
return(ICERR_OK);
}
#else
/* H.263 case */
#ifdef USE_BILINEAR_MSH26X
DWORD PASCAL CompressQuery(LPINST pi, LPBITMAPINFOHEADER lParam1, LPBITMAPINFOHEADER lParam2)
#else
DWORD PASCAL CompressQuery(LPCODINST lpInst, LPBITMAPINFOHEADER lParam1, LPBITMAPINFOHEADER lParam2)
#endif
{
#ifdef USE_BILINEAR_MSH26X
LPCODINST lpInst = (LPCODINST)pi->CompPtr;
#endif
FX_ENTRY("CompressQuery")
#if defined(H263P)
BOOL bH263PlusState = FALSE;
if (lpInst)
CustomGetH263PlusState(lpInst, (DWORD FAR *)&bH263PlusState);
#endif
// Check for good input format
if(lParam1 == NULL)
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
if( (lParam1->biCompression != BI_RGB) &&
(lParam1->biCompression != FOURCC_YVU9) &&
(lParam1->biCompression != FOURCC_YUV12) &&
(lParam1->biCompression != FOURCC_IYUV) &&
(lParam1->biCompression != FOURCC_UYVY) &&
(lParam1->biCompression != FOURCC_YUY2) )
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
if( (lParam1->biCompression == BI_RGB) &&
( (lParam1->biBitCount != 24) &&
#ifdef H263P
(lParam1->biBitCount != 32) &&
#endif
(lParam1->biBitCount != 16) &&
(lParam1->biBitCount != 8) &&
(lParam1->biBitCount != 4) ) )
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
#ifndef H263P
#ifdef USE_BILINEAR_MSH26X
if (pi->fccHandler == FOURCC_H26X)
{
if ((lParam1->biWidth & 0x3) || (lParam1->biHeight & 0x3) ||
(lParam1->biWidth < 4) || (lParam1->biWidth > 352) ||
(lParam1->biHeight < 4) || (lParam1->biHeight > 288))
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
}
else
{
#endif
if(!
( ((lParam1->biWidth == 128) && (lParam1->biHeight == 96)) ||
((lParam1->biWidth == 176) && (lParam1->biHeight == 144)) ||
#ifdef USE_BILINEAR_MSH26X
((lParam1->biWidth == 80) && (lParam1->biHeight == 64)) ||
#endif
((lParam1->biWidth == 352) && (lParam1->biHeight == 288))
#ifndef MICROSOFT
||
( ( (lParam1->biCompression == FOURCC_YVU9) ||
(lParam1->biCompression == FOURCC_YUY2) ||
(lParam1->biCompression == FOURCC_UYVY) ||
(lParam1->biCompression == FOURCC_YUV12) ||
(lParam1->biCompression == FOURCC_IYUV) ||
(lParam1->biCompression == BI_RGB) )
&& ((lParam1->biWidth == 160) && (lParam1->biHeight == 120)) )
||
( ( (lParam1->biCompression == FOURCC_YVU9) ||
(lParam1->biCompression == FOURCC_YUY2) ||
(lParam1->biCompression == FOURCC_UYVY) ||
(lParam1->biCompression == FOURCC_YUV12) ||
(lParam1->biCompression == FOURCC_IYUV) ||
(lParam1->biCompression == BI_RGB) )
&& ((lParam1->biWidth == 240) && (lParam1->biHeight == 180)) )
||
( ( (lParam1->biCompression == FOURCC_YUV12) || (lParam1->biCompression == FOURCC_IYUV) )
&& ((lParam1->biWidth == 320) && (lParam1->biHeight == 240)) )
#endif
))
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
#ifdef USE_BILINEAR_MSH26X
}
#endif
#else
if (((FOURCC_YVU9 == lParam1->biCompression) && (9 != lParam1->biBitCount)) ||
((FOURCC_YUY2 == lParam1->biCompression) && (16 != lParam1->biBitCount)) ||
((FOURCC_UYVY == lParam1->biCompression) && (16 != lParam1->biBitCount)) ||
// The following check for 9-bit YUV12 is a hack to work around a VPhone 1.x bug.
((FOURCC_YUV12 == lParam1->biCompression) &&
!((12 == lParam1->biBitCount) || (9 == lParam1->biBitCount))) ||
((FOURCC_IYUV == lParam1->biCompression) &&
!((12 == lParam1->biBitCount) || (9 == lParam1->biBitCount)))) {
ERRORMESSAGE(("%s: Incorrect bit width (ICERR_BADFORMAT)\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
// The H263+ message indicates whether arbitrary frame
// sizes are to be supported. If arbitrary frames are needed,
// the H263+ message must be sent before the first call to
// CompressQuery.
if (bH263PlusState) {
if ((lParam1->biWidth & 0x3) || (lParam1->biHeight & 0x3) ||
(lParam1->biWidth < 4) || (lParam1->biWidth > 352) ||
(lParam1->biHeight < 4) || (lParam1->biHeight > 288)) {
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
} else {
if(!
( ((lParam1->biWidth == 128) && (lParam1->biHeight == 96)) ||
((lParam1->biWidth == 176) && (lParam1->biHeight == 144)) ||
((lParam1->biWidth == 352) && (lParam1->biHeight == 288)) ||
( ( (lParam1->biCompression == FOURCC_YVU9) ||
(lParam1->biCompression == FOURCC_YUY2) ||
(lParam1->biCompression == FOURCC_UYVY) ||
(lParam1->biCompression == FOURCC_YUV12) ||
(lParam1->biCompression == FOURCC_IYUV) ||
(lParam1->biCompression == BI_RGB) )
&& ((lParam1->biWidth == 160) && (lParam1->biHeight == 120)) ) ||
( ( (lParam1->biCompression == FOURCC_YVU9) ||
(lParam1->biCompression == FOURCC_YUY2) ||
(lParam1->biCompression == FOURCC_UYVY) ||
(lParam1->biCompression == FOURCC_YUV12) ||
(lParam1->biCompression == FOURCC_IYUV) ||
(lParam1->biCompression == BI_RGB) )
&& ((lParam1->biWidth == 240) && (lParam1->biHeight == 180)) ) ||
( ( (lParam1->biCompression == FOURCC_YUV12) || (lParam1->biCompression == FOURCC_IYUV) )
&& ((lParam1->biWidth == 320) && (lParam1->biHeight == 240)) ) ))
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
}
#endif
if( lParam1->biPlanes != 1 )
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return((DWORD)ICERR_BADFORMAT);
}
if(lParam2 == 0) // Checking input only
return(ICERR_OK);
// TODO: Do we want to check frame dimensions of output?
#ifdef USE_BILINEAR_MSH26X
if( (lParam2->biCompression != FOURCC_H263) && (lParam2->biCompression != FOURCC_H26X) )
#else
if( lParam2->biCompression != FOURCC_H263 )
#endif
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return ((DWORD)ICERR_BADFORMAT);
}
#if defined(H263P)
if (bH263PlusState) {
if ((lParam1->biWidth != lParam2->biWidth) ||
(lParam1->biHeight != lParam2->biHeight)) {
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return ((DWORD)ICERR_BADFORMAT);
}
} else {
if(!
(( ( ((lParam1->biWidth == 128) && (lParam1->biHeight == 96)) ||
((lParam1->biWidth == 176) && (lParam1->biHeight == 144)) ||
((lParam1->biWidth == 352) && (lParam1->biHeight == 288)) ) &&
(lParam1->biWidth == lParam2->biWidth) && (lParam1->biHeight == lParam2->biHeight) ) ||
(((lParam1->biCompression == FOURCC_YVU9) ||
(lParam1->biCompression == FOURCC_YUY2) ||
(lParam1->biCompression == FOURCC_UYVY) ||
(lParam1->biCompression == FOURCC_YUV12) ||
(lParam1->biCompression == FOURCC_IYUV) ||
(lParam1->biCompression == BI_RGB)) &&
(((lParam1->biWidth == 160) && (lParam1->biHeight == 120)) &&
((lParam2->biWidth == 128) && (lParam2->biHeight == 96)))) ||
(((lParam1->biCompression == FOURCC_YVU9) ||
(lParam1->biCompression == FOURCC_YUY2) ||
(lParam1->biCompression == FOURCC_UYVY) ||
(lParam1->biCompression == FOURCC_YUV12) ||
(lParam1->biCompression == FOURCC_IYUV) ||
(lParam1->biCompression == BI_RGB)) &&
(((lParam1->biWidth == 240) && (lParam1->biHeight == 180)) &&
((lParam2->biWidth == 176) && (lParam2->biHeight == 144)))) ||
(((lParam1->biCompression == FOURCC_YUV12) || (lParam1->biCompression == FOURCC_IYUV)) &&
(((lParam1->biWidth == 320) && (lParam1->biHeight == 240)) &&
((lParam2->biWidth == 320) && (lParam2->biHeight == 240)))) ) )
{
ERRORMESSAGE(("%s: ICERR_BADFORMAT\r\n", _fx_));
return ((DWORD)ICERR_BADFORMAT);
}
}
#endif
return(ICERR_OK);
}
#endif
;////////////////////////////////////////////////////////////////////////////
;// Function: DWORD PASCAL CompressQuery(LPCODINST, LPBITMAPINFOHEADER, LPBITMAPINFOHEADER);
;//
;// Description:
;//
;// History: 05/11/94 -BEN-
;////////////////////////////////////////////////////////////////////////////
DWORD PASCAL CompressFramesInfo(LPCODINST lpCompInst, ICCOMPRESSFRAMES *lParam1, int lParam2)
{
FX_ENTRY("CompressFramesInfo");
// Check to see if we are given a nonzero pointer.
if (lpCompInst == NULL)
{
ERRORMESSAGE(("%s: CompressFramesInfo called with NULL parameter - returning ICERR_BADFORMAT", _fx_));
return ((DWORD)ICERR_BADFORMAT);
}
// lParam2 should be the size of the structure.
if (lParam2 != sizeof(ICCOMPRESSFRAMES))
{
ERRORMESSAGE(("%s: wrong size of ICOMPRESSFRAMES structure", _fx_));
return ((DWORD)ICERR_BADFORMAT);
}
if (!lParam1 || (lParam1->dwScale == 0))
{
ERRORMESSAGE(("%s: dwScale is zero", _fx_));
return ((DWORD)ICERR_BADFORMAT);
}
lpCompInst->FrameRate = (float)lParam1->dwRate / (float)lParam1->dwScale;
lpCompInst->DataRate = (U32)lParam1->lDataRate;
DEBUGMSG(ZONE_BITRATE_CONTROL, ("%s: Setting frame rate at %ld.%ld fps and bitrate at %ld bps", _fx_, (DWORD)lpCompInst->FrameRate, (DWORD)((lpCompInst->FrameRate - (float)(DWORD)lpCompInst->FrameRate) * 100.0f), lpCompInst->DataRate * 8UL));
return ((DWORD)ICERR_OK);
}
;////////////////////////////////////////////////////////////////////////////
;// Function: BOOL bIsOkRes(LPCODINST);
;//
;// Description: This function checks whether the desired height and
;// width are possible.
;//
;// History: 05/11/94 -BEN-
;////////////////////////////////////////////////////////////////////////////
BOOL bIsOkRes(LPCODINST lpCompInst)
{
BOOL bRet;
// Check for NULL pointer
if (lpCompInst == NULL)
return 0;
bRet = lpCompInst->xres <= 352
&& lpCompInst->yres <= 288
&& lpCompInst->xres >= 4
&& lpCompInst->yres >= 4
&& (lpCompInst->xres & ~3) == lpCompInst->xres
&& (lpCompInst->yres & ~3) == lpCompInst->yres;
return(bRet);
}
;////////////////////////////////////////////////////////////////////////////
;// Function: DWORD PASCAL CompressBegin(LPCODINST, LPBITMAPINFOHEADER, LPBITMAPINFOHEADER);
;//
;// Description:
;//
;// History: 05/11/94 -BEN-
;////////////////////////////////////////////////////////////////////////////
DWORD PASCAL CompressBegin(
#ifdef USE_BILINEAR_MSH26X
LPINST pi,
#else
LPCODINST lpCompInst,
#endif
LPBITMAPINFOHEADER lParam1,
LPBITMAPINFOHEADER lParam2
)
{
#ifdef USE_BILINEAR_MSH26X
LPCODINST lpCompInst = (LPCODINST)pi->CompPtr;
#endif
DWORD dwQuery;
LRESULT retval;
#if defined(H263P)
BOOL bH263PlusState = FALSE;
if (lpCompInst)
CustomGetH263PlusState(lpCompInst, (DWORD FAR *)&bH263PlusState);
#endif
// Check input and output format.
#ifdef USE_BILINEAR_MSH26X
if( (dwQuery = CompressQuery(pi, lParam1, lParam2)) != ICERR_OK)
#else
if( (dwQuery = CompressQuery(lpCompInst, lParam1, lParam2)) != ICERR_OK)
#endif
return(dwQuery);
// Check instance pointer
if (!lpCompInst || !lParam1)
return ICERR_ERROR;
#if defined(H263P) || defined(USE_BILINEAR_MSH26X)
lpCompInst->InputCompression = lParam1->biCompression;
lpCompInst->InputBitWidth = lParam1->biBitCount;
if (((FOURCC_YUV12 == lParam1->biCompression) || (FOURCC_IYUV == lParam1->biCompression)) && (9 == lParam1->biBitCount)) {
lpCompInst->InputBitWidth = 12;
}
#endif
#if defined(H263P)
if ( lParam2 && bH263PlusState)
{
// This is the "new" style for indicating if the input should
// be cropped/stretched to a standard frame size.
// Old applications may pass in NULL or junk for lparam2.
// New applications should pass a valid lParam2 that indicates
// the desired output frame size. Also, the H263Plus flag must
// be set in the configuration structure before calling CompressBegin()
lpCompInst->xres = (WORD)lParam2->biWidth;
lpCompInst->yres = (WORD)lParam2->biHeight;
} else
#endif // H263P
{
lpCompInst->xres = (WORD)lParam1->biWidth;
lpCompInst->yres = (WORD)lParam1->biHeight;
lpCompInst->Is160x120 = FALSE;
lpCompInst->Is240x180 = FALSE;
lpCompInst->Is320x240 = FALSE;
if ( (lParam1->biWidth == 160) && (lParam1->biHeight == 120) )
{
lpCompInst->xres = 128;
lpCompInst->yres = 96;
lpCompInst->Is160x120 = TRUE;
}
else if ( (lParam1->biWidth == 240) && (lParam1->biHeight == 180) )
{
lpCompInst->xres = 176;
lpCompInst->yres = 144;
lpCompInst->Is240x180 = TRUE;
}
else if ( (lParam1->biWidth == 320) && (lParam1->biHeight == 240) )
{
lpCompInst->xres = 352;
lpCompInst->yres = 288;
lpCompInst->Is320x240 = TRUE;
}
}
if(!bIsOkRes(lpCompInst))
return((DWORD)ICERR_BADIMAGESIZE);
// Set frame size.
if (lpCompInst->xres == 128 && lpCompInst->yres == 96)
lpCompInst->FrameSz = SQCIF;
else if (lpCompInst->xres == 176 && lpCompInst->yres == 144)
lpCompInst->FrameSz = QCIF;
else if (lpCompInst->xres == 352 && lpCompInst->yres == 288)
lpCompInst->FrameSz = CIF;
#ifdef USE_BILINEAR_MSH26X
else if (pi->fccHandler == FOURCC_H26X)
lpCompInst->FrameSz = fCIF;
#endif
#ifdef H263P
else
lpCompInst->FrameSz = CUSTOM;
#else
else // unsupported frame size.
return (DWORD)ICERR_BADIMAGESIZE;
#endif
// Allocate and Initialize tables and memory that are specific to
// this instance.
#if defined(H263P) || defined(USE_BILINEAR_MSH26X)
retval = H263InitEncoderInstance(lParam1,lpCompInst);
#else
retval = H263InitEncoderInstance(lpCompInst);
#endif
return(retval);
}
;////////////////////////////////////////////////////////////////////////////
;// Function: DWORD PASCAL CompressEnd(LPCODINST);
;//
;// Description:
;//
;// History: 05/11/94 -BEN-
;////////////////////////////////////////////////////////////////////////////
DWORD PASCAL CompressEnd(LPCODINST lpInst)
{
LRESULT retval;
retval = H263TermEncoderInstance(lpInst);
return(retval);
}
;////////////////////////////////////////////////////////////////////////////
;// Function: DWORD PASCAL Compress(LPCODINST, ICCOMPRESS FAR *, DWORD);
;//
;// Description:
;//
;// History: 05/11/94 -BEN-
;////////////////////////////////////////////////////////////////////////////
DWORD PASCAL Compress(
#ifdef USE_BILINEAR_MSH26X
LPINST pi,
#else
LPCODINST lpInst, // ptr to Compressor instance information.
#endif
ICCOMPRESS FAR * lpCompInfo, // ptr to ICCOMPRESS structure.
DWORD dOutbufSize // size, in bytes, of the ICCOMPRESS structure.
)
{
#ifdef USE_BILINEAR_MSH26X
LPCODINST lpInst = (LPCODINST)pi->CompPtr; // ptr to Compressor instance information.
#endif
DWORD dwRet;
FX_ENTRY("Compress")
// Check to see if we are given a NULL pointer.
if(lpInst == NULL || lpCompInfo == NULL)
{
ERRORMESSAGE(("%s: called with NULL parameter\r\n", _fx_));
return( (DWORD) ICERR_ERROR );
}
try
{
#ifdef USE_BILINEAR_MSH26X
dwRet = H263Compress(pi, lpCompInfo);
#else
dwRet = H263Compress(lpInst, lpCompInfo);
#endif
}
catch (...)
{
// For a DEBUG build, display a message and pass the exception up.
// For a release build, stop the exception here and return an error
// code. This gives upstream code a chance to gracefully recover.
// We also need to clear the floating point control word, otherwise
// the upstream code may incur an exception the next time it tries
// a floating point operation (presuming this exception was due
// to a floating point problem).
#if defined(DEBUG) || defined(_DEBUG)
ERRORMESSAGE(("%s: Exception occured!!!\r\n", _fx_));
throw;
#else
_clearfp();
return (DWORD) ICERR_ERROR;
#endif
}
if(dwRet != ICERR_OK)
{
ERRORMESSAGE(("%s: Failed!!!\r\n", _fx_));
}
// now transfer the information.
lpCompInfo->lpbiOutput->biSize =sizeof(BITMAPINFOHEADER);
#ifdef USE_BILINEAR_MSH26X
lpCompInfo->lpbiOutput->biCompression = pi->fccHandler;
#else
lpCompInfo->lpbiOutput->biCompression = FOURCC_H263;
#endif
lpCompInfo->lpbiOutput->biPlanes = 1;
lpCompInfo->lpbiOutput->biBitCount = 24;
lpCompInfo->lpbiOutput->biWidth = lpInst->xres;
lpCompInfo->lpbiOutput->biHeight = lpInst->yres;
lpCompInfo->lpbiOutput->biSizeImage = lpInst->CompressedSize;
lpCompInfo->lpbiOutput->biClrUsed = 0;
lpCompInfo->lpbiOutput->biClrImportant = 0;
// lpCompInfo->dwFlags is set inside the compressor.
// set the chunk idea if requested
if (lpCompInfo->lpckid)
{
*(lpCompInfo->lpckid) = TWOCC_H26X;
}
return(dwRet);
}