You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
934 lines
28 KiB
934 lines
28 KiB
/***************************************************************************
|
|
* Copyright (C) 2005 by Jean-Michel Petit *
|
|
* jm_petit@laposte.net *
|
|
* *
|
|
* This program is free software; you can redistribute it and/or modify *
|
|
* it under the terms of the GNU General Public License as published by *
|
|
* the Free Software Foundation; either version 2 of the License, or *
|
|
* (at your option) any later version. *
|
|
* *
|
|
* This program 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 General Public License for more details. *
|
|
* *
|
|
* You should have received a copy of the GNU General Public License *
|
|
* along with this program; if not, write to the *
|
|
* Free Software Foundation, Inc., *
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
|
|
***************************************************************************/
|
|
#include "k9ifo2.h"
|
|
#include "bswap.h"
|
|
#include "dvdread.h"
|
|
|
|
|
|
k9Ifo2::k9Ifo2(k9DVDRead *_dvdread)
|
|
: QObject(NULL,"") {
|
|
_ifo=NULL;
|
|
m_dvd=_dvdread;
|
|
}
|
|
|
|
|
|
k9Ifo2::~k9Ifo2() {
|
|
closeIFO();
|
|
|
|
|
|
}
|
|
|
|
/*!
|
|
\fn k9Ifo2::setDevice(QString &_device)
|
|
*/
|
|
void k9Ifo2::setDevice(QString &_device) {
|
|
device=_device;
|
|
}
|
|
|
|
|
|
/*!
|
|
\fn k9Ifo2::setOutput(QString &_output)
|
|
*/
|
|
void k9Ifo2::setOutput(QString &_output) {
|
|
output=_output;
|
|
}
|
|
|
|
|
|
/*!
|
|
\fn k9Ifo2::openIFO(int _num)
|
|
*/
|
|
ifo_handle_t * k9Ifo2::openIFO(int _num) {
|
|
_ifo = ifoOpen(m_dvd->getDvd(), _num);
|
|
numIfo=_num;
|
|
return _ifo;
|
|
|
|
}
|
|
|
|
|
|
/*!
|
|
\fn k9Ifo2::IFOClose( ifo_handle_t* _ifo)
|
|
*/
|
|
void k9Ifo2::closeIFO() {
|
|
if(_ifo!=NULL) {
|
|
ifoClose(_ifo);
|
|
_ifo=NULL;
|
|
}
|
|
}
|
|
|
|
ifo_handle_t * k9Ifo2::getIFO() {
|
|
return _ifo;
|
|
}
|
|
|
|
/*!
|
|
\fn k9Ifo2::saveIFO(ifo_handle_t* _ifo)
|
|
*/
|
|
void k9Ifo2::saveIFO() {
|
|
uint32_t size;
|
|
/* DVD handler */
|
|
k9DVDFile *dvdfile;
|
|
bool mainIfo = (numIfo==0);
|
|
|
|
QString filename,filename2;
|
|
if (mainIfo) {
|
|
filename="/VIDEO_TS/VIDEO_TS.IFO";
|
|
} else {
|
|
filename.sprintf("/VIDEO_TS/VTS_%02d_0.IFO",numIfo);
|
|
}
|
|
if (numIfo==0)
|
|
size=_ifo->vmgi_mat->vmgi_last_sector +1;
|
|
else
|
|
size=_ifo->vtsi_mat->vtsi_last_sector +1;
|
|
|
|
size*=DVD_BLOCK_LEN;
|
|
// if (k9UDFFindFile(dvd,(char*) filename.latin1(), &size)) {
|
|
if (size > 0) {
|
|
uchar *buffer ;
|
|
buffer=new uchar[size];
|
|
bufCopy=new uchar[size];
|
|
m_buffer=new uchar[size*2];
|
|
memset(buffer,0,size);
|
|
memset(bufCopy,0,size);
|
|
memset(m_buffer,0,size*2);
|
|
//Lecture du fichier IFO original et sauvegarde dans buffer
|
|
if ((dvdfile = m_dvd->openIfo(numIfo))== 0) {
|
|
fprintf(stderr, "Failed opening IFO for titleset %d\n", 0);
|
|
free(buffer);
|
|
return;
|
|
}
|
|
|
|
/* if(DVDFileSeek(dvd_file, 0)!=0) {
|
|
return;
|
|
}
|
|
*/
|
|
|
|
if (dvdfile->readBytes(buffer,size) != size) {
|
|
fprintf(stderr, "Error reading IFO for titleset %d\n", 0);
|
|
free(buffer);
|
|
dvdfile->close();
|
|
return;
|
|
}
|
|
|
|
memcpy(bufCopy,buffer,size);
|
|
bufCopySize=size;
|
|
dvdfile->close();
|
|
if (mainIfo) {
|
|
m_position=0;
|
|
updateVMG(buffer);
|
|
m_position=1024; //sizeof(vmgi_mat_t);
|
|
updateFP_PGC(buffer);
|
|
_ifo->vmgi_mat->vmgi_last_byte=m_position -1;
|
|
|
|
updateTT_SRPT(buffer);
|
|
updatePGCI_UT(buffer);
|
|
|
|
updateVTS_ATRT(buffer);
|
|
updateTXTDT_MGI(buffer);
|
|
updateC_ADT(buffer,true);
|
|
updateVOBU_ADMAP(buffer,true);
|
|
|
|
updateVMG(buffer);
|
|
|
|
} else {
|
|
m_position=0;
|
|
updateVTS(buffer);
|
|
m_position=sizeof(vtsi_mat_t);
|
|
//_ifo->vtsi_mat->vtsi_last_byte=m_position -1;
|
|
updateVTS_PTT_SRPT(buffer);
|
|
updatePGCIT(buffer);
|
|
updatePGCI_UT(buffer);
|
|
updateVTS_TMAPT(buffer);
|
|
updateC_ADT(buffer,true);
|
|
updateVOBU_ADMAP(buffer,true);
|
|
updateC_ADT(buffer, false);
|
|
updateVOBU_ADMAP(buffer,false);
|
|
updateVTS(buffer);
|
|
}
|
|
m_position=round(m_position);
|
|
|
|
//sauvegarder buffer dans fichier
|
|
if (mainIfo) {
|
|
filename=output + "/VIDEO_TS.IFO";
|
|
filename2=output + "/VIDEO_TS.BUP";
|
|
} else {
|
|
filename.sprintf("/VTS_%02d_0.IFO",numIfo);
|
|
filename=output + filename;
|
|
|
|
filename2.sprintf("/VTS_%02d_0.BUP",numIfo);
|
|
filename2=output + filename2;
|
|
}
|
|
|
|
QFile ifofile (filename);
|
|
ifofile.open(IO_WriteOnly);
|
|
if (ifofile.writeBlock((char*)m_buffer,m_position) == -1) {
|
|
QString sError("erreur");
|
|
}
|
|
ifofile.close();
|
|
|
|
QFile ifofile2 (filename2);
|
|
ifofile2.open(IO_WriteOnly);
|
|
if (ifofile2.writeBlock((char*)m_buffer,m_position) == -1) {
|
|
QString sError("erreur");
|
|
}
|
|
ifofile2.close();
|
|
|
|
|
|
delete buffer;
|
|
delete bufCopy;
|
|
delete m_buffer;
|
|
}
|
|
}
|
|
|
|
|
|
/*!
|
|
\fn k9Ifo2::updateVMG(uchar * _buffer)
|
|
*/
|
|
void k9Ifo2::updateVMG(uchar * _buffer) {
|
|
vmgi_mat_t vmgi_mat;
|
|
memcpy(&vmgi_mat,_ifo->vmgi_mat,sizeof(vmgi_mat_t));
|
|
uint32_t lastSector=vmgi_mat.vmgi_last_sector;
|
|
uint32_t lastByte=vmgi_mat.vmgi_last_byte;
|
|
//JMP : à vérifier
|
|
if (m_position !=0) {
|
|
lastSector=(round(m_position) - DVD_BLOCK_LEN) / DVD_BLOCK_LEN;
|
|
//lastByte=vmgi_mat.vmgi_last_sector * DVD_BLOCK_LEN +DVD_BLOCK_LEN -1;
|
|
vmgi_mat.vmg_last_sector+=2*(lastSector-vmgi_mat.vmgi_last_sector);
|
|
}
|
|
|
|
vmgi_mat.vmgi_last_sector=lastSector;
|
|
if (vmgi_mat.vmgm_vobs !=0)
|
|
vmgi_mat.vmgm_vobs=vmgi_mat.vmgi_last_sector +1;
|
|
|
|
B2N_32(vmgi_mat.vmg_last_sector);
|
|
B2N_32(vmgi_mat.vmgi_last_sector);
|
|
B2N_32(vmgi_mat.vmg_category);
|
|
B2N_16(vmgi_mat.vmg_nr_of_volumes);
|
|
B2N_16(vmgi_mat.vmg_this_volume_nr);
|
|
B2N_16(vmgi_mat.vmg_nr_of_title_sets);
|
|
B2N_64(vmgi_mat.vmg_pos_code);
|
|
B2N_32(vmgi_mat.vmgi_last_byte);
|
|
B2N_32(vmgi_mat.first_play_pgc);
|
|
B2N_32(vmgi_mat.vmgm_vobs);
|
|
B2N_32(vmgi_mat.tt_srpt);
|
|
B2N_32(vmgi_mat.vmgm_pgci_ut);
|
|
B2N_32(vmgi_mat.ptl_mait);
|
|
B2N_32(vmgi_mat.vts_atrt);
|
|
B2N_32(vmgi_mat.txtdt_mgi);
|
|
B2N_32(vmgi_mat.vmgm_c_adt);
|
|
B2N_32(vmgi_mat.vmgm_vobu_admap);
|
|
B2N_16(vmgi_mat.vmgm_audio_attr.lang_code);
|
|
B2N_16(vmgi_mat.vmgm_subp_attr.lang_code);
|
|
|
|
memcpy(m_buffer,&vmgi_mat,sizeof(vmgi_mat_t));
|
|
}
|
|
|
|
|
|
/*!
|
|
\fn k9Ifo2::updateFP_PGC(uchar * _buffer)
|
|
*/
|
|
void k9Ifo2::updateFP_PGC(uchar * _buffer) {
|
|
//int offset=_ifo->vmgi_mat->first_play_pgc;
|
|
_ifo->vmgi_mat->first_play_pgc=m_position;
|
|
return updatePGC(_buffer,_ifo->first_play_pgc,m_position);
|
|
|
|
|
|
}
|
|
|
|
/*!
|
|
\fn k9Ifo2::updatePGC(uchar *_buffer,pgc_t *_pgc,int _offset)
|
|
*/
|
|
void k9Ifo2::updatePGC(uchar *_buffer,pgc_t *_pgc,int _offset) {
|
|
pgc_t pgc;
|
|
memcpy(&pgc,_pgc,sizeof(pgc_t));
|
|
uint start=m_position;
|
|
m_position+=PGC_SIZE;
|
|
|
|
if(pgc.command_tbl_offset != 0) {
|
|
pgc.command_tbl_offset=PGC_SIZE;
|
|
updatePGC_COMMAND_TBL(_buffer ,pgc.command_tbl ,m_position);
|
|
}
|
|
if(pgc.program_map_offset != 0) {
|
|
pgc.program_map_offset=m_position-start;
|
|
updatePGC_PROGRAM_MAP(_buffer, pgc.program_map,pgc.nr_of_programs ,m_position);
|
|
}
|
|
if(pgc.cell_playback_offset != 0) {
|
|
pgc.cell_playback_offset=m_position-start;
|
|
updatePGC_CELL_PLAYBACK_TBL(_buffer ,pgc.cell_playback,pgc.nr_of_cells,m_position);
|
|
}
|
|
|
|
if(pgc.cell_position_offset != 0) {
|
|
pgc.cell_position_offset=m_position-start;
|
|
updatePGC_CELL_POSITION_TBL(_buffer,pgc.cell_position,pgc.nr_of_cells,m_position);
|
|
}
|
|
|
|
B2N_16(pgc.command_tbl_offset);
|
|
B2N_16(pgc.next_pgc_nr);
|
|
B2N_16(pgc.prev_pgc_nr);
|
|
B2N_16(pgc.goup_pgc_nr);
|
|
B2N_16(pgc.program_map_offset);
|
|
B2N_16(pgc.cell_playback_offset);
|
|
B2N_16(pgc.cell_position_offset);
|
|
|
|
int i;
|
|
for(i = 0; i < 8; i++)
|
|
B2N_16(pgc.audio_control[i]);
|
|
for(i = 0; i < 32; i++)
|
|
B2N_32(pgc.subp_control[i]);
|
|
for(i = 0; i < 16; i++)
|
|
B2N_32(pgc.palette[i]);
|
|
|
|
memcpy(m_buffer+_offset,&pgc,PGC_SIZE);
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
\fn k9Ifo2::updatePGC_COMMAND_TBL(uchar *_buffer,pgc_command_tbl_t *_cmd_tbl,int offset)
|
|
*/
|
|
void k9Ifo2::updatePGC_COMMAND_TBL(uchar *_buffer,pgc_command_tbl_t *_cmd_tbl,int _offset) {
|
|
struct {
|
|
uint16_t nr_of_pre;
|
|
uint16_t nr_of_post;
|
|
uint16_t nr_of_cell;
|
|
uint16_t last_byte;
|
|
}
|
|
ATTRIBUTE_PACKED cmd_tbl;
|
|
|
|
memcpy(&cmd_tbl,_cmd_tbl,sizeof(cmd_tbl));
|
|
|
|
//moves the offset to save vm_cmd
|
|
m_position+=sizeof(cmd_tbl);
|
|
|
|
if(_cmd_tbl->nr_of_pre != 0) {
|
|
unsigned int pre_cmds_size = _cmd_tbl->nr_of_pre * COMMAND_DATA_SIZE;
|
|
memcpy(m_buffer + m_position,_cmd_tbl->pre_cmds,pre_cmds_size);
|
|
m_position+=pre_cmds_size;
|
|
}
|
|
if(_cmd_tbl->nr_of_post != 0) {
|
|
unsigned int post_cmds_size = _cmd_tbl->nr_of_post * COMMAND_DATA_SIZE;
|
|
memcpy(m_buffer + m_position,_cmd_tbl->post_cmds,post_cmds_size);
|
|
m_position+=post_cmds_size;
|
|
}
|
|
if(_cmd_tbl->nr_of_cell != 0) {
|
|
unsigned int cell_cmds_size = _cmd_tbl->nr_of_cell * COMMAND_DATA_SIZE;
|
|
memcpy(m_buffer +m_position,_cmd_tbl->cell_cmds,cell_cmds_size);
|
|
m_position+=cell_cmds_size;
|
|
}
|
|
B2N_16(cmd_tbl.nr_of_pre);
|
|
B2N_16(cmd_tbl.nr_of_post);
|
|
B2N_16(cmd_tbl.nr_of_cell);
|
|
cmd_tbl.last_byte=m_position-_offset-1;
|
|
B2N_16(cmd_tbl.last_byte);
|
|
memcpy(m_buffer+_offset,&cmd_tbl,sizeof(cmd_tbl));
|
|
|
|
|
|
}
|
|
|
|
|
|
/*!
|
|
\fn k9Ifo2::updatePGC_PROGRAM_MAP(uchar *_buffer, pgc_program_map_t *_program_map, int _nr, int_offset
|
|
*/
|
|
void k9Ifo2::updatePGC_PROGRAM_MAP(uchar *_buffer, pgc_program_map_t *_program_map, int _nr, int _offset) {
|
|
int size = _nr * sizeof(pgc_program_map_t);
|
|
memcpy(m_buffer+_offset, _program_map, size);
|
|
// pad to word boundary
|
|
size += size % 2;
|
|
m_position += size;
|
|
}
|
|
|
|
|
|
/*!
|
|
\fn k9Ifo2::updatePGC_CELL_PLAYBACK_TBL(uchar *_buffer, cell_playback_t *_cell_playback,int _nr, int _offset)
|
|
*/
|
|
void k9Ifo2::updatePGC_CELL_PLAYBACK_TBL(uchar *_buffer, cell_playback_t *_cell_playback,int _nr, int _offset) {
|
|
cell_playback_t *cell_playback;
|
|
int size = _nr * sizeof(cell_playback_t);
|
|
cell_playback=(cell_playback_t*) malloc(size);
|
|
memcpy(cell_playback,_cell_playback,size);
|
|
|
|
for( int i = 0; i < _nr; i++) {
|
|
B2N_32(cell_playback[i].first_sector);
|
|
B2N_32(cell_playback[i].first_ilvu_end_sector);
|
|
B2N_32(cell_playback[i].last_vobu_start_sector);
|
|
B2N_32(cell_playback[i].last_sector);
|
|
}
|
|
|
|
memcpy(m_buffer + _offset,cell_playback,size);
|
|
m_position+=size;
|
|
|
|
free(cell_playback);
|
|
}
|
|
|
|
|
|
/*!
|
|
\fn k9Ifo2::updatePGC_CELL_POSITION_TBL(uchar *_buffer,cell_position_t *_cell_position,int _nr, int _offset)
|
|
*/
|
|
void k9Ifo2::updatePGC_CELL_POSITION_TBL(uchar *_buffer,cell_position_t *_cell_position,int _nr, int _offset) {
|
|
cell_position_t * cell_position;
|
|
int size = _nr * sizeof(cell_position_t);
|
|
cell_position=(cell_position_t*)malloc(size);
|
|
memcpy(cell_position,_cell_position,size);
|
|
for( int i = 0; i < _nr; i++) {
|
|
B2N_16(cell_position[i].vob_id_nr);
|
|
}
|
|
memcpy(m_buffer + _offset,cell_position,size);
|
|
m_position+=size;
|
|
free(cell_position);
|
|
}
|
|
|
|
int k9Ifo2::round(int _value) {
|
|
return ((_value-1)|(DVD_BLOCK_LEN-1))+1; // DVD_BLOCK_LEN-1 = 0x7FF
|
|
}
|
|
|
|
/*!
|
|
\fn k9Ifo2::updateTT_SRPT(uchar *_buffer)
|
|
*/
|
|
void k9Ifo2::updateTT_SRPT(uchar *_buffer) {
|
|
if(_ifo->vmgi_mat->tt_srpt != 0) {
|
|
tt_srpt_t * tt_srpt;
|
|
tt_srpt=(tt_srpt_t*) malloc(sizeof(tt_srpt_t));
|
|
// int offset= _ifo->vmgi_mat->tt_srpt * DVD_BLOCK_LEN;
|
|
|
|
m_position=round(m_position);
|
|
int offset=m_position;
|
|
_ifo->vmgi_mat->tt_srpt=m_position/DVD_BLOCK_LEN;
|
|
|
|
memcpy (tt_srpt,_ifo->tt_srpt,sizeof(tt_srpt_t));
|
|
|
|
|
|
int info_length = tt_srpt->last_byte + 1 - TT_SRPT_SIZE;
|
|
title_info_t * title_info;
|
|
title_info =(title_info_t*) malloc(info_length);
|
|
memcpy(title_info, tt_srpt->title,info_length);
|
|
|
|
for(int i = 0; i < tt_srpt->nr_of_srpts; i++) {
|
|
B2N_16(title_info[i].nr_of_ptts);
|
|
B2N_16(title_info[i].parental_id);
|
|
B2N_32(title_info[i].title_set_sector);
|
|
}
|
|
memcpy(m_buffer+offset+TT_SRPT_SIZE,title_info,info_length);
|
|
|
|
free(title_info);
|
|
m_position +=info_length;
|
|
|
|
B2N_16(tt_srpt->nr_of_srpts);
|
|
B2N_32(tt_srpt->last_byte);
|
|
memcpy(m_buffer+offset,tt_srpt,TT_SRPT_SIZE);
|
|
free (tt_srpt);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
void k9Ifo2::updatePGCI_UT(uchar *_buffer) {
|
|
int sector,sector2;
|
|
if(_ifo->vmgi_mat) {
|
|
if(_ifo->vmgi_mat->vmgm_pgci_ut == 0)
|
|
return;
|
|
m_position=round(m_position);
|
|
_ifo->vmgi_mat->vmgm_pgci_ut=m_position/DVD_BLOCK_LEN;
|
|
} else if(_ifo->vtsi_mat) {
|
|
if(_ifo->vtsi_mat->vtsm_pgci_ut == 0)
|
|
return;
|
|
m_position=round(m_position);
|
|
_ifo->vtsi_mat->vtsm_pgci_ut=m_position/DVD_BLOCK_LEN;
|
|
} else {
|
|
return;
|
|
}
|
|
|
|
|
|
sector2=sector=m_position;
|
|
if (_ifo->pgci_ut !=NULL) {
|
|
pgci_ut_t * pgci_ut;
|
|
pgci_ut = (pgci_ut_t*) malloc( sizeof(pgci_ut_t));
|
|
memcpy (pgci_ut,_ifo->pgci_ut,sizeof(pgci_ut_t));
|
|
m_position+=PGCI_UT_SIZE;
|
|
sector2=m_position;
|
|
|
|
pgci_lu_t pgci_lu[_ifo->pgci_ut->nr_of_lus];
|
|
m_position+=_ifo->pgci_ut->nr_of_lus*PGCI_LU_SIZE;
|
|
memcpy(pgci_lu,_ifo->pgci_ut->lu,_ifo->pgci_ut->nr_of_lus*sizeof(pgci_lu_t));
|
|
|
|
for(int i = 0; i < _ifo->pgci_ut->nr_of_lus; i++) {
|
|
B2N_16(pgci_lu[i].lang_code);
|
|
pgci_lu[i].lang_start_byte=m_position - sector;
|
|
B2N_32(pgci_lu[i].lang_start_byte);
|
|
updatePGCIT_internal(_buffer,_ifo->pgci_ut->lu[i].pgcit,m_position);
|
|
}
|
|
for (int i=0;i <_ifo->pgci_ut->nr_of_lus;i++)
|
|
memcpy(m_buffer+sector2+i*PGCI_LU_SIZE ,&(pgci_lu[i]),PGCI_LU_SIZE);
|
|
|
|
B2N_16(pgci_ut->nr_of_lus);
|
|
pgci_ut->last_byte=m_position-sector-1;
|
|
B2N_32(pgci_ut->last_byte);
|
|
memcpy(m_buffer+sector,pgci_ut,PGCI_UT_SIZE);
|
|
free(pgci_ut);
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void k9Ifo2::updatePGCIT(uchar *_buffer) {
|
|
if(!_ifo->vtsi_mat)
|
|
return ;
|
|
|
|
if(_ifo->vtsi_mat->vts_pgcit == 0) /* mandatory */
|
|
return ;
|
|
m_position=round(m_position);
|
|
_ifo->vtsi_mat->vts_pgcit=m_position / DVD_BLOCK_LEN;
|
|
updatePGCIT_internal(_buffer,_ifo->vts_pgcit,_ifo->vtsi_mat->vts_pgcit * DVD_BLOCK_LEN);
|
|
}
|
|
|
|
|
|
void k9Ifo2::updatePGCIT_internal(uchar *_buffer, pgcit_t *_pgcit, int _offset) {
|
|
pgcit_t * pgcit;
|
|
pgcit=(pgcit_t*)malloc(sizeof(pgcit_t));
|
|
memcpy(pgcit,_pgcit,sizeof(pgcit_t));
|
|
|
|
int offset=m_position+PGCIT_SIZE;
|
|
m_position+=PGCIT_SIZE;
|
|
|
|
pgci_srp_t pgci_srp[_pgcit->nr_of_pgci_srp];
|
|
memcpy(pgci_srp,_pgcit->pgci_srp,sizeof(pgci_srp_t)*_pgcit->nr_of_pgci_srp);
|
|
|
|
m_position+=_pgcit->nr_of_pgci_srp*PGCI_SRP_SIZE;
|
|
|
|
for(int i = 0; i < _pgcit->nr_of_pgci_srp; i++) {
|
|
B2N_16(pgci_srp[i].ptl_id_mask);
|
|
pgci_srp[i].pgc_start_byte=m_position-_offset;
|
|
B2N_32(pgci_srp[i].pgc_start_byte); //JMP:faux
|
|
updatePGC(_buffer,_pgcit->pgci_srp[i].pgc,m_position);
|
|
}
|
|
|
|
for(int i = 0; i < _pgcit->nr_of_pgci_srp; i++)
|
|
memcpy(m_buffer+offset+i*PGCI_SRP_SIZE,&(pgci_srp[i]),PGCI_SRP_SIZE);
|
|
|
|
B2N_16(pgcit->nr_of_pgci_srp);
|
|
pgcit->last_byte=m_position-_offset-1;
|
|
B2N_32(pgcit->last_byte);
|
|
memcpy(m_buffer+_offset ,pgcit,PGCIT_SIZE);
|
|
free(pgcit);
|
|
|
|
}
|
|
|
|
|
|
void k9Ifo2::updatePTL_MAIT(uchar *_buffer) {
|
|
if(!_ifo->vmgi_mat)
|
|
return;
|
|
|
|
_ifo->vmgi_mat->ptl_mait = 0;
|
|
return;
|
|
}
|
|
|
|
void k9Ifo2::updateVTS_ATRT(uchar *_buffer) {
|
|
if(_ifo->vmgi_mat->vts_atrt == 0)
|
|
return;
|
|
uint32_t orig=_ifo->vmgi_mat->vts_atrt * DVD_BLOCK_LEN;
|
|
m_position=round(m_position);
|
|
_ifo->vmgi_mat->vts_atrt=m_position/DVD_BLOCK_LEN;
|
|
|
|
memcpy(m_buffer+m_position,_buffer+orig,_ifo->vts_atrt->last_byte+1);
|
|
|
|
m_position+=_ifo->vts_atrt->last_byte+1;
|
|
/*
|
|
int sector = _ifo->vmgi_mat->vts_atrt * DVD_BLOCK_LEN;
|
|
vts_atrt_t *vts_atrt;
|
|
vts_atrt = (vts_atrt_t*)malloc(sizeof(vts_atrt_t));
|
|
memcpy(vts_atrt,_ifo->vts_atrt,VTS_ATRT_SIZE);
|
|
B2N_16(vts_atrt->nr_of_vtss);
|
|
B2N_32(vts_atrt->last_byte);
|
|
memcpy(m_buffer+sector,vts_atrt,VTS_ATRT_SIZE);
|
|
free(vts_atrt);
|
|
m_position+=VTS_ATRT_SIZE;
|
|
|
|
sector += VTS_ATRT_SIZE;
|
|
|
|
memcpy(m_buffer+sector,_buffer +orig+VTS_ATRT_SIZE ,VTS_ATTRIBUTES_SIZE*_ifo->vts_atrt->nr_of_vtss);
|
|
|
|
m_position+=VTS_ATTRIBUTES_SIZE*_ifo->vts_atrt->nr_of_vtss;
|
|
*/
|
|
}
|
|
|
|
void k9Ifo2::updateTXTDT_MGI(uchar * _buffer) {
|
|
if(_ifo->vmgi_mat->txtdt_mgi == 0)
|
|
return;
|
|
|
|
struct {
|
|
char disc_name[15];
|
|
char nr_of_language_units;
|
|
uint32_t last_byte;
|
|
txtdt_lu_t *lu;
|
|
} ATTRIBUTE_PACKED txtdtmgi;
|
|
|
|
|
|
m_position=round(m_position);
|
|
int orig=_ifo->vmgi_mat->txtdt_mgi*DVD_BLOCK_LEN;
|
|
int offset=m_position;
|
|
_ifo->vmgi_mat->txtdt_mgi =m_position/ DVD_BLOCK_LEN;
|
|
memcpy(&txtdtmgi,_buffer+orig,sizeof(txtdtmgi));
|
|
|
|
B2N_32(txtdtmgi.last_byte);
|
|
memcpy(m_buffer+offset,_buffer+orig ,txtdtmgi.last_byte+1);
|
|
m_position+=txtdtmgi.last_byte+1;
|
|
}
|
|
|
|
void k9Ifo2::updateC_ADT(uchar * _buffer, bool _isMenu) {
|
|
if(_ifo->vmgi_mat) {
|
|
if(_ifo->vmgi_mat->vmgm_c_adt != 0) {
|
|
m_position =round(m_position); //
|
|
_ifo->vmgi_mat->vmgm_c_adt=m_position/ DVD_BLOCK_LEN;
|
|
updateC_ADT_Internal(_buffer,_ifo->menu_c_adt,m_position);
|
|
}
|
|
} else if(_ifo->vtsi_mat) {
|
|
if(_ifo->vtsi_mat->vtsm_c_adt != 0 && _isMenu) {
|
|
m_position=round(m_position); //sector =
|
|
_ifo->vtsi_mat->vtsm_c_adt=m_position / DVD_BLOCK_LEN;
|
|
updateC_ADT_Internal(_buffer,_ifo->menu_c_adt,m_position);
|
|
}
|
|
if (_ifo->vtsi_mat->vts_c_adt !=0 && !_isMenu) {
|
|
m_position=round(m_position); //sector =
|
|
_ifo->vtsi_mat->vts_c_adt=m_position / DVD_BLOCK_LEN;
|
|
updateC_ADT_Internal(_buffer,_ifo->vts_c_adt,m_position);
|
|
}
|
|
} else {
|
|
return ;
|
|
}
|
|
|
|
}
|
|
|
|
void k9Ifo2::updateC_ADT_Internal(uchar *_buffer,c_adt_t *_c_adt,int _sector) {
|
|
c_adt_t * c_adt;
|
|
c_adt =(c_adt_t*) malloc (sizeof(c_adt_t));
|
|
memcpy(c_adt,_c_adt,sizeof(c_adt_t));
|
|
|
|
|
|
int offset =_sector + C_ADT_SIZE;
|
|
m_position+=C_ADT_SIZE;
|
|
|
|
int info_length = _c_adt->last_byte + 1 - C_ADT_SIZE;
|
|
|
|
cell_adr_t *cell_adr,*ptr;
|
|
cell_adr=(cell_adr_t*) malloc(sizeof(cell_adr_t));
|
|
|
|
ptr= _c_adt->cell_adr_table;
|
|
for(int i = 0; i < info_length/sizeof(cell_adr_t); i++) {
|
|
memcpy(cell_adr,&(ptr[i]),sizeof(cell_adr_t));
|
|
B2N_16(cell_adr->vob_id);
|
|
B2N_32(cell_adr->start_sector);
|
|
B2N_32(cell_adr->last_sector);
|
|
memcpy(m_buffer+offset,cell_adr,sizeof(cell_adr_t));
|
|
offset+=sizeof(cell_adr_t);
|
|
//ptr+=sizeof(cell_adr_t);
|
|
}
|
|
|
|
m_position+=info_length;
|
|
free(cell_adr);
|
|
|
|
B2N_16(c_adt->nr_of_vobs);
|
|
c_adt->last_byte=m_position-_sector-1;
|
|
B2N_32(c_adt->last_byte);
|
|
memcpy(m_buffer+_sector,c_adt,C_ADT_SIZE);
|
|
free(c_adt);
|
|
}
|
|
|
|
void k9Ifo2::updateVOBU_ADMAP(uchar * _buffer, bool _isMenu) {
|
|
int sector;
|
|
if(_ifo->vmgi_mat) {
|
|
|
|
if(_ifo->vmgi_mat->vmgm_vobu_admap == 0)
|
|
return ;
|
|
sector = m_position=round(m_position);//_ifo->vmgi_mat->vmgm_vobu_admap * DVD_BLOCK_LEN;
|
|
_ifo->vmgi_mat->vmgm_vobu_admap=m_position/DVD_BLOCK_LEN;
|
|
updateVOBU_ADMAP_Internal(_buffer,_ifo->menu_vobu_admap,sector);
|
|
|
|
} else if(_ifo->vtsi_mat) {
|
|
|
|
if(_ifo->vtsi_mat->vtsm_vobu_admap != 0 && _isMenu) {
|
|
sector = m_position=round(m_position);//sector = _ifo->vtsi_mat->vtsm_vobu_admap * DVD_BLOCK_LEN;
|
|
_ifo->vtsi_mat->vtsm_vobu_admap=m_position/DVD_BLOCK_LEN;
|
|
updateVOBU_ADMAP_Internal(_buffer,_ifo->menu_vobu_admap,sector);
|
|
}
|
|
if (_ifo->vtsi_mat->vts_vobu_admap !=0 && !_isMenu) {
|
|
sector = m_position=round(m_position);//sector = _ifo->vtsi_mat->vts_vobu_admap * DVD_BLOCK_LEN;
|
|
_ifo->vtsi_mat->vts_vobu_admap=m_position/DVD_BLOCK_LEN;
|
|
updateVOBU_ADMAP_Internal(_buffer,_ifo->vts_vobu_admap,sector);
|
|
}
|
|
} else {
|
|
return ;
|
|
}
|
|
}
|
|
|
|
void k9Ifo2::updateVOBU_ADMAP_Internal(uchar *_buffer,vobu_admap_t *_vobu_admap,int _sector) {
|
|
vobu_admap_t *vobu_admap;
|
|
vobu_admap=(vobu_admap_t*)malloc(sizeof(vobu_admap_t));
|
|
memcpy(vobu_admap,_vobu_admap,sizeof(vobu_admap_t));
|
|
|
|
int offset = _sector + VOBU_ADMAP_SIZE;
|
|
m_position+=VOBU_ADMAP_SIZE;
|
|
|
|
int info_length = _vobu_admap->last_byte + 1 - VOBU_ADMAP_SIZE;
|
|
uint32_t *vobu_start_sectors;
|
|
vobu_start_sectors=(uint32_t*)malloc(info_length);
|
|
memcpy(vobu_start_sectors,_vobu_admap->vobu_start_sectors,info_length);
|
|
|
|
for(int i = 0; i < info_length/sizeof(uint32_t); i++)
|
|
B2N_32(vobu_start_sectors[i]);
|
|
|
|
memcpy(m_buffer+offset,vobu_start_sectors,info_length);
|
|
|
|
m_position+=info_length;
|
|
free(vobu_start_sectors);
|
|
vobu_admap->last_byte=m_position-_sector-1;
|
|
B2N_32(vobu_admap->last_byte);
|
|
memcpy(m_buffer+_sector,vobu_admap,VOBU_ADMAP_SIZE);
|
|
free(vobu_admap);
|
|
|
|
}
|
|
|
|
void k9Ifo2::updateVTS(uchar *_buffer) {
|
|
vtsi_mat_t *vtsi_mat;
|
|
vtsi_mat = (vtsi_mat_t *)malloc(sizeof(vtsi_mat_t));
|
|
memcpy(vtsi_mat,_ifo->vtsi_mat,sizeof(vtsi_mat_t));
|
|
|
|
uint32_t lastSector=vtsi_mat->vtsi_last_sector;
|
|
//uint32_t lastByte=vtsi_mat->vtsi_last_byte;
|
|
//JMP : à vérifier
|
|
if (m_position >0) {
|
|
lastSector=(round(m_position)-DVD_BLOCK_LEN) /DVD_BLOCK_LEN;
|
|
//lastByte=vtsi_mat->vtsi_last_sector*DVD_BLOCK_LEN +DVD_BLOCK_LEN-1;
|
|
vtsi_mat->vts_last_sector+=2*(lastSector-vtsi_mat->vtsi_last_sector);
|
|
vtsi_mat->vtstt_vobs+=lastSector-vtsi_mat->vtsi_last_sector;
|
|
}
|
|
|
|
vtsi_mat->vtsi_last_sector=lastSector;
|
|
//vtsi_mat->vtsi_last_byte=lastByte;
|
|
if (vtsi_mat->vtsm_vobs !=0)
|
|
vtsi_mat->vtsm_vobs= vtsi_mat->vtsi_last_sector +1 ;
|
|
|
|
B2N_32(vtsi_mat->vts_last_sector);
|
|
B2N_32(vtsi_mat->vtsi_last_sector);
|
|
B2N_32(vtsi_mat->vts_category);
|
|
|
|
B2N_32(vtsi_mat->vtsi_last_byte);
|
|
B2N_32(vtsi_mat->vtsm_vobs);
|
|
B2N_32(vtsi_mat->vtstt_vobs);
|
|
B2N_32(vtsi_mat->vts_ptt_srpt);
|
|
B2N_32(vtsi_mat->vts_pgcit);
|
|
B2N_32(vtsi_mat->vtsm_pgci_ut);
|
|
B2N_32(vtsi_mat->vts_tmapt);
|
|
B2N_32(vtsi_mat->vtsm_c_adt);
|
|
B2N_32(vtsi_mat->vtsm_vobu_admap);
|
|
B2N_32(vtsi_mat->vts_c_adt);
|
|
B2N_32(vtsi_mat->vts_vobu_admap);
|
|
B2N_16(vtsi_mat->vtsm_audio_attr.lang_code);
|
|
B2N_16(vtsi_mat->vtsm_subp_attr.lang_code);
|
|
for(int i = 0; i < 8; i++)
|
|
B2N_16(vtsi_mat->vts_audio_attr[i].lang_code);
|
|
for(int i = 0; i < 32; i++)
|
|
B2N_16(vtsi_mat->vts_subp_attr[i].lang_code);
|
|
|
|
memcpy(m_buffer,vtsi_mat,sizeof(vtsi_mat_t));
|
|
|
|
free(vtsi_mat);
|
|
}
|
|
|
|
void k9Ifo2::updateVTS_PTT_SRPT(uchar *_buffer) {
|
|
if(!_ifo->vtsi_mat)
|
|
return ;
|
|
|
|
if(_ifo->vtsi_mat->vts_ptt_srpt == 0) /* mandatory */
|
|
return ;
|
|
|
|
vts_ptt_srpt_t * vts_ptt_srpt;
|
|
vts_ptt_srpt = (vts_ptt_srpt_t *)malloc(sizeof(vts_ptt_srpt_t));
|
|
memcpy(vts_ptt_srpt,_ifo->vts_ptt_srpt,sizeof(vts_ptt_srpt_t));
|
|
int orig=_ifo->vtsi_mat->vts_ptt_srpt * DVD_BLOCK_LEN;
|
|
int offset = m_position=round(m_position);//_ifo->vtsi_mat->vts_ptt_srpt * DVD_BLOCK_LEN;
|
|
_ifo->vtsi_mat->vts_ptt_srpt=m_position/DVD_BLOCK_LEN;
|
|
|
|
B2N_16(vts_ptt_srpt->nr_of_srpts);
|
|
B2N_32(vts_ptt_srpt->last_byte);
|
|
|
|
memcpy(m_buffer+offset,vts_ptt_srpt,VTS_PTT_SRPT_SIZE);
|
|
free(vts_ptt_srpt);
|
|
m_position+=VTS_PTT_SRPT_SIZE;
|
|
|
|
int info_length = _ifo->vts_ptt_srpt->last_byte + 1 - VTS_PTT_SRPT_SIZE;
|
|
|
|
memcpy(m_buffer+m_position,_buffer+orig+VTS_PTT_SRPT_SIZE, info_length);
|
|
m_position+=info_length;
|
|
}
|
|
|
|
|
|
/*!
|
|
\fn k9Ifo2::checkBuffer()
|
|
*/
|
|
|
|
|
|
void k9Ifo2::updateVTS_TMAPT(uchar *_buffer) {
|
|
if(!_ifo->vtsi_mat)
|
|
return ;
|
|
|
|
if(_ifo->vtsi_mat->vts_tmapt == 0) { /* optional(?) */
|
|
return ;
|
|
}
|
|
|
|
vts_tmapt_t *vts_tmapt1,*vts_tmapt;
|
|
vts_tmapt=_ifo->vts_tmapt;
|
|
int info_length = vts_tmapt->nr_of_tmaps * 4;
|
|
|
|
uint32_t offsets[vts_tmapt->nr_of_tmaps];
|
|
|
|
vts_tmapt1 = (vts_tmapt_t *)malloc(VTS_TMAPT_SIZE);
|
|
memcpy(vts_tmapt1,_ifo->vts_tmapt,VTS_TMAPT_SIZE);
|
|
uint32_t offset = m_position=round(m_position);
|
|
|
|
_ifo->vtsi_mat->vts_tmapt =m_position / DVD_BLOCK_LEN;
|
|
|
|
int offset0=offset;
|
|
offset+=VTS_TMAPT_SIZE;
|
|
m_position +=VTS_TMAPT_SIZE;
|
|
|
|
|
|
offset += info_length;
|
|
m_position+=info_length;
|
|
|
|
info_length=0;
|
|
vts_tmap_t *tmap=vts_tmapt->tmap;
|
|
//loop on tmaps to compute total size
|
|
for (int i=0;i <vts_tmapt->nr_of_tmaps;i++) {
|
|
int tmapSize=VTS_TMAP_SIZE+tmap[i].nr_of_entries*sizeof(map_ent_t);
|
|
info_length+=tmapSize+4;
|
|
}
|
|
|
|
tmap = (vts_tmap_t *)malloc(sizeof(vts_tmap_t)* vts_tmapt->nr_of_tmaps);
|
|
memcpy(tmap,vts_tmapt->tmap,sizeof(vts_tmap_t)* vts_tmapt->nr_of_tmaps);
|
|
|
|
vts_tmapt1->last_byte=VTS_TMAPT_SIZE+info_length-1;
|
|
for(int i = 0; i < vts_tmapt->nr_of_tmaps; i++) {
|
|
int tmapSize=VTS_TMAP_SIZE+tmap[i].nr_of_entries*sizeof(map_ent_t);
|
|
int nr_map_ent=tmap[i].nr_of_entries;
|
|
B2N_16(tmap[i].nr_of_entries);
|
|
offsets[i]=m_position-offset0;
|
|
B2N_32(offsets[i]);
|
|
memcpy(m_buffer+m_position,&(tmap[i]),VTS_TMAP_SIZE);
|
|
m_position+=VTS_TMAP_SIZE;
|
|
offset+=VTS_TMAP_SIZE;
|
|
|
|
if(nr_map_ent == 0) { // Early out if zero entries
|
|
continue;
|
|
}
|
|
int info_length2 = tmapSize-VTS_TMAP_SIZE;
|
|
map_ent_t *map_ent = (map_ent_t *)malloc(info_length2);
|
|
memcpy(map_ent,tmap[i].map_ent,info_length2);
|
|
for(int j = 0; j < nr_map_ent; j++) {
|
|
B2N_32(map_ent[j]);
|
|
memcpy(m_buffer+m_position,&(map_ent[j]),sizeof(map_ent_t));
|
|
offset+=sizeof(map_ent_t);
|
|
m_position+=sizeof(map_ent_t);
|
|
}
|
|
free(map_ent);
|
|
}
|
|
|
|
free(tmap);
|
|
B2N_16(vts_tmapt1->nr_of_tmaps);
|
|
B2N_32(vts_tmapt1->last_byte);
|
|
memcpy(m_buffer+offset0,vts_tmapt1,VTS_TMAPT_SIZE);
|
|
memcpy(m_buffer+offset0+VTS_TMAPT_SIZE,offsets,vts_tmapt->nr_of_tmaps*4);
|
|
free(vts_tmapt1);
|
|
|
|
|
|
}
|
|
void k9Ifo2::checkBuffer(QString lib,uchar* _buffer) {
|
|
/* for (int j=0;j<bufCopySize;j++) {
|
|
uchar *c,*c2;
|
|
c=(uchar*) (_buffer+j);
|
|
c2=(uchar*)(bufCopy+j);
|
|
if (*c!=*c2) {
|
|
return;
|
|
}
|
|
}
|
|
*/
|
|
return;
|
|
if (memcmp(bufCopy,_buffer,bufCopySize)!=0) {
|
|
QString c;
|
|
c="Buffer de sortie altéré: " +lib;
|
|
qDebug(c.latin1());
|
|
}
|
|
}
|
|
|
|
void k9Ifo2::navRead_PCI(pci_t *pci, uchar *buffer) {
|
|
int i, j;
|
|
|
|
memcpy(pci, buffer, sizeof(pci_t));
|
|
|
|
/* Endian conversions */
|
|
|
|
/* pci pci_gi */
|
|
B2N_32(pci->pci_gi.nv_pck_lbn);
|
|
B2N_16(pci->pci_gi.vobu_cat);
|
|
B2N_32(pci->pci_gi.vobu_s_ptm);
|
|
B2N_32(pci->pci_gi.vobu_e_ptm);
|
|
B2N_32(pci->pci_gi.vobu_se_e_ptm);
|
|
|
|
/* pci nsml_agli */
|
|
for(i = 0; i < 9; i++)
|
|
B2N_32(pci->nsml_agli.nsml_agl_dsta[i]);
|
|
|
|
/* pci hli hli_gi */
|
|
B2N_16(pci->hli.hl_gi.hli_ss);
|
|
B2N_32(pci->hli.hl_gi.hli_s_ptm);
|
|
B2N_32(pci->hli.hl_gi.hli_e_ptm);
|
|
B2N_32(pci->hli.hl_gi.btn_se_e_ptm);
|
|
|
|
/* pci hli btn_colit */
|
|
for(i = 0; i < 3; i++)
|
|
for(j = 0; j < 2; j++)
|
|
B2N_32(pci->hli.btn_colit.btn_coli[i][j]);
|
|
|
|
/* NOTE: I've had to change the structure from the disk layout to get
|
|
* the packing to work with Sun's Forte C compiler. */
|
|
|
|
/* pci hli btni */
|
|
for(i = 0; i < 36; i++) {
|
|
char tmp[sizeof(pci->hli.btnit[i])], swap;
|
|
memcpy(tmp, &(pci->hli.btnit[i]), sizeof(pci->hli.btnit[i]));
|
|
/* Byte 4 to 7 are 'rotated' was: ABCD EFGH IJ is: ABCG DEFH IJ */
|
|
swap = tmp[6];
|
|
tmp[6] = tmp[5];
|
|
tmp[5] = tmp[4];
|
|
tmp[4] = tmp[3];
|
|
tmp[3] = swap;
|
|
|
|
/* Then there are the two B2N_24(..) calls */
|
|
#ifndef WORDS_BIGENDIAN
|
|
swap = tmp[0];
|
|
tmp[0] = tmp[2];
|
|
tmp[2] = swap;
|
|
|
|
swap = tmp[4];
|
|
tmp[4] = tmp[6];
|
|
tmp[6] = swap;
|
|
#endif
|
|
memcpy(&(pci->hli.btnit[i]), tmp, sizeof(pci->hli.btnit[i]));
|
|
}
|
|
|
|
}
|