2008-01-10 15:20:57 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
cc -g -c isoburn.c
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
Class core of libisoburn.
|
|
|
|
|
|
|
|
Copyright 2007 Vreixo Formoso Lopes <metalpain2002@yahoo.es>
|
|
|
|
and Thomas Schmitt <scdbackup@gmx.net>
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* ( derived from stub generated by CgeN on Sat, 01 Sep 2007 12:04:36 GMT ) */
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2008-01-20 10:30:43 +00:00
|
|
|
#ifndef Xorriso_standalonE
|
|
|
|
|
2008-01-10 15:20:57 +00:00
|
|
|
#include <libburn/libburn.h>
|
|
|
|
|
2008-01-26 00:26:57 +00:00
|
|
|
#include <libisofs/libisofs.h>
|
2008-01-10 15:20:57 +00:00
|
|
|
|
2008-01-20 10:30:43 +00:00
|
|
|
#else /* ! Xorriso_standalonE */
|
|
|
|
|
|
|
|
#include "../libisofs/libisofs.h"
|
|
|
|
#include "../libburn/libburn.h"
|
|
|
|
|
|
|
|
#endif /* Xorriso_standalonE */
|
|
|
|
|
|
|
|
|
2008-01-10 15:20:57 +00:00
|
|
|
#include "libisoburn.h"
|
2008-01-26 12:06:26 +00:00
|
|
|
|
2008-01-10 15:20:57 +00:00
|
|
|
#include "isoburn.h"
|
|
|
|
|
2008-01-26 12:06:26 +00:00
|
|
|
#include "../version.h"
|
2008-01-10 15:20:57 +00:00
|
|
|
|
|
|
|
|
|
|
|
/* -------------------------- isoburn ----------------------- */
|
|
|
|
|
|
|
|
|
|
|
|
/* The global list of isoburn objects. Usually there is only one.
|
|
|
|
>>> we are not ready for multiple control threads yet. See >>> mutex .
|
|
|
|
Multiple burns under one control thread should work.
|
|
|
|
*/
|
|
|
|
struct isoburn *isoburn_list_start= NULL;
|
|
|
|
|
|
|
|
|
|
|
|
int isoburn_new(struct isoburn **objpt, int flag)
|
|
|
|
{
|
|
|
|
struct isoburn *o;
|
|
|
|
int i, ret;
|
|
|
|
|
|
|
|
*objpt= o= (struct isoburn *) malloc(sizeof(struct isoburn));
|
|
|
|
if(o==NULL)
|
|
|
|
return(-1);
|
|
|
|
|
|
|
|
o->drive= NULL;
|
|
|
|
o->emulation_mode= 0;
|
|
|
|
o->min_start_byte= 0;
|
|
|
|
o->nwa= 0;
|
2008-01-24 20:23:02 +00:00
|
|
|
|
|
|
|
#ifdef Libisoburn_no_fifO
|
|
|
|
o->iso_source= NULL;
|
|
|
|
#else
|
2008-01-10 15:20:57 +00:00
|
|
|
o->fifo= NULL;
|
2008-01-24 20:23:02 +00:00
|
|
|
#endif
|
|
|
|
|
2008-01-10 15:20:57 +00:00
|
|
|
o->wrote_well= -1;
|
|
|
|
o->fabricated_disc_status= BURN_DISC_UNREADY;
|
|
|
|
for(i=0;i<65536;i++)
|
|
|
|
o->target_iso_head[i]= 0;
|
|
|
|
o->image= NULL;
|
|
|
|
o->prev= NULL;
|
|
|
|
o->next= NULL;
|
2008-01-16 14:17:24 +00:00
|
|
|
ret= iso_image_new("ISOIMAGE", &o->image);
|
2008-01-10 15:20:57 +00:00
|
|
|
if(ret<0)
|
|
|
|
goto failed;
|
|
|
|
isoburn_link(o, isoburn_list_start, 1);
|
|
|
|
return(1);
|
|
|
|
failed:;
|
|
|
|
isoburn_destroy(objpt, 0);
|
|
|
|
return(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int isoburn_destroy(struct isoburn **objpt, int flag)
|
|
|
|
{
|
|
|
|
struct isoburn *o;
|
|
|
|
|
|
|
|
o= *objpt;
|
|
|
|
if(o==NULL)
|
|
|
|
return(0);
|
|
|
|
|
|
|
|
/* >>> mutex */
|
|
|
|
|
|
|
|
if(o==isoburn_list_start)
|
|
|
|
isoburn_list_start= o->next;
|
|
|
|
if(o->prev!=NULL)
|
|
|
|
o->prev->next= o->next;
|
|
|
|
if(o->next!=NULL)
|
|
|
|
o->next->prev= o->prev;
|
|
|
|
|
|
|
|
/* >>> end mutex */
|
|
|
|
|
|
|
|
if(o->image!=NULL)
|
|
|
|
iso_image_unref(o->image);
|
2008-01-24 20:23:02 +00:00
|
|
|
|
|
|
|
#ifdef Libisoburn_no_fifO
|
|
|
|
if(o->iso_source!=NULL)
|
|
|
|
burn_source_free(o->iso_source);
|
|
|
|
#else
|
2008-01-10 15:20:57 +00:00
|
|
|
if(o->fifo!=NULL)
|
|
|
|
burn_source_free(o->fifo);
|
2008-01-24 20:23:02 +00:00
|
|
|
#endif /* ! Libisoburn_no_fifO */
|
|
|
|
|
2008-01-10 15:20:57 +00:00
|
|
|
|
|
|
|
free((char *) o);
|
|
|
|
*objpt= NULL;
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int isoburn_destroy_all(struct isoburn **objpt, int flag)
|
|
|
|
{
|
|
|
|
struct isoburn *o,*n;
|
|
|
|
|
|
|
|
o= *objpt;
|
|
|
|
if(o==NULL)
|
|
|
|
return(0);
|
|
|
|
for(;o->prev!=NULL;o= o->prev);
|
|
|
|
for(;o!=NULL;o= n) {
|
|
|
|
n= o->next;
|
|
|
|
isoburn_destroy(&o,0);
|
|
|
|
}
|
|
|
|
*objpt= NULL;
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int isoburn_get_target_image(struct isoburn *o, IsoImage **pt, int flag)
|
|
|
|
{
|
|
|
|
*pt= o->image;
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int isoburn_get_prev(struct isoburn *o, struct isoburn **pt, int flag)
|
|
|
|
{
|
|
|
|
*pt= o->prev;
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int isoburn_get_next(struct isoburn *o, struct isoburn **pt, int flag)
|
|
|
|
{
|
|
|
|
*pt= o->next;
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int isoburn_link(struct isoburn *o, struct isoburn *link, int flag)
|
|
|
|
/*
|
|
|
|
bit0= insert as link->prev rather than as link->next
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
|
|
|
|
/* >>> mutex */
|
|
|
|
|
|
|
|
if(isoburn_list_start==NULL ||
|
|
|
|
(isoburn_list_start==link && (flag&1)))
|
|
|
|
isoburn_list_start= o;
|
|
|
|
if(o->prev!=NULL)
|
|
|
|
o->prev->next= o->next;
|
|
|
|
if(o->next!=NULL)
|
|
|
|
o->next->prev= o->prev;
|
|
|
|
o->prev= o->next= NULL;
|
|
|
|
if(link==NULL)
|
|
|
|
return(1);
|
|
|
|
if(flag&1) {
|
|
|
|
o->next= link;
|
|
|
|
o->prev= link->prev;
|
|
|
|
if(o->prev!=NULL)
|
|
|
|
o->prev->next= o;
|
|
|
|
link->prev= o;
|
|
|
|
} else {
|
|
|
|
o->prev= link;
|
|
|
|
o->next= link->next;
|
|
|
|
if(o->next!=NULL)
|
|
|
|
o->next->prev= o;
|
|
|
|
link->next= o;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* >>> end mutex */
|
|
|
|
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int isoburn_count(struct isoburn *o, int flag)
|
|
|
|
/* flag: bit1= count from start of list */
|
|
|
|
{
|
|
|
|
int counter= 0;
|
|
|
|
|
|
|
|
if(flag&2)
|
|
|
|
for(;o->prev!=NULL;o= o->prev);
|
|
|
|
for(;o!=NULL;o= o->next)
|
|
|
|
counter++;
|
|
|
|
return(counter);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int isoburn_by_idx(struct isoburn *o, int idx, struct isoburn **pt, int flag)
|
|
|
|
/* flag: bit0= fetch first (idx<0) or last (idx>0) item in list
|
|
|
|
bit1= address from start of list */
|
|
|
|
{
|
|
|
|
int i,abs_idx;
|
|
|
|
struct isoburn *npt;
|
|
|
|
|
|
|
|
if(flag&2)
|
|
|
|
for(;o->prev!=NULL;o= o->prev);
|
|
|
|
abs_idx= (idx>0?idx:-idx);
|
|
|
|
*pt= o;
|
|
|
|
for(i= 0;(i<abs_idx || (flag&1)) && *pt!=NULL;i++) {
|
|
|
|
if(idx>0)
|
|
|
|
npt= o->next;
|
|
|
|
else
|
|
|
|
npt= o->prev;
|
|
|
|
if(npt==NULL && (flag&1))
|
|
|
|
break;
|
|
|
|
*pt= npt;
|
|
|
|
}
|
|
|
|
return(*pt!=NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int isoburn_find_by_drive(struct isoburn **pt, struct burn_drive *d, int flag)
|
|
|
|
{
|
|
|
|
struct isoburn *o;
|
|
|
|
|
|
|
|
*pt= NULL;
|
|
|
|
for(o= isoburn_list_start;o!=NULL;o= o->next)
|
|
|
|
if(o->drive==d) {
|
|
|
|
*pt= o;
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
int isoburn_prepare_disc_aux(struct burn_drive *d, struct burn_disc **disc,
|
|
|
|
struct isoburn_source_opts *opts, int new_img)
|
|
|
|
{
|
|
|
|
struct burn_source *wsrc;
|
|
|
|
struct burn_session *session;
|
|
|
|
struct burn_track *track;
|
|
|
|
struct isoburn *o;
|
2008-01-28 17:20:10 +00:00
|
|
|
IsoWriteOpts *wopts= NULL;
|
2008-01-10 15:20:57 +00:00
|
|
|
enum burn_disc_status state;
|
2008-01-24 20:23:02 +00:00
|
|
|
int ret, fifo_chunks;
|
2008-01-10 15:20:57 +00:00
|
|
|
|
|
|
|
ret= isoburn_find_emulator(&o, d, 0);
|
|
|
|
if(ret<0 || o==NULL)
|
2008-01-28 17:20:10 +00:00
|
|
|
{ret= -1; goto ex;}
|
2008-01-10 15:20:57 +00:00
|
|
|
o->wrote_well= 0; /* early end will be registered as failure */
|
|
|
|
|
|
|
|
state = isoburn_disc_get_status(d);
|
|
|
|
if (state != BURN_DISC_BLANK && state != BURN_DISC_APPENDABLE
|
|
|
|
&& (state != BURN_DISC_FULL || ! new_img)) {
|
|
|
|
/* unsuitable status */
|
2008-01-28 17:20:10 +00:00
|
|
|
{ret= -2; goto ex;}
|
2008-01-10 15:20:57 +00:00
|
|
|
}
|
2008-01-24 20:23:02 +00:00
|
|
|
|
|
|
|
fifo_chunks= 32;
|
|
|
|
if(opts->fifo_size >= 64*1024 && opts->fifo_size <= 1024.0 * 1024.0 * 1024.0){
|
|
|
|
fifo_chunks= opts->fifo_size/2048;
|
|
|
|
if(fifo_chunks*2048 < opts->fifo_size)
|
|
|
|
fifo_chunks++;
|
|
|
|
}
|
2008-01-10 15:20:57 +00:00
|
|
|
|
2008-01-28 17:20:10 +00:00
|
|
|
ret = iso_write_opts_new(&wopts, 0);
|
|
|
|
if (ret < 0)
|
|
|
|
goto ex;
|
|
|
|
iso_write_opts_set_iso_level(wopts, opts->level);
|
|
|
|
iso_write_opts_set_rockridge(wopts, opts->rockridge);
|
|
|
|
iso_write_opts_set_joliet(wopts, opts->joliet);
|
|
|
|
iso_write_opts_set_iso1999(wopts, opts->iso1999);
|
|
|
|
iso_write_opts_set_omit_version_numbers(wopts, opts->omit_version_numbers);
|
|
|
|
iso_write_opts_set_allow_deep_paths(wopts, opts->allow_deep_paths);
|
|
|
|
iso_write_opts_set_allow_longer_paths(wopts, opts->allow_longer_paths);
|
|
|
|
iso_write_opts_set_max_37_char_filenames(wopts, opts->max_37_char_filenames);
|
|
|
|
iso_write_opts_set_no_force_dots(wopts, opts->no_force_dots);
|
|
|
|
iso_write_opts_set_allow_lowercase(wopts, opts->allow_lowercase);
|
|
|
|
iso_write_opts_set_allow_full_ascii(wopts, opts->allow_full_ascii);
|
|
|
|
iso_write_opts_set_joliet_longer_paths(wopts, opts->joliet_longer_paths);
|
|
|
|
iso_write_opts_set_sort_files(wopts, opts->sort_files);
|
|
|
|
iso_write_opts_set_replace_mode(wopts, opts->replace_dir_mode,
|
|
|
|
opts->replace_file_mode, opts->replace_uid, opts->replace_gid);
|
|
|
|
iso_write_opts_set_default_dir_mode(wopts, opts->dir_mode);
|
|
|
|
iso_write_opts_set_default_file_mode(wopts, opts->file_mode);
|
|
|
|
iso_write_opts_set_default_uid(wopts, opts->uid);
|
|
|
|
iso_write_opts_set_default_gid(wopts, opts->gid);
|
|
|
|
iso_write_opts_set_output_charset(wopts, opts->output_charset);
|
2008-01-24 20:23:02 +00:00
|
|
|
#ifdef Libisoburn_no_fifO
|
2008-01-28 17:20:10 +00:00
|
|
|
iso_write_opts_set_fifo_size(wopts, fifo_chunks);
|
2008-01-24 20:23:02 +00:00
|
|
|
#endif /* Libisoburn_no_fifO */
|
2008-01-10 15:20:57 +00:00
|
|
|
|
|
|
|
if (new_img) {
|
2008-01-28 17:20:10 +00:00
|
|
|
iso_write_opts_set_ms_block(wopts, 0);
|
|
|
|
iso_write_opts_set_appendable(wopts, 0);
|
|
|
|
iso_write_opts_set_overwrite_buf(wopts, NULL);
|
2008-01-10 15:20:57 +00:00
|
|
|
} else {
|
|
|
|
int lba, nwa;
|
|
|
|
ret = isoburn_disc_track_lba_nwa(d, NULL, 0, &lba, &nwa);
|
|
|
|
if (ret != 1)
|
2008-01-28 17:20:10 +00:00
|
|
|
{ret= -3; goto ex;}
|
2008-01-10 15:20:57 +00:00
|
|
|
if (nwa == 0 && state == BURN_DISC_APPENDABLE) {
|
|
|
|
/* invalid nwa */
|
2008-01-28 17:20:10 +00:00
|
|
|
{ret= -4; goto ex;}
|
2008-01-10 15:20:57 +00:00
|
|
|
}
|
2008-01-28 17:20:10 +00:00
|
|
|
iso_write_opts_set_ms_block(wopts, nwa);
|
|
|
|
iso_write_opts_set_appendable(wopts, 1);
|
|
|
|
iso_write_opts_set_overwrite_buf(wopts, o->target_iso_head);
|
2008-01-10 15:20:57 +00:00
|
|
|
}
|
|
|
|
|
2008-01-28 17:20:10 +00:00
|
|
|
ret = iso_image_create_burn_source(o->image, wopts, &wsrc);
|
2008-01-10 15:20:57 +00:00
|
|
|
if (ret < 0)
|
2008-01-28 17:20:10 +00:00
|
|
|
{ret= -1; goto ex;}
|
2008-01-10 15:20:57 +00:00
|
|
|
|
|
|
|
/* TODO check return values for failure. propertly clean-up on error */
|
|
|
|
|
2008-01-24 20:23:02 +00:00
|
|
|
#ifdef Libisoburn_no_fifO
|
|
|
|
o->iso_source= wsrc;
|
|
|
|
#else
|
|
|
|
o->fifo = burn_fifo_source_new(wsrc, 2048, fifo_chunks, 0);
|
2008-01-10 15:20:57 +00:00
|
|
|
burn_source_free(wsrc);
|
|
|
|
if (o->fifo == NULL) {
|
|
|
|
fprintf(stderr, "Cannot attach fifo\n");
|
2008-01-28 17:20:10 +00:00
|
|
|
{ret= -1; goto ex;}
|
2008-01-10 15:20:57 +00:00
|
|
|
}
|
2008-01-24 20:23:02 +00:00
|
|
|
#endif /* ! Libisoburn_no_fifO */
|
|
|
|
|
2008-01-10 15:20:57 +00:00
|
|
|
*disc = burn_disc_create();
|
|
|
|
session = burn_session_create();
|
|
|
|
burn_disc_add_session(*disc, session, BURN_POS_END);
|
|
|
|
track = burn_track_create();
|
2008-01-24 20:23:02 +00:00
|
|
|
|
|
|
|
#ifdef Libisoburn_no_fifO
|
|
|
|
burn_track_set_source(track, o->iso_source);
|
|
|
|
#else
|
2008-01-10 15:20:57 +00:00
|
|
|
burn_track_set_source(track, o->fifo);
|
2008-01-24 20:23:02 +00:00
|
|
|
#endif /* ! Libisoburn_no_fifO */
|
|
|
|
|
2008-01-10 15:20:57 +00:00
|
|
|
burn_session_add_track(session, track, BURN_POS_END);
|
|
|
|
|
|
|
|
/* give up local references */
|
|
|
|
burn_track_free(track);
|
|
|
|
burn_session_free(session);
|
|
|
|
|
|
|
|
o->wrote_well= -1; /* neutral */
|
2008-01-28 17:20:10 +00:00
|
|
|
ret= 1;
|
|
|
|
ex:
|
|
|
|
if(wopts!=NULL)
|
|
|
|
{iso_write_opts_free(wopts); wopts= NULL;}
|
|
|
|
return ret;
|
2008-01-10 15:20:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int isoburn_prepare_disc(struct burn_drive *d, struct burn_disc **disc,
|
|
|
|
struct isoburn_source_opts *opts)
|
|
|
|
{
|
|
|
|
return isoburn_prepare_disc_aux(d, disc, opts, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int isoburn_prepare_new_image(struct burn_drive *d, struct burn_disc **disc,
|
2008-01-26 17:12:47 +00:00
|
|
|
struct isoburn_source_opts *opts,
|
|
|
|
struct burn_drive *out_drive)
|
2008-01-10 15:20:57 +00:00
|
|
|
{
|
2008-01-26 17:12:47 +00:00
|
|
|
int ret;
|
|
|
|
struct isoburn *in_o, *out_o;
|
|
|
|
|
|
|
|
ret= isoburn_prepare_disc_aux(d, disc, opts, 1);
|
|
|
|
if (ret<=0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
#ifdef Libisoburn_no_fifO
|
|
|
|
/* Hand over source reference for optional fifo status inquiry */
|
|
|
|
if(out_drive==NULL)
|
|
|
|
return 1;
|
|
|
|
ret= isoburn_find_emulator(&out_o, out_drive, 0);
|
|
|
|
if(ret<0 || out_o==NULL)
|
|
|
|
return 1;
|
|
|
|
ret= isoburn_find_emulator(&in_o, d, 0);
|
|
|
|
if(ret<0 || in_o==NULL)
|
|
|
|
return 1; /* then without fifo status inquiry */
|
|
|
|
if(out_o->iso_source!=NULL)
|
|
|
|
burn_source_free(out_o->iso_source);
|
|
|
|
out_o->iso_source= in_o->iso_source;
|
|
|
|
in_o->iso_source= NULL;
|
|
|
|
#endif /* Libisoburn_no_fifO */
|
|
|
|
|
|
|
|
return 1;
|
2008-01-10 15:20:57 +00:00
|
|
|
}
|
2008-01-26 12:06:26 +00:00
|
|
|
|
|
|
|
|
|
|
|
void isoburn_version(int *major, int *minor, int *micro)
|
|
|
|
{
|
|
|
|
*major = ISOBURN_MAJOR_VERSION;
|
|
|
|
*minor = ISOBURN_MINOR_VERSION;
|
|
|
|
*micro = ISOBURN_MICRO_VERSION;
|
|
|
|
}
|
|
|
|
|