MPL

From MultimediaWiki
Jump to navigation Jump to search
  • Extensions: mpl, mpls
  • This is a playlist file used in BluRays and AVCHD (Lite) video camcorders and foto cams.

Please correct or beautify the language used in this article!

Introduction

This site tries to collect information about this file. This information partly comes from reverse engineering. There is no free specification available on the net.

The intention was to extract the time stamps of the captured videos from the mpl files. The Panasonic GH1 doesn't write the time stamp into the SEI part of the h264 stream. Other cams are known to do this (e.g. Canon HG-20):

http://forum.videohelp.com/threads/296805-Reading-AVCHD-Playlist-files-BDMV-Playlist-*-mpl

This document might be helpful to understand the connection between a playlist file a cpi file and the actual stream file.

http://www.blu-raydisc.com/Assets/Downloadablefile/BD-RE_Part3_V2.1_WhitePaper_080406-15271.pdf

Technical Part

The file is located in a directory structure like this:

/BDMV/PLAYLIST/00000.MPL

it corresponds with the files in this directory:

/BDMV/STREAM/00000.MTS

In Bluray specification the files hav a differnt endig: *.mpls and *.m2ts.

The following information are reverse engineered from MPL files from a Panasonic GH1 foto cam.

This file sizes are known:

  • 638 bytes
  • 800 bytes
  • 962 bytes
  • 1124 bytes
  • 1286 bytes
  • 1448 bytes
  • 1772 bytes
  • 2096 bytes
  • 2420 bytes

The difference in size is always a multiple of 162 bytes.

File Structure

The values are written only when they are equal in every file.

Header
 M2TS File Meta Data [1]
 M2TS File Meta Data [2]
 ...
 Next Part 1
 Next Part 2 [2]                only there when more than one file is described in this document
 Next Part 2 [3]
 ...
 PLEX Part
 Next Part 3
 Video Part [1]
 Video Part [2]
 ...
 Trailer

Header

char[4] 'MPLS'            FOURCC?
char[4] '0100'            might be version number
char[4] {0, 0, 0, 0x3A}
char[2] {0, 0}
char[2]                   this corresponds with the file size
char[2] {0, 0}
char[2]                   this corresponds with the file size differs from the former one
char[12] {0, 0, ...}
char[8] {0, 0, ...}
char[4] {0, 0, 0, 0x0E}
char[4] {0, 1, 0, 0}
char[4] {0, 0, 1, 0xCF}
char[4] {0x40, 0, 0, 0}
char[4] {0, 0, 0, 0}
char[2]                    this corresponds with the file size
char[2] {0, 0}
char[2] {0, X}             file size = 476 + 162*X in bytes
char[2] {0, 0}

M2TS File Meta Data ?

This part ist contained X-times and is 82 bytes in size. The corresponding M2TS file name on the Panasonic is 00012.MTS.

char[2] {0, 'P'}
char[10] "00012M2TS"       this is a null terminated string of the corresponding MTS file name (without dot '.')
char[4] {1, 0, 0, 0}
char[2] {0x40, 0x74}
char[2]
char[2]
char[2] {0, 0}
char[4] {0, 0, 0, 0}
char[4] {0, 0, 0, 0}
char[4] {0, 0, 0, 0x2E}
char[4] {0, 0, 1, 1}
char[4] {0, 0, 0, 0}
char[4] {0, 0, 0, 0}
char[4] {0, 0, 9, 1}
char[4] {0x10, 0x11, 0, 0}
char[4] {0, 0, 0, 0}
char[4] {5, 0x1B, 0x43, 0}
char[4] {0, 0, 9, 1}
char[4] {0x11, 0, 0, 0}
char[4] {0, 0, 0, 0}
char[4] {5, 0x81, 0x31, 0x75}
char[2] {0x6E, 0x64}

Next Part 1

unknown contents

char[4] {0, 0, 0, A}                         A corresponds with file size
char[4] {0, B, 0, 1}                         B = number of described MTS files
char[4] {0, 0, 0, 0}
char[4] {0x40, 0x74, 0xFF, 0xFF}

Next Part 2

This part is 14 bytes in size. One of this parts is inlcuded for each file description starting with the second.

