|  | /* | 
|  | *  Copyright 2012 The LibYuv Project Authors. All rights reserved. | 
|  | * | 
|  | *  Use of this source code is governed by a BSD-style license | 
|  | *  that can be found in the LICENSE file in the root of the source | 
|  | *  tree. An additional intellectual property rights grant can be found | 
|  | *  in the file PATENTS. All contributing project authors may | 
|  | *  be found in the AUTHORS file in the root of the source tree. | 
|  | */ | 
|  |  | 
|  | #include "libyuv/mjpeg_decoder.h" | 
|  |  | 
|  | #include <string.h>  // For memchr. | 
|  |  | 
|  | #ifdef __cplusplus | 
|  | namespace libyuv { | 
|  | extern "C" { | 
|  | #endif | 
|  |  | 
|  | // Helper function to scan for EOI marker (0xff 0xd9). | 
|  | static LIBYUV_BOOL ScanEOI(const uint8_t* src_mjpg, size_t src_size_mjpg) { | 
|  | if (src_size_mjpg >= 2) { | 
|  | const uint8_t* end = src_mjpg + src_size_mjpg - 1; | 
|  | const uint8_t* it = src_mjpg; | 
|  | while (it < end) { | 
|  | // TODO(fbarchard): scan for 0xd9 instead. | 
|  | it = (const uint8_t*)(memchr(it, 0xff, end - it)); | 
|  | if (it == NULL) { | 
|  | break; | 
|  | } | 
|  | if (it[1] == 0xd9) { | 
|  | return LIBYUV_TRUE;  // Success: Valid jpeg. | 
|  | } | 
|  | ++it;  // Skip over current 0xff. | 
|  | } | 
|  | } | 
|  | // ERROR: Invalid jpeg end code not found. Size src_size_mjpg | 
|  | return LIBYUV_FALSE; | 
|  | } | 
|  |  | 
|  | // Helper function to validate the jpeg appears intact. | 
|  | LIBYUV_BOOL ValidateJpeg(const uint8_t* src_mjpg, size_t src_size_mjpg) { | 
|  | // Maximum size that ValidateJpeg will consider valid. | 
|  | const size_t kMaxJpegSize = 0x7fffffffull; | 
|  | const size_t kBackSearchSize = 1024; | 
|  | if (src_size_mjpg < 64 || src_size_mjpg > kMaxJpegSize || !src_mjpg) { | 
|  | // ERROR: Invalid jpeg size: src_size_mjpg | 
|  | return LIBYUV_FALSE; | 
|  | } | 
|  | // SOI marker | 
|  | if (src_mjpg[0] != 0xff || src_mjpg[1] != 0xd8 || src_mjpg[2] != 0xff) { | 
|  | // ERROR: Invalid jpeg initial start code | 
|  | return LIBYUV_FALSE; | 
|  | } | 
|  |  | 
|  | // Look for the End Of Image (EOI) marker near the end of the buffer. | 
|  | if (src_size_mjpg > kBackSearchSize) { | 
|  | if (ScanEOI(src_mjpg + src_size_mjpg - kBackSearchSize, kBackSearchSize)) { | 
|  | return LIBYUV_TRUE;  // Success: Valid jpeg. | 
|  | } | 
|  | // Reduce search size for forward search. | 
|  | src_size_mjpg = src_size_mjpg - kBackSearchSize + 1; | 
|  | } | 
|  | // Step over SOI marker and scan for EOI. | 
|  | return ScanEOI(src_mjpg + 2, src_size_mjpg - 2); | 
|  | } | 
|  |  | 
|  | #ifdef __cplusplus | 
|  | }  // extern "C" | 
|  | }  // namespace libyuv | 
|  | #endif |