linux-media.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
From: Ang Way Chuang <wcang79@gmail.com>
To: Jarod Wilson <jarod@redhat.com>
Cc: linux-media@vger.kernel.org
Subject: Re: dvb-core: Fix ULE decapsulation bug when less than 4 bytes of ULE SNDU is packed into the remaining bytes of a MPEG2-TS frame
Date: Fri, 21 May 2010 11:40:34 +0800	[thread overview]
Message-ID: <4BF600B2.1080305@gmail.com> (raw)
In-Reply-To: <20100520192213.GA19133@redhat.com>

Hi Jarod,
    Thanks for the review. My answers are inlined.

Jarod Wilson wrote:
> On Thu, May 06, 2010 at 02:52:22PM -0000, Ang Way Chuang wrote:
>> ULE (Unidirectional Lightweight Encapsulation RFC 4326) decapsulation 
>> code has a bug that incorrectly treats ULE SNDU packed into the 
>> remaining 2 or 3 bytes of a MPEG2-TS frame as having invalid pointer 
>> field on the subsequent MPEG2-TS frame.
>>
>> This patch was generated and tested against v2.6.34-rc6. I suspect 
>> that this bug was introduced in kernel version 2.6.15, but had not 
>> verified it.
>>
>> Care has been taken not to introduce more bug by fixing this bug, but
>> please scrutinize the code because I always produces buggy code.
>>
>> Signed-off-by: Ang Way Chuang <wcang@nav6.org>
>>
>> ---
>>
>> diff --git a/drivers/media/dvb/dvb-core/dvb_net.c b/drivers/media/dvb/dvb-core/dvb_net.c
>> index 441c064..35a4afb 100644
>> --- a/drivers/media/dvb/dvb-core/dvb_net.c
>> +++ b/drivers/media/dvb/dvb-core/dvb_net.c
>> @@ -458,8 +458,9 @@ static void dvb_net_ule( struct net_device *dev, const u8 *buf, size_t buf_len )
>> 						       "field: %u.\n", priv->ts_count, *from_where);
>>
>> 						/* Drop partly decoded SNDU, reset state, resync on PUSI. */
>> -						if (priv->ule_skb) {
>> -							dev_kfree_skb( priv->ule_skb );
>> +						if (priv->ule_skb || priv->ule_sndu_remain) {
>> +							if (priv->ule_skb)
>> +								dev_kfree_skb( priv->ule_skb );
>> 							dev->stats.rx_errors++;
>> 							dev->stats.rx_frame_errors++;
>> 						}
> 
> That code block looks odd that way, but after staring at it for a minute,
> it makes sense. Another way to do it that might read cleaner (and reduce
> excessive tab indent levels) would be to add a 'bool errors', then:
> 
> 	bool errors = false;
> 	...
> 	if (priv->ule_skb) {
> 		errors = true;
> 		dev_kfree_skb(priv->ule_skb);
> 	}
> 
> 	if (errors || priv->ule_sndu_remain) {
> 		dev->stats.rx_errors++;
> 		dev->stats.rx_frame_errors++;
> 	}
> 
> 

Yeap, I'm a lazy person.

>> @@ -534,6 +535,7 @@ static void dvb_net_ule( struct net_device *dev, const u8 *buf, size_t buf_len )
>> 				from_where += 2;
>> 			}
>>
>> +			priv->ule_sndu_remain = priv->ule_sndu_len + 2;
>> 			/*
>> 			 * State of current TS:
>> 			 *   ts_remain (remaining bytes in the current TS cell)
> 
> Is this *always* true? Your description says "...the remaining 2 or 3
> bytes", indicating this could sometimes need to be +3. Is +0 also a
> possibility?
> 
> 

Not sure whether I understand your question correctly. Here is my attempt to answer your question. 
The encapsulation format always mandate that at least of 2 bytes of ULE SNDU (the LENGTH field) must 
be present within a MPEG2-TS frame. So, if only 1 byte of the ULE SNDU get packed into the 
remaining MPEG2-TS frame, then it is invalid. Of course, there is no issue regarding 0 byte as that 
would be the case of filling MPEG2-TS frame up to its boundary. New ULE SNDU will have to packed 
into the next MPEG2-TS frame in that case.

Now the problem with existing code is the interpretation of remainder length when 2 or 3 bytes of ULE 
SNDU are packed into the remainder of MPEG2-TS frame. In the 2 bytes case, only the LENGTH field is 
available while in the case 3 bytes, only the 1st octet of the 2-octets TYPE field and the LENGTH field 
are available. The ule_sndu_remain should carry the value of length of ULE SNDU following the the TYPE 
field. Technically, this would mean that remainder byte of ULE SNDU that need to be received is going 
to be:

Value(LENGTH) + 2 (We owe 2 bytes of TYPE field here) if only 2 bytes of ULE SNDU is packed (as in the 
case of case 0: at line 550). 
This is addressed by adding the priv->ule_sndu_remain = priv->ule_sndu_len + 2;

Value(LENGTH) + 1 (We owe 1 byte of TYPE field here) if 3 bytes of ULE SNDU is packed (as in the case of 
case 1: at 545). 
This is addressed by adding priv->ule_sndu_remain--;

If complete ULE header (>= 4 bytes) is available:
priv->ule_sndu_remain = priv->ule_sndu_len; at line 582 takes care of the rest and it works just fine in 
the existing code.

Due to the wrong interpretation of remaining length of ULE SNDU when 2 or 3 bytes of ULE SNDU are packed 
into a MPEG2-TS frame, the subsequent checking of payload pointer (line 455) always fails leading to 
unnecessary packet drops.

Looking back at the fix after a few months, I had trouble understanding how these few lines fixed the 
problem at first glance.


>> @@ -543,6 +545,7 @@ static void dvb_net_ule( struct net_device *dev, const u8 *buf, size_t buf_len )
>> 			 */
>> 			switch (ts_remain) {
>> 				case 1:
>> +					priv->ule_sndu_remain--;
>> 					priv->ule_sndu_type = from_where[0] << 8;
>> 					priv->ule_sndu_type_1 = 1; /* first byte of ule_type is set. */
>> 					ts_remain -= 1; from_where += 1;
>> @@ -556,6 +559,7 @@ static void dvb_net_ule( struct net_device *dev, const u8 *buf, size_t buf_len )
>> 				default: /* complete ULE header is present in current TS. */
>> 					/* Extract ULE type field. */
>> 					if (priv->ule_sndu_type_1) {
>> +						priv->ule_sndu_type_1 = 0;

this is an extra precaution. Not needed as it has been addressed elsewhere, but I am a bit concerned 
if there is any corner cases where this is not set in the current code.

>> 						priv->ule_sndu_type |= from_where[0];
>> 						from_where += 1; /* points to payload start. */
>> 						ts_remain -= 1;
> 


  reply	other threads:[~2010-05-21  3:47 UTC|newest]

Thread overview: 5+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2010-05-06 14:52 [PATCH] dvb-core: Fix ULE decapsulation bug when less than 4 bytes of ULE SNDU is packed into the remaining bytes of a MPEG2-TS frame Ang Way Chuang
2010-05-20 19:22 ` Jarod Wilson
2010-05-21  3:40   ` Ang Way Chuang [this message]
2010-05-21 17:54     ` Jarod Wilson
2010-05-22  3:37       ` Ang Way Chuang

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=4BF600B2.1080305@gmail.com \
    --to=wcang79@gmail.com \
    --cc=jarod@redhat.com \
    --cc=linux-media@vger.kernel.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).