char[4] {0, 0, 0, 0}
char[4] {0, 1, 0, A}             A = number of description
char[4] {0, 0, _, _}
char[2] {0xFF, 0xFF}

PLEX Part

unknown contents

char[4] {0, 0, 0, 0}
char[4] {0, _, _, _}                        
char[4] {0, 0, 0, 0x18}
char[4] {0, 0, 0, 1}
char[4] {0x10, 0, 1, 0}
char[4] {0, 0, 0, 0x18}
char[4] {0, 0, _, _}
char[4] 'PLEX'
char[4] {0, 0, 0, 0}
char[4] {0, 0, 01, 0x44}
char[4] {0, 0, _, _}
char[4] {0, 0, 0, 0}
char[4] {0, 0, 0, 0}
char[4] {0, 0, 0, 0}
char[4] {0, 0, 0, 0}
char[4] {0, 0, 0, 0}
char[4] {0, 0, 0, 0}
char[4] {0, 0, 1, 0x18}
char[4] {1, 3, 5, 0x18}
char[4] {0, 0, 0, 0}
char[2] {0xFF, 0xFF}

Next Part 3

This part is 276 bytes in size.

It cntains a time stamp that might be the time when the file was closed by the cam.

char[1] {0x1E}
char[1] {C}                  C = BCD (Century)
char[1] {D}                  D = BCD (Decade)
char[1] {E}                  E = BCD (Month)
char[1] {F}                  F = BCD (Day)
char[1] {G}                  G = BCD (Hour)
char[1] {H}                  H = BCD (Minute)
char[1] {I}                  I = BCD (Second)
char[2] {0x90, 0x0A}  or  {0x90, 0x0C}
char[10]                     date as string (no time) e.g. '2010. 4.18'
char[4] {0, 0, 0, 0}
...
char[4] {0, 0, 0, 0}         
...
char[4] {0, 0, 0, 0}         
char[1] {0, _, 0, A}         A = number of Video Parts following

Video Part

This part contains time stamps of the corresponding MTS file. Right before the first occurance of one of such part is a char that tells how many such parts are following.

This part is 66 bytes in size.

char[4] {1, 3, 5, 0x81}
char[4] {0, 0, 0, 0}
char[2] {A, B}               (A<<8 + B).MTS = 00012.MTS, A=0, B=0x0C; 
char[1] {0x1E}
char[1] {C}                  C = BCD (Century)
char[1] {D}                  D = BCD (Decade)
char[1] {E}                  E = BCD (Month)
char[1] {F}                  F = BCD (Day)
char[1] {G}                  G = BCD (Hour)
char[1] {H}                  H = BCD (Minute)
char[1] {I}                  I = BCD (Second)
char[2] {0x90, 0x0A} or {0x90, 0x0C}
char[10]                     date as string (no time) e.g. '2010. 4.18'
char[2] {0, 0}
char[4] {0, 0, 0, 0}
char[4] {0, 0, 0, 0}
char[4] {0, 0, 0, 0}
char[4] {0x11, 0x10, 0, 0}
char[4] {0, I, J, _}         I = 0x65 or 0x66; J=2 for SH or 0 for FHD
char[1] {0x03}
char[3] 'FHD'; 'SH\0'        video format
char[4] {0, 0, 0, 0}
char[4] {0xFF, 0xFF, 0xFF, 0xFF}
char[2] {0, 0}

A and B tell the name of the corresponding MTS file e.g. (A<<8 + B).MTS = 00012.MTS, A=0, B=0x0C

Trailer

This part is 50 bytes in size.

char[4] {0, 0, 0, 0}
char[4] {_, _, _, _}
...
char[4] {_, _, _, _}
char[4] {0, 0, 0, 0}
char[2] {0, 0}

Definitions

BCD
Binary Coded Decimal
_
unknow but different in different files
FHD
Full HD Quality (1080p25)
SH
Super High Quality (720p50)

Time Stamp Parser

This is a small C programm to parse all time stamps of MTS files that are written in playlist files in a directory. This programm is developed based on this file:

http://www.avsforum.com/avs-vb/showthread.php?p=14578210#post14578210

