diff --git a/NEWS b/NEWS
index 94de7aacb138d0992c62bbb1b25253b6b8dae03b..05abf36dc608e58c5c8d42149f31dd12bfb3ce7f 100644
--- a/NEWS
+++ b/NEWS
@@ -49,6 +49,7 @@ Demuxers:
  * Fixed TS playback with PAT/PMT less recordings
  * Support for lame's replaygain extension in mpeg files
  * Fixes for DTS detection in WAV and MKV files
+ * Basic support for MPEG4-SL in TS and T-DMB
 
 Stream filter:
  * Added ARIB STD-B25 TS streams decoder
diff --git a/modules/demux/ts.c b/modules/demux/ts.c
index b4a500dd4c23caa719a05844535378712bfb1954..4fa8579b3f44f4aa5b2806e0e57dea9c0ca8f62d 100644
--- a/modules/demux/ts.c
+++ b/modules/demux/ts.c
@@ -490,6 +490,9 @@ static int64_t TimeStampWrapAround( ts_pmt_t *, int64_t );
 
 /* MPEG4 related */
 static const es_mpeg4_descriptor_t * GetMPEG4DescByEsId( const ts_pmt_t *, uint16_t );
+static ts_pes_es_t * GetPMTESBySLEsId( ts_pmt_t *, uint16_t );
+static bool SetupISO14496LogicalStream( demux_t *, const decoder_config_descriptor_t *,
+                                        es_format_t * );
 
 #define TS_USER_PMT_NUMBER (0)
 static int UserPmt( demux_t *p_demux, const char * );
@@ -1394,8 +1397,13 @@ static void UpdatePESFilters( demux_t *p_demux, bool b_all )
                 es_out_Control( p_demux->out, ES_OUT_GET_ES_STATE,
                                 espid->u.p_pes->es.id, &b_stream_selected );
 
-            if( !p_sys->b_es_all && espid->u.p_pes->es.fmt.i_cat == UNKNOWN_ES )
-                b_stream_selected = false;
+            if( espid->u.p_pes->es.fmt.i_cat == UNKNOWN_ES )
+            {
+                if( espid->u.p_pes->i_stream_type == 0x13 ) /* Object channel */
+                    b_stream_selected = true;
+                else if( !p_sys->b_es_all )
+                    b_stream_selected = false;
+            }
 
             if( b_stream_selected )
                 msg_Dbg( p_demux, "enabling pid %d from program %d", espid->i_pid, p_pmt->i_number );
@@ -2371,13 +2379,54 @@ static void ParsePES( demux_t *p_demux, ts_pid_t *pid, block_t *p_pes )
                         p_block->i_pts += (p_pmt->pcr.i_pcroffset * 100 / 9);
                 }
 
-                for( int i = 0; i < pid->u.p_pes->extra_es.i_size; i++ )
+                /* SL in PES */
+                if( pid->u.p_pes->i_stream_type == 0x12 &&
+                    ((i_stream_id & 0xFA) == 0xFA) /* 0xFA || 0xFB */ )
                 {
-                    es_out_Send( p_demux->out, pid->u.p_pes->extra_es.p_elems[i]->id,
-                            block_Duplicate( p_block ) );
+                    const es_mpeg4_descriptor_t *p_desc =
+                            GetMPEG4DescByEsId( p_pmt, pid->u.p_pes->es.i_sl_es_id );
+                    if(!p_desc)
+                    {
+                        block_Release( p_block );
+                        p_block = NULL;
+                    }
+                    else
+                    {
+                        sl_header_data header = DecodeSLHeader( p_block->i_buffer, p_block->p_buffer,
+                                                                &p_mpeg4desc->sl_descr );
+                        p_block->i_buffer -= header.i_size;
+                        p_block->p_buffer += header.i_size;
+                        p_block->i_dts = header.i_dts ? header.i_dts : p_block->i_dts;
+                        p_block->i_pts = header.i_pts ? header.i_pts : p_block->i_pts;
+
+                        /* Assemble access units */
+                        if( header.b_au_start && pid->u.p_pes->sl.p_data )
+                        {
+                            block_ChainRelease( pid->u.p_pes->sl.p_data );
+                            pid->u.p_pes->sl.p_data = NULL;
+                            pid->u.p_pes->sl.pp_last = &pid->u.p_pes->sl.p_data;
+                        }
+                        block_ChainLastAppend( &pid->u.p_pes->sl.pp_last, p_block );
+                        p_block = NULL;
+                        if( header.b_au_end )
+                        {
+                            p_block = block_ChainGather( pid->u.p_pes->sl.p_data );
+                            pid->u.p_pes->sl.p_data = NULL;
+                            pid->u.p_pes->sl.pp_last = &pid->u.p_pes->sl.p_data;
+                        }
+                    }
                 }
 
