121 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			121 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| 
								 | 
							
								//  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_
							 |