2014-09-13 23:40:36 +00:00
|
|
|
/*
|
|
|
|
* 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"
|
|
|
|
|
2016-01-11 17:19:48 +00:00
|
|
|
#include <string.h> // For memchr.
|
|
|
|
|
2014-09-13 23:40:36 +00:00
|
|
|
#ifdef __cplusplus
|
|
|
|
namespace libyuv {
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2016-01-11 17:19:48 +00:00
|
|
|
// Helper function to scan for EOI marker (0xff 0xd9).
|
2020-08-14 16:58:22 +00:00
|
|
|
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;
|
2016-01-11 17:19:48 +00:00
|
|
|
while (it < end) {
|
|
|
|
// TODO(fbarchard): scan for 0xd9 instead.
|
2020-08-14 16:58:22 +00:00
|
|
|
it = (const uint8_t*)(memchr(it, 0xff, end - it));
|
2016-01-11 17:19:48 +00:00
|
|
|
if (it == NULL) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (it[1] == 0xd9) {
|
|
|
|
return LIBYUV_TRUE; // Success: Valid jpeg.
|
|
|
|
}
|
|
|
|
++it; // Skip over current 0xff.
|
|
|
|
}
|
|
|
|
}
|
2020-08-14 16:58:22 +00:00
|
|
|
// ERROR: Invalid jpeg end code not found. Size src_size_mjpg
|
2016-01-11 17:19:48 +00:00
|
|
|
return LIBYUV_FALSE;
|
|
|
|
}
|
|
|
|
|
2014-09-13 23:40:36 +00:00
|
|
|
// Helper function to validate the jpeg appears intact.
|
2020-08-14 16:58:22 +00:00
|
|
|
LIBYUV_BOOL ValidateJpeg(const uint8_t* src_mjpg, size_t src_size_mjpg) {
|
2016-01-11 17:19:48 +00:00
|
|
|
// Maximum size that ValidateJpeg will consider valid.
|
|
|
|
const size_t kMaxJpegSize = 0x7fffffffull;
|
|
|
|
const size_t kBackSearchSize = 1024;
|
2020-08-14 16:58:22 +00:00
|
|
|
if (src_size_mjpg < 64 || src_size_mjpg > kMaxJpegSize || !src_mjpg) {
|
|
|
|
// ERROR: Invalid jpeg size: src_size_mjpg
|
2014-09-13 23:40:36 +00:00
|
|
|
return LIBYUV_FALSE;
|
|
|
|
}
|
2020-08-14 16:58:22 +00:00
|
|
|
// SOI marker
|
|
|
|
if (src_mjpg[0] != 0xff || src_mjpg[1] != 0xd8 || src_mjpg[2] != 0xff) {
|
2014-09-13 23:40:36 +00:00
|
|
|
// ERROR: Invalid jpeg initial start code
|
|
|
|
return LIBYUV_FALSE;
|
|
|
|
}
|
2016-01-11 17:19:48 +00:00
|
|
|
|
|
|
|
// Look for the End Of Image (EOI) marker near the end of the buffer.
|
2020-08-14 16:58:22 +00:00
|
|
|
if (src_size_mjpg > kBackSearchSize) {
|
|
|
|
if (ScanEOI(src_mjpg + src_size_mjpg - kBackSearchSize, kBackSearchSize)) {
|
2016-01-11 17:19:48 +00:00
|
|
|
return LIBYUV_TRUE; // Success: Valid jpeg.
|
2014-09-13 23:40:36 +00:00
|
|
|
}
|
2016-01-11 17:19:48 +00:00
|
|
|
// Reduce search size for forward search.
|
2020-08-14 16:58:22 +00:00
|
|
|
src_size_mjpg = src_size_mjpg - kBackSearchSize + 1;
|
2014-09-13 23:40:36 +00:00
|
|
|
}
|
2016-01-11 17:19:48 +00:00
|
|
|
// Step over SOI marker and scan for EOI.
|
2020-08-14 16:58:22 +00:00
|
|
|
return ScanEOI(src_mjpg + 2, src_size_mjpg - 2);
|
2014-09-13 23:40:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
} // extern "C"
|
|
|
|
} // namespace libyuv
|
|
|
|
#endif
|