shithub: openh264

Download patch

ref: f149843f7adbdd39affd549cc65bf1d4f695567f
parent: c0e72338c13f5b2a62e66cba05d0c4c016a8f588
parent: 8e201827f4751b5ecbc4c306b53d8b8cb72d446a
author: huili2 <[email protected]>
date: Wed Jan 22 17:56:39 EST 2014

Merge pull request #190 from licaiguo/add-level-limits

Add level limits for decoder

--- a/codec/decoder/core/inc/parameter_sets.h
+++ b/codec/decoder/core/inc/parameter_sets.h
@@ -39,6 +39,18 @@
 
 namespace WelsDec {
 
+typedef struct TagLevelLimits {
+  int32_t iMaxMBPS; // Max macroblock processing rate(MB/s)
+  int32_t iMaxFS;   // Max frame sizea(MBs)
+  int32_t iMaxDPBMbs;// Max decoded picture buffer size(MBs)
+  int32_t iMaxBR; // Max video bit rate
+  int32_t iMaxCPB; // Max CPB size
+  int16_t iMinVmv; // Vertical MV component range upper bound
+  int16_t iMaxVmv; // Vertical MV component range lower bound
+  int16_t iMinCR;  // Min compression ration
+  int16_t iMaxMvsPer2Mb; // Max number of motion vectors per two consecutive MBs
+} SLevelLimits;
+
 /* Sequence Parameter Set, refer to Page 57 in JVT X201wcm */
 typedef struct TagSps {
 int32_t	    iSpsId;
@@ -87,6 +99,7 @@
 bool_t		bQpPrimeYZeroTransfBypassFlag;
 bool_t		bSeqScalingMatrixPresentFlag;
 bool_t		bSeqScalingListPresentFlag[12];
+const SLevelLimits *pSLevelLimits;
 } SSps, *PSps;
 
 
--- a/codec/decoder/core/src/au_parser.cpp
+++ b/codec/decoder/core/src/au_parser.cpp
@@ -481,6 +481,7 @@
   int32_t iErr				= ERR_NONE;
   if (kiSrcLen <= 0)
     return iErr;
+
   pBs	     = &pCtx->sBs;	// SBitStringAux instance for non VCL NALs decoding
   iBitSize = (kiSrcLen << 3) - BsGetTrailingBits (pRbsp + kiSrcLen - 1); // convert into bit
   eNalType = pCtx->sCurNalHead.eNalUnitType;
@@ -629,6 +630,69 @@
   return 0;
 }
 
