aboutsummaryrefslogtreecommitdiffstats
path: root/fs/hpfs/map.c
blob: 840d033ecee832561f7f9ed91cec78527a62e38f (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
/*
 *  linux/fs/hpfs/map.c
 *
 *  Mikulas Patocka (mikulas@artax.karlin.mff.cuni.cz), 1998-1999
 *
 *  mapping structures to memory with some minimal checks
 */

#include "hpfs_fn.h"

unsigned *hpfs_map_dnode_bitmap(struct super_block *s, struct quad_buffer_head *qbh)
{
	return hpfs_map_4sectors(s, hpfs_sb(s)->sb_dmap, qbh, 0);
}

unsigned int *hpfs_map_bitmap(struct super_block *s, unsigned bmp_block,
			 struct quad_buffer_head *qbh, char *id)
{
	secno sec;
	if (hpfs_sb(s)->sb_chk) if (bmp_block * 16384 > hpfs_sb(s)->sb_fs_size) {
		hpfs_error(s, "hpfs_map_bitmap called with bad parameter: %08x at %s", bmp_block, id);
		return NULL;
	}
	sec = hpfs_sb(s)->sb_bmp_dir[bmp_block];
	if (!sec || sec > hpfs_sb(s)->sb_fs_size-4) {
		hpfs_error(s, "invalid bitmap block pointer %08x -> %08x at %s", bmp_block, sec, id);
		return NULL;
	}
	return hpfs_map_4sectors(s, sec, qbh, 4);
}

/*
 * Load first code page into kernel memory, return pointer to 256-byte array,
 * first 128 bytes are uppercasing table for chars 128-255, next 128 bytes are
 * lowercasing table
 */

unsigned char *hpfs_load_code_page(struct super_block *s, secno cps)
{
	struct buffer_head *bh;
	secno cpds;
	unsigned cpi;
	unsigned char *ptr;
	unsigned char *cp_table;
	int i;
	struct code_page_data *cpd;
	struct code_page_directory *cp = hpfs_map_sector(s, cps, &bh, 0);
	if (!cp) return NULL;
	if (cp->magic != CP_DIR_MAGIC) {
		printk("HPFS: Code page directory magic doesn't match (magic = %08x)\n", cp->magic);
		brelse(bh);
		return NULL;
	}
	if (!cp->n_code_pages) {
		printk("HPFS: n_code_pages == 0\n");
		brelse(bh);
		return NULL;
	}
	cpds = cp->array[0].code_page_data;
	cpi = cp->array[0].index;
	brelse(bh);

	if (cpi >= 3) {
		printk("HPFS: Code page index out of array\n");
		return NULL;
	}
	
	if (!(cpd = hpfs_map_sector(s, cpds, &bh, 0))) return NULL;
	if ((unsigned)cpd->offs[cpi] > 0x178) {
		printk("HPFS: Code page index out of sector\n");
		brelse(bh);
		return NULL;
	}
	ptr = (unsigned char *)cpd + cpd->offs[cpi] + 6;
	if (!(cp_table = kmalloc(256, GFP_KERNEL))) {
		printk("HPFS: out of memory for code page table\n");
		brelse(bh);
		return NULL;
	}
	memcpy(cp_table, ptr, 128);
	brelse(bh);

	/* Try to build lowercasing table from uppercasing one */

	for (i=128; i<256; i++) cp_table[i]=i;
	for (i=128; i<256; i++) if (cp_table[i-128]!=i && cp_table[i-128]>=128)
		cp_table[cp_table[i-128]] = i;
	
	return cp_table;
}

secno *hpfs_load_bitmap_directory(struct super_block *s, secno bmp)
{
	struct buffer_head *bh;
	int n = (hpfs_sb(s)->sb_fs_size + 0x200000 - 1) >> 21;
	int i;
	secno *b;
	if (!(b = kmalloc(n * 512, GFP_KERNEL))) {
		printk("HPFS: can't allocate memory for bitmap directory\n");
		return NULL;
	}	
	for (i=0;i<n;i++) {
		secno *d = hpfs_map_sector(s, bmp+i, &bh, n - i - 1);
		if (!d) {
			kfree(b);
			return NULL;
		}
		memcpy((char *)b + 512 * i, d, 512);
		brelse(bh);
	}
	return b;
}

/*
 * Load fnode to memory
 */

struct fnode *hpfs_map_fnode(struct super_block *s, ino_t ino, struct buffer_head **bhp)
{
	struct fnode *fnode;
	if (hpfs_sb(s)->sb_chk) if (hpfs_chk_sectors(s, ino, 1, "fnode")) {
		return NULL;
	}
	if ((fnode = hpfs_map_sector(s, ino, bhp, FNODE_RD_AHEAD))) {
		if (hpfs_sb(s)->sb_chk) {
			struct extended_attribute *ea;
			struct extended_attribute *ea_end;
			if (fnode->magic != FNODE_MAGIC) {
				hpfs_error(s, "bad magic on fnode %08lx",
					(unsigned long)ino);
				goto bail;
			}
			if (!fnode->dirflag) {
				if ((unsigned)fnode->btree.n_used_nodes + (unsigned)fnode->btree.n_free_nodes !=
				    (fnode->btree.internal ? 12 : 8)) {
					hpfs_error(s,
					   "bad number of nodes in fnode %08lx",
					    (unsigned long)ino);
					goto bail;
				}
				if (fnode->btree.first_free !=
				    8 + fnode->btree.n_used_nodes * (fnode->btree.internal ? 8 : 12)) {
					hpfs_error(s,
					    "bad first_free pointer in fnode %08lx",
					    (unsigned long)ino);
					goto bail;
				}
			}
			if (fnode->ea_size_s && ((signed int)fnode->ea_offs < 0xc4 ||
			   (signed int)fnode->ea_offs + fnode->acl_size_s + fnode->ea_size_s > 0x200)) {
				hpfs_error(s,
					"bad EA info in fnode %08lx: ea_offs == %04x ea_size_s == %04x",
					(unsigned long)ino,
					fnode->ea_offs, fnode->ea_size_s);
				goto bail;
			}
			ea = fnode_ea(fnode);
			ea_end = fnode_end_ea(fnode);
			while (ea != ea_end) {
				if (ea > ea_end) {
					hpfs_error(s, "bad EA in fnode %08lx",
						(unsigned long)ino);
					goto bail;
				}
				ea = next_ea(ea);
			}
		}
	}
	return fnode;
	bail:
	brelse(*bhp);
	return NULL;
}

struct anode *hpfs_map_anode(struct super_block *s, anode_secno ano, struct buffer_head **bhp)
{
	struct anode *anode;
	if (hpfs_sb(s)->sb_chk) if (hpfs_chk_sectors(s, ano, 1, "anode")) return NULL;
	if ((anode = hpfs_map_sector(s, ano, bhp, ANODE_RD_AHEAD)))
		if (hpfs_sb(s)->sb_chk) {
			if (anode->magic != ANODE_MAGIC || anode->self != ano) {
				hpfs_error(s, "bad magic on anode %08x", ano);
				goto bail;
			}
			if ((unsigned)anode->btree.n_used_nodes + (unsigned)anode->btree.n_free_nodes !=
			    (anode->btree.internal ? 60 : 40)) {
				hpfs_error(s, "bad number of nodes in anode %08x", ano);
				goto bail;
			}
			if (anode->btree.first_free !=
			    8 + anode->btree.n_used_nodes * (anode->btree.internal ? 8 : 12)) {
				hpfs_error(s, "bad first_free pointer in anode %08x", ano);
				goto bail;
			}
		}
	return anode;
	bail:
	brelse(*bhp);
	return NULL;
}

/*
 * Load dnode to memory and do some checks
 */

struct dnode *hpfs_map_dnode(struct super_block *s, unsigned secno,
			     struct quad_buffer_head *qbh)
{
	struct dnode *dnode;
	if (hpfs_sb(s)->sb_chk) {
		if (hpfs_chk_sectors(s, secno, 4, "dnode")) return NULL;
		if (secno & 3) {
			hpfs_error(s, "dnode %08x not byte-aligned", secno);
			return NULL;
		}	
	}
	if ((dnode = hpfs_map_4sectors(s, secno, qbh, DNODE_RD_AHEAD)))
		if (hpfs_sb(s)->sb_chk) {
			unsigned p, pp = 0;
			unsigned char *d = (unsigned char *)dnode;
			int b = 0;
			if (dnode->magic != DNODE_MAGIC) {
				hpfs_error(s, "bad magic on dnode %08x", secno);
				goto bail;
			}
			if (dnode->self != secno)
				hpfs_error(s, "bad self pointer on dnode %08x self = %08x", secno, dnode->self);
			/* Check dirents - bad dirents would cause infinite
			   loops or shooting to memory */
			if (dnode->first_free > 2048/* || dnode->first_free < 84*/) {
				hpfs_error(s, "dnode %08x has first_free == %08x", secno, dnode->first_free);
				goto bail;
			}
			for (p = 20; p < dnode->first_free; p += d[p] + (d[p+1] << 8)) {
				struct hpfs_dirent *de = (struct hpfs_dirent *)((char *)dnode + p);
				if (de->length > 292 || (de->length < 32) || (de->length & 3) || p + de->length > 2048) {
					hpfs_error(s, "bad dirent size in dnode %08x, dirent %03x, last %03x", secno, p, pp);
					goto bail;
				}
				if (((31 + de->namelen + de->down*4 + 3) & ~3) != de->length) {
					if (((31 + de->namelen + de->down*4 + 3) & ~3) < de->length && s->s_flags & MS_RDONLY) goto ok;
					hpfs_error(s, "namelen does not match dirent size in dnode %08x, dirent %03x, last %03x", secno, p, pp);
					goto bail;
				}
				ok:
				if (hpfs_sb(s)->sb_chk >= 2) b |= 1 << de->down;
				if (de->down) if (de_down_pointer(de) < 0x10) {
					hpfs_error(s, "bad down pointer in dnode %08x, dirent %03x, last %03x", secno, p, pp);
					goto bail;
				}
				pp = p;
				
			}
			if (p != dnode->first_free) {
				hpfs_error(s, "size on last dirent does not match first_free; dnode %08x", secno);
				goto bail;
			}
			if (d[pp + 30] != 1 || d[pp + 31] != 255) {
				hpfs_error(s, "dnode %08x does not end with \\377 entry", secno);
				goto bail;
			}
			if (b == 3) printk("HPFS: warning: unbalanced dnode tree, dnode %08x; see hpfs.txt 4 more info\n", secno);
		}
	return dnode;
	bail:
	hpfs_brelse4(qbh);
	return NULL;
}

dnode_secno hpfs_fnode_dno(struct super_block *s, ino_t ino)
{
	struct buffer_head *bh;
	struct fnode *fnode;
	dnode_secno dno;

	fnode = hpfs_map_fnode(s, ino, &bh);
	if (!fnode)
		return 0;

	dno = fnode->u.external[0].disk_secno;
	brelse(bh);
	return dno;
}
* * (c) 1994 Eberhard Moenkeberg <emoenke@gwdg.de> * may be used & enhanced freely * * Due to non-existent sync bytes at the beginning of each audio frame (or due * to a firmware bug within all known drives?), it is currently a kind of * fortune if two consecutive frames fit together. * Usually, they overlap, or a little piece is missing. This happens in units * of 24-byte chunks. It has to get fixed by higher-level software (reading * until an overlap occurs, and then eliminate the overlapping chunks). * ftp.gwdg.de:/pub/linux/misc/cdda2wav-sbpcd.*.tar.gz holds an example of * such an algorithm. * This example program further is missing to obtain the SubChannel data * which belong to each frame. * * This is only an example of the low-level access routine. The read data are * pure 16-bit CDDA values; they have to get converted to make sound out of * them. * It is no fun to listen to it without prior overlap/underlap correction! */ #include <stdio.h> #include <sys/ioctl.h> #include <sys/types.h> #include <linux/cdrom.h> static struct cdrom_tochdr hdr; static struct cdrom_tocentry entry[101]; static struct cdrom_read_audio arg; static u_char buffer[CD_FRAMESIZE_RAW]; static int datafile, drive; static int i, j, limit, track, err; static char filename[32]; int main(int argc, char *argv[]) { /* * open /dev/cdrom */ drive=open("/dev/cdrom", 0); if (drive<0) { fprintf(stderr, "can't open drive.\n"); exit (-1); } /* * get TocHeader */ fprintf(stdout, "getting TocHeader...\n"); err=ioctl(drive, CDROMREADTOCHDR, &hdr); if (err!=0) { fprintf(stderr, "can't get TocHeader (error %d).\n", err); exit (-1); } else fprintf(stdout, "TocHeader: %d %d\n", hdr.cdth_trk0, hdr.cdth_trk1); /* * get and display all TocEntries */ fprintf(stdout, "getting TocEntries...\n"); for (i=1;i<=hdr.cdth_trk1+1;i++) { if (i!=hdr.cdth_trk1+1) entry[i].cdte_track = i; else entry[i].cdte_track = CDROM_LEADOUT; entry[i].cdte_format = CDROM_LBA; err=ioctl(drive, CDROMREADTOCENTRY, &entry[i]); if (err!=0) { fprintf(stderr, "can't get TocEntry #%d (error %d).\n", i, err); exit (-1); } else { fprintf(stdout, "TocEntry #%d: %1X %1X %06X %02X\n", entry[i].cdte_track, entry[i].cdte_adr, entry[i].cdte_ctrl, entry[i].cdte_addr.lba, entry[i].cdte_datamode); } } fprintf(stdout, "got all TocEntries.\n"); /* * ask for track number (not implemented here) */ track=1; #if 0 /* just read a little piece (4 seconds) */ entry[track+1].cdte_addr.lba=entry[track].cdte_addr.lba+300; #endif /* * read track into file */ sprintf(filename, "track%02d\0", track); datafile=creat(filename, 0755); if (datafile<0) { fprintf(stderr, "can't open datafile %s.\n", filename); exit (-1); } arg.addr.lba=entry[track].cdte_addr.lba; arg.addr_format=CDROM_LBA; /* CDROM_MSF would be possible here, too. */ arg.nframes=1; arg.buf=&buffer[0]; limit=entry[track+1].cdte_addr.lba; for (;arg.addr.lba<limit;arg.addr.lba++) { err=ioctl(drive, CDROMREADAUDIO, &arg); if (err!=0) { fprintf(stderr, "can't read abs. frame #%d (error %d).\n", arg.addr.lba, err); } j=write(datafile, &buffer[0], CD_FRAMESIZE_RAW); if (j!=CD_FRAMESIZE_RAW) { fprintf(stderr,"I/O error (datafile) at rel. frame %d\n", arg.addr.lba-entry[track].cdte_addr.lba); } arg.addr.lba++; } return 0; } /*===================== end program ========================================*/ At ftp.gwdg.de:/pub/linux/misc/cdda2wav-sbpcd.*.tar.gz is an adapted version of Heiko Eissfeldt's digital-audio to .WAV converter (the original is there, too). This is preliminary, as Heiko himself will care about it. Known problems: --------------- Currently, the detection of disk change or removal is actively disabled. Most attempts to read the UPC/EAN code result in a stream of zeroes. All my drives are mostly telling there is no UPC/EAN code on disk or there is, but it is an all-zero number. I guess now almost no CD holds such a number. Bug reports, comments, wishes, donations (technical information is a donation, too :-) etc. to emoenke@gwdg.de. SnailMail address, preferable for CD editors if they want to submit a free "cooperation" copy: Eberhard Moenkeberg Reinholdstr. 14 D-37083 Goettingen Germany --- Appendix -- the "cdtester" utility: /* * cdtester.c -- test the audio functions of a CD driver * * (c) 1995 Eberhard Moenkeberg <emoenke@gwdg.de> * published under the GPL * * made under heavy use of the "Tiny Audio CD Player" * from Werner Zimmermann <zimmerma@rz.fht-esslingen.de> * (see linux/drivers/block/README.aztcd) */ #undef AZT_PRIVATE_IOCTLS /* not supported by every CDROM driver */ #define SBP_PRIVATE_IOCTLS /* not supported by every CDROM driver */ #include <stdio.h> #include <stdio.h> #include <malloc.h> #include <sys/ioctl.h> #include <sys/types.h> #include <linux/cdrom.h> #ifdef AZT_PRIVATE_IOCTLS #include <linux/../../drivers/cdrom/aztcd.h> #endif /* AZT_PRIVATE_IOCTLS */ #ifdef SBP_PRIVATE_IOCTLS #include <linux/../../drivers/cdrom/sbpcd.h> #include <linux/fs.h> #endif /* SBP_PRIVATE_IOCTLS */ struct cdrom_tochdr hdr; struct cdrom_tochdr tocHdr; struct cdrom_tocentry TocEntry[101]; struct cdrom_tocentry entry; struct cdrom_multisession ms_info; struct cdrom_read_audio read_audio; struct cdrom_ti ti; struct cdrom_subchnl subchnl; struct cdrom_msf msf; struct cdrom_volctrl volctrl; #ifdef AZT_PRIVATE_IOCTLS union { struct cdrom_msf msf; unsigned char buf[CD_FRAMESIZE_RAW]; } azt; #endif /* AZT_PRIVATE_IOCTLS */ int i, i1, i2, i3, j, k; unsigned char sequence=0; unsigned char command[80]; unsigned char first=1, last=1; char *default_device="/dev/cdrom"; char dev[20]; char filename[20]; int drive; int datafile; int rc; void help(void) { printf("Available Commands:\n"); printf("STOP s EJECT e QUIT q\n"); printf("PLAY TRACK t PAUSE p RESUME r\n"); printf("NEXT TRACK n REPEAT LAST l HELP h\n"); printf("SUBCHANNEL_Q c TRACK INFO i PLAY AT a\n"); printf("READ d READ RAW w READ AUDIO A\n"); printf("MS-INFO M TOC T START S\n"); printf("SET EJECTSW X DEVICE D DEBUG Y\n"); printf("AUDIO_BUFSIZ Z RESET R SET VOLUME v\n"); printf("GET VOLUME V\n"); } /* * convert MSF number (3 bytes only) to Logical_Block_Address */ int msf2lba(u_char *msf) { int i; i=(msf[0] * CD_SECS + msf[1]) * CD_FRAMES + msf[2] - CD_BLOCK_OFFSET; if (i<0) return (0); return (i); } /* * convert logical_block_address to m-s-f_number (3 bytes only) */ void lba2msf(int lba, unsigned char *msf) { lba += CD_BLOCK_OFFSET; msf[0] = lba / (CD_SECS*CD_FRAMES); lba %= CD_SECS*CD_FRAMES; msf[1] = lba / CD_FRAMES; msf[2] = lba % CD_FRAMES; } int init_drive(char *dev) { unsigned char msf_ent[3]; /* * open the device */ drive=open(dev,0); if (drive<0) return (-1); /* * get TocHeader */ printf("getting TocHeader...\n"); rc=ioctl(drive,CDROMREADTOCHDR,&hdr); if (rc!=0) { printf("can't get TocHeader (error %d).\n",rc); return (-2); } else first=hdr.cdth_trk0; last=hdr.cdth_trk1; printf("TocHeader: %d %d\n",hdr.cdth_trk0,hdr.cdth_trk1); /* * get and display all TocEntries */ printf("getting TocEntries...\n"); for (i=1;i<=hdr.cdth_trk1+1;i++) { if (i!=hdr.cdth_trk1+1) TocEntry[i].cdte_track = i; else TocEntry[i].cdte_track = CDROM_LEADOUT; TocEntry[i].cdte_format = CDROM_LBA; rc=ioctl(drive,CDROMREADTOCENTRY,&TocEntry[i]); if (rc!=0) { printf("can't get TocEntry #%d (error %d).\n",i,rc); } else { lba2msf(TocEntry[i].cdte_addr.lba,&msf_ent[0]); if (TocEntry[i].cdte_track==CDROM_LEADOUT) { printf("TocEntry #%02X: %1X %1X %02d:%02d:%02d (lba: 0x%06X) %02X\n", TocEntry[i].cdte_track, TocEntry[i].cdte_adr, TocEntry[i].cdte_ctrl, msf_ent[0], msf_ent[1], msf_ent[2], TocEntry[i].cdte_addr.lba, TocEntry[i].cdte_datamode); } else { printf("TocEntry #%02d: %1X %1X %02d:%02d:%02d (lba: 0x%06X) %02X\n", TocEntry[i].cdte_track, TocEntry[i].cdte_adr, TocEntry[i].cdte_ctrl, msf_ent[0], msf_ent[1], msf_ent[2], TocEntry[i].cdte_addr.lba, TocEntry[i].cdte_datamode); } } } return (hdr.cdth_trk1); /* number of tracks */ } void display(int size,unsigned char *buffer) { k=0; getchar(); for (i=0;i<(size+1)/16;i++) { printf("%4d:",i*16); for (j=0;j<16;j++) { printf(" %02X",buffer[i*16+j]); } printf(" "); for (j=0;j<16;j++) { if (isalnum(buffer[i*16+j])) printf("%c",buffer[i*16+j]); else printf("."); } printf("\n"); k++; if (k>=20) { printf("press ENTER to continue\n"); getchar(); k=0; } } } int main(int argc, char *argv[]) { printf("\nTesting tool for a CDROM driver's audio functions V0.1\n"); printf("(C) 1995 Eberhard Moenkeberg <emoenke@gwdg.de>\n"); printf("initializing...\n"); rc=init_drive(default_device); if (rc<0) printf("could not open %s (rc=%d).\n",default_device,rc); help(); while (1) { printf("Give a one-letter command (h = help): "); scanf("%s",command); command[1]=0; switch (command[0]) { case 'D': printf("device name (f.e. /dev/sbpcd3): ? "); scanf("%s",&dev); close(drive); rc=init_drive(dev); if (rc<0) printf("could not open %s (rc %d).\n",dev,rc); break; case 'e': rc=ioctl(drive,CDROMEJECT); if (rc<0) printf("CDROMEJECT: rc=%d.\n",rc); break; case 'p': rc=ioctl(drive,CDROMPAUSE); if (rc<0) printf("CDROMPAUSE: rc=%d.\n",rc); break; case 'r': rc=ioctl(drive,CDROMRESUME); if (rc<0) printf("CDROMRESUME: rc=%d.\n",rc); break; case 's': rc=ioctl(drive,CDROMSTOP); if (rc<0) printf("CDROMSTOP: rc=%d.\n",rc); break; case 'S': rc=ioctl(drive,CDROMSTART); if (rc<0) printf("CDROMSTART: rc=%d.\n",rc); break; case 't': rc=ioctl(drive,CDROMREADTOCHDR,&tocHdr); if (rc<0) { printf("CDROMREADTOCHDR: rc=%d.\n",rc); break; } first=tocHdr.cdth_trk0; last= tocHdr.cdth_trk1; if ((first==0)||(first>last)) { printf ("--got invalid TOC data.\n"); } else { printf("--enter track number(first=%d, last=%d): ",first,last); scanf("%d",&i1); ti.cdti_trk0=i1; if (ti.cdti_trk0<first) ti.cdti_trk0=first; if (ti.cdti_trk0>last) ti.cdti_trk0=last; ti.cdti_ind0=0; ti.cdti_trk1=last; ti.cdti_ind1=0; rc=ioctl(drive,CDROMSTOP); rc=ioctl(drive,CDROMPLAYTRKIND,&ti); if (rc<0) printf("CDROMPLAYTRKIND: rc=%d.\n",rc); } break; case 'n': rc=ioctl(drive,CDROMSTOP); if (++ti.cdti_trk0>last) ti.cdti_trk0=last; ti.cdti_ind0=0; ti.cdti_trk1=last; ti.cdti_ind1=0; rc=ioctl(drive,CDROMPLAYTRKIND,&ti); if (rc<0) printf("CDROMPLAYTRKIND: rc=%d.\n",rc); break; case 'l': rc=ioctl(drive,CDROMSTOP); if (--ti.cdti_trk0<first) ti.cdti_trk0=first; ti.cdti_ind0=0; ti.cdti_trk1=last; ti.cdti_ind1=0; rc=ioctl(drive,CDROMPLAYTRKIND,&ti); if (rc<0) printf("CDROMPLAYTRKIND: rc=%d.\n",rc); break; case 'c': subchnl.cdsc_format=CDROM_MSF; rc=ioctl(drive,CDROMSUBCHNL,&subchnl); if (rc<0) printf("CDROMSUBCHNL: rc=%d.\n",rc); else { printf("AudioStatus:%s Track:%d Mode:%d MSF=%02d:%02d:%02d\n", subchnl.cdsc_audiostatus==CDROM_AUDIO_PLAY ? "PLAYING":"NOT PLAYING", subchnl.cdsc_trk,subchnl.cdsc_adr, subchnl.cdsc_absaddr.msf.minute, subchnl.cdsc_absaddr.msf.second, subchnl.cdsc_absaddr.msf.frame); } break; case 'i': printf("Track No.: "); scanf("%d",&i1); entry.cdte_track=i1; if (entry.cdte_track<first) entry.cdte_track=first; if (entry.cdte_track>last) entry.cdte_track=last; entry.cdte_format=CDROM_MSF; rc=ioctl(drive,CDROMREADTOCENTRY,&entry); if (rc<0) printf("CDROMREADTOCENTRY: rc=%d.\n",rc); else { printf("Mode %d Track, starts at %02d:%02d:%02d\n", entry.cdte_adr, entry.cdte_addr.msf.minute, entry.cdte_addr.msf.second, entry.cdte_addr.msf.frame); } break; case 'a': printf("Address (min:sec:frm) "); scanf("%d:%d:%d",&i1,&i2,&i3); msf.cdmsf_min0=i1; msf.cdmsf_sec0=i2; msf.cdmsf_frame0=i3; if (msf.cdmsf_sec0>59) msf.cdmsf_sec0=59; if (msf.cdmsf_frame0>74) msf.cdmsf_frame0=74; lba2msf(TocEntry[last+1].cdte_addr.lba-1,&msf.cdmsf_min1); rc=ioctl(drive,CDROMSTOP); rc=ioctl(drive,CDROMPLAYMSF,&msf); if (rc<0) printf("CDROMPLAYMSF: rc=%d.\n",rc); break; case 'V': rc=ioctl(drive,CDROMVOLREAD,&volctrl); if (rc<0) printf("CDROMVOLCTRL: rc=%d.\n",rc); printf("Volume: channel 0 (left) %d, channel 1 (right) %d\n",volctrl.channel0,volctrl.channel1); break; case 'R': rc=ioctl(drive,CDROMRESET); if (rc<0) printf("CDROMRESET: rc=%d.\n",rc); break; #ifdef AZT_PRIVATE_IOCTLS /*not supported by every CDROM driver*/ case 'd': printf("Address (min:sec:frm) "); scanf("%d:%d:%d",&i1,&i2,&i3); azt.msf.cdmsf_min0=i1; azt.msf.cdmsf_sec0=i2; azt.msf.cdmsf_frame0=i3; if (azt.msf.cdmsf_sec0>59) azt.msf.cdmsf_sec0=59; if (azt.msf.cdmsf_frame0>74) azt.msf.cdmsf_frame0=74; rc=ioctl(drive,CDROMREADMODE1,&azt.msf); if (rc<0) printf("CDROMREADMODE1: rc=%d.\n",rc); else display(CD_FRAMESIZE,azt.buf); break; case 'w': printf("Address (min:sec:frame) "); scanf("%d:%d:%d",&i1,&i2,&i3); azt.msf.cdmsf_min0=i1; azt.msf.cdmsf_sec0=i2; azt.msf.cdmsf_frame0=i3; if (azt.msf.cdmsf_sec0>59) azt.msf.cdmsf_sec0=59; if (azt.msf.cdmsf_frame0>74) azt.msf.cdmsf_frame0=74; rc=ioctl(drive,CDROMREADMODE2,&azt.msf); if (rc<0) printf("CDROMREADMODE2: rc=%d.\n",rc); else display(CD_FRAMESIZE_RAW,azt.buf); /* currently only 2336 */ break; #endif case 'v': printf("--Channel 0 (Left) (0-255): "); scanf("%d",&i1); volctrl.channel0=i1; printf("--Channel 1 (Right) (0-255): "); scanf("%d",&i1); volctrl.channel1=i1; volctrl.channel2=0; volctrl.channel3=0; rc=ioctl(drive,CDROMVOLCTRL,&volctrl); if (rc<0) printf("CDROMVOLCTRL: rc=%d.\n",rc); break; case 'q': close(drive); exit(0); case 'h': help(); break; case 'T': /* display TOC entry - without involving the driver */ scanf("%d",&i); if ((i<hdr.cdth_trk0)||(i>hdr.cdth_trk1)) printf("invalid track number.\n"); else printf("TocEntry %02d: adr=%01X ctrl=%01X msf=%02d:%02d:%02d mode=%02X\n", TocEntry[i].cdte_track, TocEntry[i].cdte_adr, TocEntry[i].cdte_ctrl, TocEntry[i].cdte_addr.msf.minute, TocEntry[i].cdte_addr.msf.second, TocEntry[i].cdte_addr.msf.frame, TocEntry[i].cdte_datamode); break; case 'A': /* read audio data into file */ printf("Address (min:sec:frm) ? "); scanf("%d:%d:%d",&i1,&i2,&i3); read_audio.addr.msf.minute=i1; read_audio.addr.msf.second=i2; read_audio.addr.msf.frame=i3; read_audio.addr_format=CDROM_MSF; printf("# of frames ? "); scanf("%d",&i1); read_audio.nframes=i1; k=read_audio.nframes*CD_FRAMESIZE_RAW; read_audio.buf=malloc(k); if (read_audio.buf==NULL) { printf("can't malloc %d bytes.\n",k); break; } sprintf(filename,"audio_%02d%02d%02d_%02d.%02d\0", read_audio.addr.msf.minute, read_audio.addr.msf.second, read_audio.addr.msf.frame, read_audio.nframes, ++sequence); datafile=creat(filename, 0755); if (datafile<0) { printf("can't open datafile %s.\n",filename); break; } rc=ioctl(drive,CDROMREADAUDIO,&read_audio); if (rc!=0) { printf("CDROMREADAUDIO: rc=%d.\n",rc); } else { rc=write(datafile,&read_audio.buf,k); if (rc!=k) printf("datafile I/O error (%d).\n",rc); } close(datafile); break; case 'X': /* set EJECT_SW (0: disable, 1: enable auto-ejecting) */ scanf("%d",&i); rc=ioctl(drive,CDROMEJECT_SW,i); if (rc!=0) printf("CDROMEJECT_SW: rc=%d.\n",rc); else printf("EJECT_SW set to %d\n",i); break; case 'M': /* get the multisession redirection info */ ms_info.addr_format=CDROM_LBA; rc=ioctl(drive,CDROMMULTISESSION,&ms_info); if (rc!=0) { printf("CDROMMULTISESSION(lba): rc=%d.\n",rc); } else { if (ms_info.xa_flag) printf("MultiSession offset (lba): %d (0x%06X)\n",ms_info.addr.lba,ms_info.addr.lba); else { printf("this CD is not an XA disk.\n"); break; } } ms_info.addr_format=CDROM_MSF; rc=ioctl(drive,CDROMMULTISESSION,&ms_info); if (rc!=0) { printf("CDROMMULTISESSION(msf): rc=%d.\n",rc); } else { if (ms_info.xa_flag) printf("MultiSession offset (msf): %02d:%02d:%02d (0x%02X%02X%02X)\n", ms_info.addr.msf.minute, ms_info.addr.msf.second, ms_info.addr.msf.frame, ms_info.addr.msf.minute, ms_info.addr.msf.second, ms_info.addr.msf.frame); else printf("this CD is not an XA disk.\n"); } break; #ifdef SBP_PRIVATE_IOCTLS case 'Y': /* set the driver's message level */ #if 0 /* not implemented yet */ printf("enter switch name (f.e. DBG_CMD): "); scanf("%s",&dbg_switch); j=get_dbg_num(dbg_switch); #else printf("enter DDIOCSDBG switch number: "); scanf("%d",&j); #endif printf("enter 0 for \"off\", 1 for \"on\": "); scanf("%d",&i); if (i==0) j|=0x80; printf("calling \"ioctl(drive,DDIOCSDBG,%d)\"\n",j); rc=ioctl(drive,DDIOCSDBG,j); printf("DDIOCSDBG: rc=%d.\n",rc); break; case 'Z': /* set the audio buffer size */ printf("# frames wanted: ? "); scanf("%d",&j); rc=ioctl(drive,CDROMAUDIOBUFSIZ,j); printf("%d frames granted.\n",rc); break; #endif /* SBP_PRIVATE_IOCTLS */ default: printf("unknown command: \"%s\".\n",command); break; } } return 0; } /*==========================================================================*/