DIVX 6 with xsub - subtitles don't show

macOS specific usage questions
Guest

DIVX 6 with xsub - subtitles don't show

Postby Guest » 26 Aug 2005 00:23

Hi everyone,

I tried to look for a similar post but didnt find, so here goes. :arrow:

Is the VLC player going to support XSUB subtitles that are generated by divx 6 converter? Everything plays fine, such as multiple audio tracks, but the subtitles dont. I get a message when I open the movie that there is an unsupported stream, it calls it DXSUB but I prosume it means xsub.

Perhaps there is a plugin somehere? or is it just too new a feature in divx for this OSX version of VLC?

thank you,

nick

nickdrj

Postby nickdrj » 26 Aug 2005 01:02

Its me again, I didnt log in correctly before

anyway, VLC gives me the following error message when i open a .divx file with xsub

main: no suitable decoder module for fourcc `DXSB'.
VLC probably does not support this sound or video format.


So does anyone have a clue about this?

nick

The DJ
Cone Master
Cone Master
Posts: 5987
Joined: 22 Nov 2003 21:52
VLC version: git
Operating System: Mac OS X
Location: Enschede, Holland
Contact:

Postby The DJ » 26 Aug 2005 22:28

VLC does not support XSUB, because the format is unknown. DivX Networks hasn't made any details available about it.
Don't use PMs for support questions.

Guest

xsub encoder

Postby Guest » 26 Jun 2006 18:24

Hi all,

As the source of the XSUB encoder are public and in DrFFMPEG here is the
format of the XSUB subtitle in .divx/DMF files :

typedef struct Color
{
uint8_t red;
uint8_t green;
uint8_t blue;
} Color;

// DivX Subpicture Packet header
typedef struct DivxSubPictPackHdr
{
// Duration in the following format
// [HH:MM:SS.XXX-hh:mm:ss.xxx]
// Note: There is no NUL at the end
char duration[27];

// Subpicture dimensions & coordinates
uint16_t width;
uint16_t height;
uint16_t left;
uint16_t top;
uint16_t right;
uint16_t bottom;
uint16_t fieldOffset;

// Background, pattern, emphasis1, emphasis2 colors
Color background;
Color pattern;
Color emphasis1;
Color emphasis2;
} DivxSubPictPackHdr;

#define SIZEOF_XSUB_PACKET 53

This is stored in little-endian with the exact size for each field. I
think Gabest shouldn't have a hard time supporting the format too.

--
robUx4 on blog <http://robux4.blogspot.com/>
-------------- next part --------------
/*
* DivX subtitle encoding for ffmpeg
* Copyright (c) 2005 DivX, Inc.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include "avcodec.h"

#ifdef CONFIG_XSUB_ENCODER

// Safe area
const uint32_t SUBTITLE_BOX_TOP_BORDER = 24;
const uint32_t SUBTITLE_BOX_LEFT_BORDER = 44;
const uint32_t SUBTITLE_BOX_BOTTOM_BORDER = 455;
const uint32_t SUBTITLE_BOX_RIGHT_BORDER = 595;

// Default screen
const uint32_t SCREEN_WIDTH = 640;
const uint32_t SCREEN_HEIGHT = 480;

typedef struct Color
{
uint8_t red;
uint8_t green;
uint8_t blue;
} Color;

// DivX Subpicture Packet header
typedef struct DivxSubPictPackHdr
{
// Duration in the following format
// [HH:MM:SS.XXX-hh:mm:ss.xxx]
// Note: There is no NUL at the end
char duration[28];

// Subpicture dimensions & coordinates
uint16_t width;
uint16_t height;
uint16_t left;
uint16_t top;
uint16_t right;
uint16_t bottom;
uint16_t fieldOffset;

// Background, pattern, emphasis1, emphasis2 colors
Color background;
Color pattern;
Color emphasis1;
Color emphasis2;
} DivxSubPictPackHdr;

#define SIZEOF_XSUB_PACKET 53

extern int get_nibble(const uint8_t *buf, int nibble_offset);

static void set_nibble(uint8_t *buffer, int32_t offset, uint8_t value)
{
int32_t b = offset / 2;
int32_t h = (offset % 2 == 1);

if (h == 0)
{
buffer = (buffer & 0x0F) | (value << 4);
}
else
{
buffer = (buffer & 0xF0) | (value << 0);
}
}

static void xsub_encode_rle(uint8_t **pq,
const uint8_t *bitmap, int linesize,
int w, int h)
{
uint8_t *q;
int x, y, len, x1, color;
int offset = 0;

q = *pq;

for(y = 0; y < h; y++) {
x = 0;

while (x < w) {
x1 = x;
color = bitmap[x1++];

while (x1 < w && bitmap[x1] == color)
x1++;
len = x1 - x;

// Run can't be longer than 255, unless it is the rest of a row
if(x1 < w && len > 255)
{
int32_t diff = len - 255;

len -= diff;
x1 -= diff;
}

if (len >= 1 && len <=3) {
// Set 2-bit len, and 2-bit color
set_nibble(q, offset++, (uint8_t)((len << 2) | color));
} else if (len >= 4 && len <= 15) {
int32_t l = (len & 0x0C) >> 2; // 2 bit left
int32_t right = (len & 0x03) >> 0; // 2 bit right

// Set 2-bit zero, and left 2-bit len
set_nibble(q, offset++, (uint8_t)l);

// Set right 2-bit len and 2-bit color
set_nibble(q, offset++, (uint8_t)((right << 2) | color));
} else if (len >= 16 && len <= 63) {
int32_t l = (len & 0x3C) >> 2; // 4 bit left
int32_t right = (len & 0x03) >> 0; // 2 bit right

// Set 4-bit zero
set_nibble(q, offset++, 0);

// Set left 4-bit len
set_nibble(q, offset++, (uint8_t)l);

// Set right 2-bit len and 2-bit color
set_nibble(q, offset++, (uint8_t)((right << 2) | color));
} else if (len >= 64 && len <= 255) {
int32_t l = (len & 0xC0) >> 6; // 2 bit left
int32_t m = (len & 0x3C) >> 2; // 4 bit middle
int32_t right = (len & 0x03) >> 0; // 2 bit right

// Set 6-bit zero and 2-bit len
set_nibble(q, offset++, 0);
set_nibble(q, offset++, (uint8_t)(0 | l));

// Set middle 4-bit len
set_nibble(q, offset++, (uint8_t)m);

// Set right 2-bit len and 2-bit color
set_nibble(q, offset++, (uint8_t)((right << 2) | color));
} else {
// Set 14-bit zero and 2-bit color
set_nibble(q, offset++, 0);
set_nibble(q, offset++, 0);
set_nibble(q, offset++, 0);
set_nibble(q, offset++, (uint8_t)(0 | color));
}

x += len;
}

// byte align
if (offset % 2) {
set_nibble(q, offset++, 0);
}

bitmap += linesize;
}

*pq = q + offset;
}

#ifdef DEBUG
static sub_pict_count = 0;
#undef fprintf
static void ppm_save(const char *filename, uint8_t *bitmap, int w, int h,
uint32_t *rgba_palette)
{
int x, y, v;
FILE *f;

f = fopen(filename, "w");
if (!f) {
perror(filename);
exit(1);
}
fprintf(f, "P6\n"
"%d %d\n"
"%d\n",
w, h, 255);
for(y = 0; y < h; y++) {
for(x = 0; x < w; x++) {
int uu = bitmap[y * w + x];
v = rgba_palette[uu];
putc((v >> 16) & 0xff, f);
putc((v >> 8) & 0xff, f);
putc((v >> 0) & 0xff, f);
}
}
fclose(f);
}

static int decode_rle(uint8_t *bitmap, int linesize, int w, int h,
const uint8_t *buf, int nibble_offset, int buf_size)
{
unsigned int v;
int x, y, len, color, nibble_end;
uint8_t *d;

nibble_end = buf_size * 2;
x = 0;
y = 0;
d = bitmap;
for(;;) {
if (nibble_offset >= nibble_end)
return -1;
v = get_nibble(buf, nibble_offset++);
if (v < 0x4) {
v = (v << 4) | get_nibble(buf, nibble_offset++);
if (v < 0x10) {
v = (v << 4) | get_nibble(buf, nibble_offset++);
if (v < 0x040) {
v = (v << 4) | get_nibble(buf, nibble_offset++);
if (v < 4) {
v |= (w - x) << 2;
}
}
}
}
len = v >> 2;
if (len > (w - x))
len = (w - x);
color = v & 0x03;
memset(d + x, color, len);
x += len;
if (x >= w) {
y++;
if (y >= h)
break;
d += linesize;
x = 0;
/* byte align */
nibble_offset += (nibble_offset & 1);
}
}
return 0;
}
#endif

#define COLOR_TO_BUFFER(col, buffer) \
*buffer++ = col.red; \
*buffer++ = col.green; \
*buffer++ = col.blue;

static void encode_xsub_header_to_buf(DivxSubPictPackHdr *hdr, uint8_t *buf)
{
memcpy( buf, hdr->duration, 27 );
buf += 27;

// Little endian fields
*buf++ = hdr->width & 0xFF;
*buf++ = hdr->width >> 8;

*buf++ = hdr->height & 0xFF;
*buf++ = hdr->height >> 8;

*buf++ = hdr->left & 0xFF;
*buf++ = hdr->left >> 8;

*buf++ = hdr->top & 0xFF;
*buf++ = hdr->top >> 8;

*buf++ = hdr->right & 0xFF;
*buf++ = hdr->right >> 8;

*buf++ = hdr->bottom & 0xFF;
*buf++ = hdr->bottom >> 8;

*buf++ = hdr->fieldOffset & 0xFF;
*buf++ = hdr->fieldOffset >> 8;

COLOR_TO_BUFFER(hdr->background, buf);
COLOR_TO_BUFFER(hdr->pattern, buf);
COLOR_TO_BUFFER(hdr->emphasis1, buf);
COLOR_TO_BUFFER(hdr->emphasis2, buf);
}

static int encode_xsub_subtitles(uint8_t *outbuf, AVSubtitle *h)
{
#ifdef DEBUG
char filename[256];
uint8_t *tmp_bitmap;
#endif

uint8_t *resized_bitmap;
uint32_t executeTime = h->end_display_time - h->start_display_time;
uint32_t startTime = h->pts / 90;

uint32_t endTime = startTime + executeTime;

uint32_t startHrs;
uint32_t startMins;
uint32_t startSecs;
uint32_t startMilliSecs;

uint32_t endHrs;
uint32_t endMins;
uint32_t endSecs;
uint32_t endMilliSecs;

DivxSubPictPackHdr hdr;
uint8_t *q;
uint8_t *top_field_offset;
uint8_t *bottom_field_offset;
uint32_t top_field_len;
uint32_t bottom_field_len;
int r;
int i = 0;

if (h->num_rects == 0 || h->rects == NULL)
return -1;

startMilliSecs = (uint32_t) startTime % 1000;

startTime /= 1000;

startSecs = (uint32_t) startTime % 60;

startTime /= 60;

startMins = (uint32_t) startTime % 60;

startTime /= 60;

startHrs = (uint32_t) startTime;

endMilliSecs = (uint32_t) endTime % 1000;

endTime /= 1000;

endSecs = (uint32_t) endTime % 60;

endTime /= 60;

endMins = (uint32_t) endTime % 60;

endTime /= 60;

endHrs = (uint32_t) endTime;

snprintf(hdr.duration, 28,
"[%02d:%02d:%02d.%03d-%02d:%02d:%02d.%03d]",
startHrs, startMins, startSecs, startMilliSecs,
endHrs, endMins, endSecs, endMilliSecs);

hdr.right = 0;
hdr.bottom = 0;

hdr.width = h->rects[0].w;
hdr.height = h->rects[0].h;

if (hdr.width % 2)
{
hdr.width++;
}

// 2 pixels required on either side of subtitle
hdr.width += 4;

if (hdr.height % 2)
{
hdr.height++;
}

hdr.left = (SCREEN_WIDTH - hdr.width) / 2;

if (hdr.left % 2)
{
hdr.left--;
}

if (hdr.left < SUBTITLE_BOX_LEFT_BORDER)
{
if (hdr.left + hdr.width < SUBTITLE_BOX_RIGHT_BORDER)
{
hdr.left = SUBTITLE_BOX_LEFT_BORDER;
}
}

if (hdr.left + hdr.width > SUBTITLE_BOX_RIGHT_BORDER)
{
if (SUBTITLE_BOX_RIGHT_BORDER - hdr.width > SUBTITLE_BOX_LEFT_BORDER)
{
hdr.left = SUBTITLE_BOX_RIGHT_BORDER - hdr.width;

if (hdr.left % 2)
{
hdr.left--;
}
}
}

//if (SUBTITLE_BOX_BOTTOM_BORDER >= hdr.height)
{
hdr.top = SUBTITLE_BOX_BOTTOM_BORDER - hdr.height;
}
//else
{
//hdr.top = 0;
}

if (hdr.top %2)
{
hdr.top--;
}

hdr.background.red = r = (h->rects[0].rgba_palette[0] >> 16) & 0xff;
hdr.background.green = (h->rects[0].rgba_palette[0] >> 8) & 0xff;
hdr.background.blue = (h->rects[0].rgba_palette[0] >> 0) & 0xff;

hdr.pattern.red = r = (h->rects[0].rgba_palette[1] >> 16) & 0xff;
hdr.pattern.green = (h->rects[0].rgba_palette[1] >> 8) & 0xff;
hdr.pattern.blue = (h->rects[0].rgba_palette[1] >> 0) & 0xff;

hdr.emphasis1.red = r = (h->rects[0].rgba_palette[2] >> 16) & 0xff;
hdr.emphasis1.green = (h->rects[0].rgba_palette[2] >> 8) & 0xff;
hdr.emphasis1.blue = (h->rects[0].rgba_palette[2] >> 0) & 0xff;

hdr.emphasis2.red = r = (h->rects[0].rgba_palette[3] >> 16) & 0xff;
hdr.emphasis2.green = (h->rects[0].rgba_palette[3] >> 8) & 0xff;
hdr.emphasis2.blue = (h->rects[0].rgba_palette[3] >> 0) & 0xff;

q = outbuf + SIZEOF_XSUB_PACKET;

resized_bitmap = av_malloc(hdr.width * hdr.height);
memset(resized_bitmap, 0, hdr.width * hdr.height);

for (i = 0; i < h->rects[0].h; i++)
{
memcpy(resized_bitmap + (hdr.width * i + 2), h->rects[0].bitmap + (h->rects[0].w * i), h->rects[0].w);
}

top_field_offset = q;
xsub_encode_rle(&q, resized_bitmap, hdr.width * 2, hdr.width, hdr.height / 2);
top_field_len = q - top_field_offset;

#ifdef DEBUG
tmp_bitmap = av_malloc(hdr.width * hdr.height / 2);
decode_rle(tmp_bitmap, hdr.width, hdr.width, hdr.height / 2, top_field_offset, 0, top_field_len);

sub_pict_count++;
snprintf(filename, 256, "bt-%d.ppm", sub_pict_count);
ppm_save(filename, tmp_bitmap,
hdr.width, hdr.height / 2, h->rects[0].rgba_palette);
av_free(tmp_bitmap);
#endif

bottom_field_offset = q;
xsub_encode_rle(&q, resized_bitmap + hdr.width, hdr.width * 2, hdr.width, hdr.height / 2);
bottom_field_len = q - bottom_field_offset;

#ifdef DEBUG
tmp_bitmap = av_malloc(hdr.width * hdr.height / 2);
decode_rle(tmp_bitmap, hdr.width, hdr.width, hdr.height / 2, bottom_field_offset, 0, bottom_field_len);

snprintf(filename, 256, "bb-%d.ppm", sub_pict_count);
ppm_save(filename, tmp_bitmap,
hdr.width, hdr.height / 2, h->rects[0].rgba_palette);
av_free(tmp_bitmap);
#endif

av_free(resized_bitmap);

hdr.fieldOffset = top_field_len;

encode_xsub_header_to_buf( &hdr, outbuf);

return (SIZEOF_XSUB_PACKET + bottom_field_len + top_field_len);
}

static int xsub_init_decoder(AVCodecContext *avctx)
{
return 0;
}

static int xsub_close_decoder(AVCodecContext *avctx)
{
return 0;
}

static int xsub_encode(AVCodecContext *avctx,
unsigned char *buf, int buf_size, void *data)
{
AVSubtitle *sub = data;
return encode_xsub_subtitles(buf, sub);
}

AVCodec xsub_encoder = {
"xsub",
CODEC_TYPE_SUBTITLE,
CODEC_ID_XSUB_SUBTITLE,
0,
xsub_init_decoder,
xsub_encode,
xsub_close_decoder,
};
#endif


Return to “VLC media player for macOS Troubleshooting”

Who is online

Users browsing this forum: No registered users and 20 guests