Initial community commit
This commit is contained in:
parent
537bcbc862
commit
fc06254474
16440 changed files with 4239995 additions and 2 deletions
120
Src/Plugins/Input/in_dshow/base/checkbmi.h
Normal file
120
Src/Plugins/Input/in_dshow/base/checkbmi.h
Normal file
|
@ -0,0 +1,120 @@
|
|||
// Copyright (c) 1992 - 1997 Microsoft Corporation. All Rights Reserved.
|
||||
|
||||
#ifndef _CHECKBMI_H_
|
||||
#define _CHECKBMI_H_
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
// Helper
|
||||
__inline BOOL MultiplyCheckOverflow(DWORD a, DWORD b, __deref_out_range(==, a * b) DWORD *pab) {
|
||||
*pab = a * b;
|
||||
if ((a == 0) || (((*pab) / a) == b)) {
|
||||
return TRUE;
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
|
||||
// Checks if the fields in a BITMAPINFOHEADER won't generate
|
||||
// overlows and buffer overruns
|
||||
// This is not a complete check and does not guarantee code using this structure will be secure
|
||||
// from attack
|
||||
// Bugs this is guarding against:
|
||||
// 1. Total structure size calculation overflowing
|
||||
// 2. biClrUsed > 256 for 8-bit palettized content
|
||||
// 3. Total bitmap size in bytes overflowing
|
||||
// 4. biSize < size of the base structure leading to accessessing random memory
|
||||
// 5. Total structure size exceeding know size of data
|
||||
//
|
||||
|
||||
__success(return != 0) __inline BOOL ValidateBitmapInfoHeader(
|
||||
const BITMAPINFOHEADER *pbmi, // pointer to structure to check
|
||||
__out_range(>=, sizeof(BITMAPINFOHEADER)) DWORD cbSize // size of memory block containing structure
|
||||
)
|
||||
{
|
||||
DWORD dwWidthInBytes;
|
||||
DWORD dwBpp;
|
||||
DWORD dwWidthInBits;
|
||||
DWORD dwHeight;
|
||||
DWORD dwSizeImage;
|
||||
DWORD dwClrUsed;
|
||||
|
||||
// Reject bad parameters - do the size check first to avoid reading bad memory
|
||||
if (cbSize < sizeof(BITMAPINFOHEADER) ||
|
||||
pbmi->biSize < sizeof(BITMAPINFOHEADER) ||
|
||||
pbmi->biSize > 4096) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Reject 0 size
|
||||
if (pbmi->biWidth == 0 || pbmi->biHeight == 0) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Use bpp of 200 for validating against further overflows if not set for compressed format
|
||||
dwBpp = 200;
|
||||
|
||||
if (pbmi->biBitCount > dwBpp) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Strictly speaking abs can overflow so cast explicitly to DWORD
|
||||
dwHeight = (DWORD)abs(pbmi->biHeight);
|
||||
|
||||
if (!MultiplyCheckOverflow(dwBpp, (DWORD)pbmi->biWidth, &dwWidthInBits)) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Compute correct width in bytes - rounding up to 4 bytes
|
||||
dwWidthInBytes = (dwWidthInBits / 8 + 3) & ~3;
|
||||
|
||||
if (!MultiplyCheckOverflow(dwWidthInBytes, dwHeight, &dwSizeImage)) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Fail if total size is 0 - this catches indivual quantities being 0
|
||||
// Also don't allow huge values > 1GB which might cause arithmetic
|
||||
// errors for users
|
||||
if (dwSizeImage > 0x40000000 ||
|
||||
pbmi->biSizeImage > 0x40000000) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Fail if biClrUsed looks bad
|
||||
if (pbmi->biClrUsed > 256) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
if (pbmi->biClrUsed == 0 && pbmi->biBitCount <= 8 && pbmi->biBitCount > 0) {
|
||||
dwClrUsed = (1 << pbmi->biBitCount);
|
||||
} else {
|
||||
dwClrUsed = pbmi->biClrUsed;
|
||||
}
|
||||
|
||||
// Check total size
|
||||
if (cbSize < pbmi->biSize + dwClrUsed * sizeof(RGBQUAD) +
|
||||
(pbmi->biCompression == BI_BITFIELDS ? 3 * sizeof(DWORD) : 0)) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// If it is RGB validate biSizeImage - lots of code assumes the size is correct
|
||||
if (pbmi->biCompression == BI_RGB || pbmi->biCompression == BI_BITFIELDS) {
|
||||
if (pbmi->biSizeImage != 0) {
|
||||
DWORD dwBits = (DWORD)pbmi->biWidth * (DWORD)pbmi->biBitCount;
|
||||
DWORD dwWidthInBytes = ((DWORD)((dwBits+31) & (~31)) / 8);
|
||||
DWORD dwTotalSize = (DWORD)abs(pbmi->biHeight) * dwWidthInBytes;
|
||||
if (dwTotalSize > pbmi->biSizeImage) {
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif // _CHECKBMI_H_
|
Loading…
Add table
Add a link
Reference in a new issue