-                es_out_Send( p_demux->out, pid->u.p_pes->es.id, p_block );
+                if ( p_block )
+                {
+                    for( int i = 0; i < pid->u.p_pes->extra_es.i_size; i++ )
+                    {
+                        es_out_Send( p_demux->out, pid->u.p_pes->extra_es.p_elems[i]->id,
+                                block_Duplicate( p_block ) );
+                    }
+
+                    es_out_Send( p_demux->out, pid->u.p_pes->es.id, p_block );
+                }
             }
             else
             {
@@ -2400,42 +2449,106 @@ static void ParseTableSection( demux_t *p_demux, ts_pid_t *pid, block_t *p_data
 {
     block_t *p_content = block_ChainGather( p_data );
 
-    if ( pid->p_parent && pid->p_parent->type == TYPE_PMT )
+    if( p_content->i_buffer <= 9 || pid->type != TYPE_PES )
     {
-        ts_pmt_t *p_pmt = pid->p_parent->u.p_pmt;
+        block_Release( p_content );
+        return;
+    }
+
+    const uint8_t i_table_id = p_content->p_buffer[0];
+    const uint8_t i_version = ( p_content->p_buffer[5] & 0x3F ) >> 1;
+    ts_pmt_t *p_pmt = pid->p_parent->u.p_pmt;
+
+    if ( pid->u.p_pes->i_stream_type == 0x82 && i_table_id == 0xC6 ) /* SCTE_27 */
+    {
+        assert( pid->u.p_pes->es.fmt.i_codec == VLC_CODEC_SCTE_27 );
         mtime_t i_date = p_pmt->pcr.i_current;
 
-        if( pid->u.p_pes->es.fmt.i_codec == VLC_CODEC_SCTE_27 )
+        /* We need to extract the truncated pts stored inside the payload */
+        int i_index = 0;
+        size_t i_offset = 4;
+        if( p_content->p_buffer[3] & 0x40 )
+        {
+            i_index = ((p_content->p_buffer[7] & 0x0f) << 8) |
+                    p_content->p_buffer[8];
+            i_offset = 9;
+        }
+        if( i_index == 0 && p_content->i_buffer > i_offset + 8 )
         {
-            /* We need to extract the truncated pts stored inside the payload */
-            if( p_content->i_buffer > 9 && p_content->p_buffer[0] == 0xc6 )
+            bool is_immediate = p_content->p_buffer[i_offset + 3] & 0x40;
+            if( !is_immediate )
             {
-                int i_index = 0;
-                size_t i_offset = 4;
-                if( p_content->p_buffer[3] & 0x40 )
-                {
-                    i_index = ((p_content->p_buffer[7] & 0x0f) << 8) |
-                             p_content->p_buffer[8];
-                    i_offset = 9;
-                }
-                if( i_index == 0 && p_content->i_buffer > i_offset + 8 )
+                mtime_t i_display_in = GetDWBE( &p_content->p_buffer[i_offset + 4] );
+                if( i_display_in < i_date )
+                    i_date = i_display_in + (1ll << 32);
+                else
+                    i_date = i_display_in;
+            }
+
+        }
+
+        p_content->i_dts = p_content->i_pts = VLC_TS_0 + i_date * 100 / 9;
+        PCRFixHandle( p_demux, p_pmt, p_content );
+    }
+    /* Object stream SL in table sections */
+    else if( pid->u.p_pes->i_stream_type == 0x13 && i_table_id == 0x05 &&
+             pid->u.p_pes->es.i_sl_es_id && p_content->i_buffer > 12 )
+    {
+        const es_mpeg4_descriptor_t *p_mpeg4desc = GetMPEG4DescByEsId( p_pmt, pid->u.p_pes->es.i_sl_es_id );
+        if( p_mpeg4desc && p_mpeg4desc->dec_descr.i_objectTypeIndication == 0x01 &&
+            p_mpeg4desc->dec_descr.i_streamType == 0x01 /* Object */ &&
+            p_pmt->od.i_version != i_version )
+        {
+            const uint8_t *p_data = p_content->p_buffer;
+            int i_data = p_content->i_buffer;
+
+            /* Forward into section */
+            uint16_t len = ((p_content->p_buffer[1] & 0x0f) << 8) | p_content->p_buffer[2];
+            p_data += 8; i_data -= 8; // SL in table
+            i_data = __MIN(i_data, len - 5);
+            i_data -= 4; // CRC
+
+            od_descriptors_t *p_ods = &p_pmt->od;
+            sl_header_data header = DecodeSLHeader( i_data, p_data, &p_mpeg4desc->sl_descr );
+
+            DecodeODCommand( VLC_OBJECT(p_demux), p_ods, i_data - header.i_size, &p_data[header.i_size] );
+            bool b_changed = false;
+
+            for( int i=0; i<p_ods->objects.i_size; i++ )
+            {
+                od_descriptor_t *p_od = p_ods->objects.p_elems[i];
+                for( int j = 0; j < ES_DESCRIPTOR_COUNT && p_od->es_descr[j].b_ok; j++ )
                 {
-                    bool is_immediate = p_content->p_buffer[i_offset + 3] & 0x40;
-                    if( !is_immediate )
+                    p_mpeg4desc = &p_od->es_descr[j];
+                    ts_pes_es_t *p_es = GetPMTESBySLEsId( p_pmt, p_mpeg4desc->i_es_id );
+                    es_format_t fmt;
+                    es_format_Init( &fmt, UNKNOWN_ES, 0 );
+                    fmt.i_id = p_es->fmt.i_id;
+                    fmt.i_group = p_es->fmt.i_group;
+
+                    if ( p_mpeg4desc && p_mpeg4desc->b_ok && p_es &&
+                         SetupISO14496LogicalStream( p_demux, &p_mpeg4desc->dec_descr, &fmt ) &&
+                         !es_format_IsSimilar( &fmt, &p_es->fmt ) )
                     {
-                        mtime_t i_display_in = GetDWBE( &p_content->p_buffer[i_offset + 4] );
-                        if( i_display_in < i_date )
-                            i_date = i_display_in + (1ll << 32);
-                        else
-                            i_date = i_display_in;
+                        es_format_Clean( &p_es->fmt );
+                        p_es->fmt = fmt;
+
+                        es_out_Del( p_demux->out, p_es->id );
+                        p_es->fmt.b_packetized = true; /* Split by access unit, no sync code */
+                        FREENULL( p_es->fmt.psz_description );
+                        p_es->id = es_out_Add( p_demux->out, &p_es->fmt );
+                        b_changed = true;
                     }
-
                 }
             }
-        }
 
-        p_content->i_dts = p_content->i_pts = VLC_TS_0 + i_date * 100 / 9;
-        PCRFixHandle( p_demux, p_pmt, p_content );
+            if( b_changed )
+                UpdatePESFilters( p_demux, p_demux->p_sys->b_es_all );
+
+            p_ods->i_version = i_version;
+        }
+        block_Release( p_content );
+        return;
     }
 
     if( pid->u.p_pes->es.id )
@@ -4006,6 +4119,17 @@ static const es_mpeg4_descriptor_t * GetMPEG4DescByEsId( const ts_pmt_t *pmt, ui
     return NULL;
 }
 
+static ts_pes_es_t * GetPMTESBySLEsId( ts_pmt_t *pmt, uint16_t i_sl_es_id )
+{
+    for( int i=0; i< pmt->e_streams.i_size; i++ )
+    {
+        ts_pes_es_t *p_es = &pmt->e_streams.p_elems[i]->u.p_pes->es;
+        if( p_es->i_sl_es_id == i_sl_es_id )
+            return p_es;
+    }
+    return NULL;
+}
+
 static bool SetupISO14496LogicalStream( demux_t *p_demux, const decoder_config_descriptor_t *dcd,
                                         es_format_t *p_fmt )
 {