/* GPL 2.1 © Elte 2011 */

#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <dirent.h>

#define VERSION         "0.1"
#define SIGNATUR        "MPLS0100"

typedef struct {
    unsigned char decade, month, day, hour, minute, second;
} MTS_date;

int main (int argc, char *argv[])
{
    struct dirent  *dp = NULL;
    DIR       *dir_ptr = NULL;
    FILE *mpl_file_ptr = NULL;

    int ret = 0;

    unsigned char signatur[] = SIGNATUR, buffer[sizeof(SIGNATUR)];

    if (argc != 2) {
        fprintf(stderr, "Version: %s\n", VERSION);
        fprintf(stderr, "Usage: %s <path to folder containing *.MPL files\n", argv[0]);
        goto ERROR;
    }

    if (chdir(argv[1]) == -1) {
	fprintf(stderr, "Cannot change directory: %s %s\n", argv[1], strerror(errno));
	goto ERROR;
    }

    if ((dir_ptr = opendir (".")) == NULL) {
        fprintf (stderr, "Error reading directory: %s %s\n", argv[1], strerror(errno));
        goto ERROR;
    }

//    printf (" STREAM       DATE       TIME\n");
//    printf ("---------  ----------  --------\n");

    while ((dp = readdir(dir_ptr)) != NULL) {

        if (!strncmp ((dp->d_name) + (strlen (dp->d_name) - 4), ".MPL", 4)) {

            unsigned char file_number[2] = {0, 0},
                                num_desc = 0;
            MTS_date                this = {0, 0, 0, 0, 0, 0};

            if ((mpl_file_ptr = fopen (dp->d_name, "r")) == NULL) {
                fprintf (stderr, "Error opening file: %s %s\n", dp->d_name, strerror(errno));
                goto NEXT_FILE;
            }

            if ((fread (buffer, sizeof(SIGNATUR), 1, mpl_file_ptr) != 1) ||
                (strncmp (buffer, SIGNATUR, sizeof(SIGNATUR)))) {
                fprintf (stderr, "Could not read file signatur. Wrong filetype?\n");
                goto NEXT_FILE;
            }

            // find out how many mts files are described
            // the 66th byte contains this number
            fseek (mpl_file_ptr, 66 - sizeof(SIGNATUR) - 1, SEEK_CUR);
            ret = fgetc(mpl_file_ptr);
            if (ret != EOF) {
                num_desc = (unsigned char) ret;
            } else {
                fprintf (stderr, "Could not read contents\n");
                goto NEXT_FILE;
            }

            // jump to the first occurance of a time stamp and print it
            // iterate till all num_desc time stamps are shown
            // trailer = 50 bytes
            // mts description = 66 bytes
            // actual info starts at 9th byte of mts description
            fseek (mpl_file_ptr, -50 - 66*num_desc - 48 +2, SEEK_END);

            while (num_desc > 0) {

                int i;
                const unsigned char time_stamp_sig[] = {1, 3, 5, 0x81, 0, 0, 0, 0};

                num_desc--;
                fseek (mpl_file_ptr, 48, SEEK_CUR);

                // scan for time stamp signatur
                for (i=0; i<8; i++) {
                    if ((ret = fgetc(mpl_file_ptr)) != time_stamp_sig[i]) {
                        fprintf (stderr, "Could not parse contents\n");
                        goto NEXT_FILE;
                    }
                }

                // scan time stamp
                if ((ret = fscanf (mpl_file_ptr, "%c%c\x1E%*1[ ]%c%c%c%c%c%c",
                              &file_number[0], &file_number[1],
                              &this.decade, &this.month, &this.day,
                              &this.hour, &this.minute, &this.second)) != 8) {
                    fprintf (stderr, "Could not parse time stamp\n");
                    goto NEXT_FILE;
                }
                printf ("%.5d.MTS  20%.2X/%.2X/%.2X  %.2X:%.2X:%.2X\n",
                        (file_number[0]<<8) + file_number[1],
                        this.decade, this.month, this.day,
                        this.hour, this.minute, this.second);
            }
NEXT_FILE:
            fclose (mpl_file_ptr);
        }
    }
ERROR:
    closedir (dir_ptr);
}