+// table A-1 - Level limits
+static const SLevelLimits g_kSLevelLimits[17] = {
+  {1485, 99, 396, 64, 175, -256, 255, 2, 0x7fff}, /* level 1 */
+  {1485, 99, 396, 128, 350, -256, 255, 2, 0x7fff}, /* level 1.b */
+  {3000, 396, 900, 192, 500, -512, 511, 2, 0x7fff}, /* level 1.1 */
+  {6000, 396, 2376, 384, 1000, -512, 511, 2, 0x7fff}, /* level 1.2 */
+  {11880, 396, 2376, 768, 2000, -512, 511, 2, 0x7fff}, /* level 1.3 */
+  {11880, 396, 2376, 2000, 2000, -512, 511, 2, 0x7fff}, /* level 2 */
+  {19800, 792, 4752, 4000, 4000, -1024, 1023, 2, 0x7fff}, /* level 2.1 */
+  {20250, 1620, 8100, 4000, 4000, -1024, 1023, 2, 0x7fff}, /* level 2.2 */
+  {40500, 1620, 8100, 10000, 10000, -1024, 1023, 2, 32 }, /* level 3 */
+  {108000, 3600, 18000, 14000, 14000, -2048, 2047, 4, 16}, /* level 3.1 */
+  {216000, 5120, 20480, 20000, 20000, -2048, 2047, 4, 16}, /* level 3.2 */
+  {245760, 8192, 32768, 20000, 25000, -2048, 2047, 4, 16}, /* level 4 */
+  {245760, 8192, 32768, 50000, 62500, -2048, 2047, 2, 16}, /* level 4.1 */
+  {522240, 8704, 34816, 50000, 62500, -2048, 2047, 2, 16}, /* level 4.2 */
+  {589824, 22080, 110400, 135000, 135000, -2048, 2047, 2, 16}, /* level 5 */
+  {983040, 36864, 184320, 240000, 240000, -2048, 2047, 2, 16}, /* level 5.1 */
+  {2073600, 36864, 184320, 240000, 240000, -2048, 2047, 2, 16} /* level 5.2 */
+};
+
+const SLevelLimits *GetLevelLimits(int32_t iLevelIdx, bool_t bConstraint3) {
+  switch (iLevelIdx) {
+  case 10:
+    return &g_kSLevelLimits[0];
+  case 11:
+    if(bConstraint3)
+      return &g_kSLevelLimits[1];
+    else
+      return &g_kSLevelLimits[2];
+  case 12:
+    return &g_kSLevelLimits[3];
+  case 13:
+    return &g_kSLevelLimits[4];
+  case 20:
+    return &g_kSLevelLimits[5];
+  case 21:
+    return &g_kSLevelLimits[6];
+  case 22:
+    return &g_kSLevelLimits[7];
+  case 30:
+    return &g_kSLevelLimits[8];
+  case 31:
+    return &g_kSLevelLimits[9];
+  case 32:
+    return &g_kSLevelLimits[10];
+  case 40:
+    return &g_kSLevelLimits[11];
+  case 41:
+    return &g_kSLevelLimits[12];
+  case 42:
+    return &g_kSLevelLimits[13];
+  case 50:
+    return &g_kSLevelLimits[14];
+  case 51:
+    return &g_kSLevelLimits[15];
+  case 52:
+    return &g_kSLevelLimits[16];
+  default:
+    return NULL;
+  }
+  return NULL;
+}
 /*!
  *************************************************************************************
  * \brief	to parse Sequence Parameter Set (SPS)
@@ -685,10 +749,7 @@
   bConstraintSetFlags[5]	= !!BsGetOneBit (pBs);	// constraint_set5_flag
   BsGetBits (pBs, 2);							// reserved_zero_2bits, equal to 0
   uiLevelIdc	= BsGetBits (pBs, 8);				// level_idc
-
   iSpsId		= BsGetUe (pBs);					// seq_parameter_set_id
-
-
   if (iSpsId >= MAX_SPS_COUNT || iSpsId < 0) {	// Modified to check invalid negative iSpsId, 12/1/2009
     WelsLog (pCtx, WELS_LOG_WARNING, " iSpsId is out of range! \n");
     return GENERATE_ERROR_NO (ERR_LEVEL_PARAM_SETS, ERR_INFO_SPS_ID_OVERFLOW);
@@ -719,7 +780,12 @@
     pCtx->bSpsAvailFlags[iSpsId] = true; // added for EC, 10/28/2009
 #endif //MOSAIC_AVOID_BASED_ON_SPS_PPS_ID
   }
-
+  const SLevelLimits *pSLevelLimits = GetLevelLimits(uiLevelIdc, bConstraintSetFlags[3]);
+  if (NULL == pSLevelLimits) {
+     WelsLog (pCtx, WELS_LOG_WARNING, "ParseSps(): level_idx (%d).\n", uiLevelIdc);
+     return GENERATE_ERROR_NO (ERR_LEVEL_PARAM_SETS, ERR_INFO_UNSUPPORTED_NON_BASELINE);
+  }
+  else pSps->pSLevelLimits = pSLevelLimits;
   // syntax elements in default
   pSps->uiChromaFormatIdc	= 1;
   pSps->uiBitDepthLuma		=