1f0d9efc0Sbeck /*
2f0d9efc0Sbeck * File joliet.c - handle Win95/WinNT long file/unicode extensions for iso9660.
3f0d9efc0Sbeck
4f0d9efc0Sbeck Copyright 1997 Eric Youngdale.
5f0d9efc0Sbeck
6f0d9efc0Sbeck This program is free software; you can redistribute it and/or modify
7f0d9efc0Sbeck it under the terms of the GNU General Public License as published by
8f0d9efc0Sbeck the Free Software Foundation; either version 2, or (at your option)
9f0d9efc0Sbeck any later version.
10f0d9efc0Sbeck
11f0d9efc0Sbeck This program is distributed in the hope that it will be useful,
12f0d9efc0Sbeck but WITHOUT ANY WARRANTY; without even the implied warranty of
13f0d9efc0Sbeck MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14f0d9efc0Sbeck GNU General Public License for more details.
15f0d9efc0Sbeck
16f0d9efc0Sbeck You should have received a copy of the GNU General Public License
17f0d9efc0Sbeck along with this program; if not, write to the Free Software
18f0d9efc0Sbeck Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
19f0d9efc0Sbeck
20f0d9efc0Sbeck /* APPLE_HYB James Pearson j.pearson@ge.ucl.ac.uk 12/3/99 */
21f0d9efc0Sbeck
22f0d9efc0Sbeck /*
23f0d9efc0Sbeck * Joliet extensions for ISO9660. These are spottily documented by
24f0d9efc0Sbeck * Microsoft. In their infinite stupidity, they completely ignored
25f0d9efc0Sbeck * the possibility of using an SUSP record with the long filename
26f0d9efc0Sbeck * in it, and instead wrote out a duplicate directory tree with the
27f0d9efc0Sbeck * long filenames in it.
28f0d9efc0Sbeck *
29f0d9efc0Sbeck * I am not sure why they did this. One reason is that they get the path
30f0d9efc0Sbeck * tables with the long filenames in them.
31f0d9efc0Sbeck *
32f0d9efc0Sbeck * There are two basic principles to Joliet, and the non-Unicode variant
33f0d9efc0Sbeck * known as Romeo. Long filenames seem to be the main one, and the second
34f0d9efc0Sbeck * is that the character set and a few other things is substantially relaxed.
35f0d9efc0Sbeck *
36f0d9efc0Sbeck * The SVD is identical to the PVD, except:
37f0d9efc0Sbeck *
38f0d9efc0Sbeck * Id is 2, not 1 (indicates SVD).
39f0d9efc0Sbeck * escape_sequences contains UCS-2 indicator (levels 1, 2 or 3).
40f0d9efc0Sbeck * The root directory record points to a different extent (with different
41f0d9efc0Sbeck * size).
42f0d9efc0Sbeck * There are different path tables for the two sets of directory trees.
43f0d9efc0Sbeck *
44f0d9efc0Sbeck * The following fields are recorded in Unicode:
45f0d9efc0Sbeck * system_id
46f0d9efc0Sbeck * volume_id
47f0d9efc0Sbeck * volume_set_id
48f0d9efc0Sbeck * publisher_id
49f0d9efc0Sbeck * preparer_id
50f0d9efc0Sbeck * application_id
51f0d9efc0Sbeck * copyright_file_id
52f0d9efc0Sbeck * abstract_file_id
53f0d9efc0Sbeck * bibliographic_file_id
54f0d9efc0Sbeck *
55f0d9efc0Sbeck * Unicode strings are always encoded in big-endian format.
56f0d9efc0Sbeck *
57f0d9efc0Sbeck * In a directory record, everything is the same as with iso9660, except
58f0d9efc0Sbeck * that the name is recorded in unicode. The name length is specified in
59f0d9efc0Sbeck * total bytes, not in number of unicode characters.
60f0d9efc0Sbeck *
61f0d9efc0Sbeck * The character set used for the names is different with UCS - the
62f0d9efc0Sbeck * restrictions are that the following are not allowed:
63f0d9efc0Sbeck *
64f0d9efc0Sbeck * Characters (00)(00) through (00)(1f) (control chars)
65f0d9efc0Sbeck * (00)(2a) '*'
66f0d9efc0Sbeck * (00)(2f) '/'
67f0d9efc0Sbeck * (00)(3a) ':'
68f0d9efc0Sbeck * (00)(3b) ';'
69f0d9efc0Sbeck * (00)(3f) '?'
70f0d9efc0Sbeck * (00)(5c) '\'
71f0d9efc0Sbeck */
72f0d9efc0Sbeck #include "config.h"
73f0d9efc0Sbeck #include "mkisofs.h"
74f0d9efc0Sbeck #include "iso9660.h"
75f0d9efc0Sbeck
7668cbdb5eSespie
77f0d9efc0Sbeck #include <stdlib.h>
78f0d9efc0Sbeck #include <time.h>
79f0d9efc0Sbeck
80f0d9efc0Sbeck static int jpath_table_index;
81f0d9efc0Sbeck static struct directory ** jpathlist;
82f0d9efc0Sbeck static int next_jpath_index = 1;
83f0d9efc0Sbeck static int sort_goof;
84f0d9efc0Sbeck
85f0d9efc0Sbeck static int generate_joliet_path_tables __PR((void));
86f0d9efc0Sbeck static int DECL(joliet_sort_directory, (struct directory_entry ** sort_dir));
87f0d9efc0Sbeck static void DECL(assign_joliet_directory_addresses, (struct directory * node));
88f0d9efc0Sbeck static int jroot_gen __PR((void));
89f0d9efc0Sbeck
90f0d9efc0Sbeck /*
91f0d9efc0Sbeck * Function: convert_to_unicode
92f0d9efc0Sbeck *
93f0d9efc0Sbeck * Purpose: Perform a 1/2 assed unicode conversion on a text
94f0d9efc0Sbeck * string.
95f0d9efc0Sbeck *
96f0d9efc0Sbeck * Notes:
97f0d9efc0Sbeck */
FDECL3(convert_to_unicode,unsigned char *,buffer,int,size,char *,source)98f0d9efc0Sbeck static void FDECL3(convert_to_unicode, unsigned char *, buffer, int, size, char *, source )
99f0d9efc0Sbeck {
100f0d9efc0Sbeck unsigned char * tmpbuf;
101f0d9efc0Sbeck int i;
102f0d9efc0Sbeck int j;
103f0d9efc0Sbeck
104f0d9efc0Sbeck /*
105f0d9efc0Sbeck * If we get a NULL pointer for the source, it means we have an inplace
106f0d9efc0Sbeck * copy, and we need to make a temporary working copy first.
107f0d9efc0Sbeck */
108f0d9efc0Sbeck if( source == NULL )
109f0d9efc0Sbeck {
110f0d9efc0Sbeck tmpbuf = (u_char *) e_malloc(size);
111f0d9efc0Sbeck memcpy( tmpbuf, buffer, size);
112f0d9efc0Sbeck }
113f0d9efc0Sbeck else
114f0d9efc0Sbeck {
115f0d9efc0Sbeck tmpbuf = (u_char *)source;
116f0d9efc0Sbeck }
117f0d9efc0Sbeck
118f0d9efc0Sbeck /*
119f0d9efc0Sbeck * Now start copying characters. If the size was specified to be 0, then
120f0d9efc0Sbeck * assume the input was 0 terminated.
121f0d9efc0Sbeck */
122f0d9efc0Sbeck j = 0;
123f0d9efc0Sbeck for(i=0; i < size ; i += 2, j++)
124f0d9efc0Sbeck {
125f0d9efc0Sbeck buffer[i] = 0;
126f0d9efc0Sbeck /*
127f0d9efc0Sbeck * JS integrated from: Achim_Kaiser@t-online.de
128f0d9efc0Sbeck *
129f0d9efc0Sbeck * Let all valid unicode characters pass through (assuming ISO-8859-1).
130f0d9efc0Sbeck * Others are set to '_' .
131f0d9efc0Sbeck */
132f0d9efc0Sbeck if( tmpbuf[j] != 0 &&
133f0d9efc0Sbeck (tmpbuf[j] <= 0x1f || (tmpbuf[j] >= 0x7F && tmpbuf[j] <= 0xA0)) )
134f0d9efc0Sbeck {
135f0d9efc0Sbeck buffer[i+1] = '_';
136f0d9efc0Sbeck }
137f0d9efc0Sbeck else
138f0d9efc0Sbeck {
139f0d9efc0Sbeck switch(tmpbuf[j])
140f0d9efc0Sbeck {
141f0d9efc0Sbeck case '*':
142f0d9efc0Sbeck case '/':
143f0d9efc0Sbeck case ':':
144f0d9efc0Sbeck case ';':
145f0d9efc0Sbeck case '?':
146f0d9efc0Sbeck case '\\':
147f0d9efc0Sbeck /*
148f0d9efc0Sbeck * Even Joliet has some standards as to what is allowed in a pathname.
149f0d9efc0Sbeck * Pretty tame in comparison to what DOS restricts you to.
150f0d9efc0Sbeck */
151f0d9efc0Sbeck buffer[i+1] = '_';
152f0d9efc0Sbeck break;
153f0d9efc0Sbeck default:
154f0d9efc0Sbeck buffer[i+1] = tmpbuf[j];
155f0d9efc0Sbeck break;
156f0d9efc0Sbeck }
157f0d9efc0Sbeck }
158f0d9efc0Sbeck }
159f0d9efc0Sbeck
160f0d9efc0Sbeck if( source == NULL )
161f0d9efc0Sbeck {
162f0d9efc0Sbeck free(tmpbuf);
163f0d9efc0Sbeck }
164f0d9efc0Sbeck }
165f0d9efc0Sbeck
166f0d9efc0Sbeck /*
167f0d9efc0Sbeck * Function: joliet_strlen
168f0d9efc0Sbeck *
169f0d9efc0Sbeck * Purpose: Return length in bytes of string after conversion to unicode.
170f0d9efc0Sbeck *
171f0d9efc0Sbeck * Notes: This is provided mainly as a convenience so that when more intelligent
172f0d9efc0Sbeck * Unicode conversion for either Multibyte or 8-bit codes is available that
173f0d9efc0Sbeck * we can easily adapt.
174f0d9efc0Sbeck */
FDECL1(joliet_strlen,const char *,string)175f0d9efc0Sbeck static int FDECL1(joliet_strlen, const char *, string)
176f0d9efc0Sbeck {
177f0d9efc0Sbeck int rtn;
178f0d9efc0Sbeck
179f0d9efc0Sbeck rtn = strlen(string) << 1;
180f0d9efc0Sbeck
181f0d9efc0Sbeck /*
182f0d9efc0Sbeck * We do clamp the maximum length of a Joliet string to be the
183f0d9efc0Sbeck * maximum path size. This helps to ensure that we don't completely
184f0d9efc0Sbeck * bolix things up with very long paths. The Joliet specs say
185f0d9efc0Sbeck * that the maximum length is 128 bytes, or 64 unicode characters.
186f0d9efc0Sbeck */
187f0d9efc0Sbeck if( rtn > 0x80)
188f0d9efc0Sbeck {
189f0d9efc0Sbeck rtn = 0x80;
190f0d9efc0Sbeck }
191f0d9efc0Sbeck return rtn;
192f0d9efc0Sbeck }
193f0d9efc0Sbeck
194f0d9efc0Sbeck /*
195f0d9efc0Sbeck * Function: get_joliet_vol_desc
196f0d9efc0Sbeck *
197f0d9efc0Sbeck * Purpose: generate a Joliet compatible volume desc.
198f0d9efc0Sbeck *
199f0d9efc0Sbeck * Notes: Assume that we have the non-joliet vol desc
200f0d9efc0Sbeck * already present in the buffer. Just modifiy the
201f0d9efc0Sbeck * appropriate fields.
202f0d9efc0Sbeck */
FDECL1(get_joliet_vol_desc,struct iso_primary_descriptor *,jvol_desc)203f0d9efc0Sbeck static void FDECL1(get_joliet_vol_desc, struct iso_primary_descriptor *, jvol_desc)
204f0d9efc0Sbeck {
205f0d9efc0Sbeck jvol_desc->type[0] = ISO_VD_SUPPLEMENTARY;
206f0d9efc0Sbeck
207f0d9efc0Sbeck /*
208f0d9efc0Sbeck * For now, always do Unicode level 3. I don't really know what 1 and 2
209f0d9efc0Sbeck * are - perhaps a more limited Unicode set.
210f0d9efc0Sbeck *
211f0d9efc0Sbeck * FIXME(eric) - how does Romeo fit in here? As mkisofs just
212f0d9efc0Sbeck * "expands" 8 bit character codes to 16 bits and does nothing
213f0d9efc0Sbeck * special with the Unicode characters, therefore shouldn't mkisofs
214f0d9efc0Sbeck * really be stating that it's using UCS-2 Level 1, not Level 3 for
215f0d9efc0Sbeck * the Joliet directory tree.
216f0d9efc0Sbeck */
217f0d9efc0Sbeck strcpy(jvol_desc->escape_sequences, "%/@");
218f0d9efc0Sbeck
219f0d9efc0Sbeck /*
220f0d9efc0Sbeck * Until we have Unicode path tables, leave these unset.
221f0d9efc0Sbeck */
222f0d9efc0Sbeck set_733((char *) jvol_desc->path_table_size, jpath_table_size);
223f0d9efc0Sbeck set_731(jvol_desc->type_l_path_table, jpath_table[0]);
224f0d9efc0Sbeck set_731(jvol_desc->opt_type_l_path_table, jpath_table[1]);
225f0d9efc0Sbeck set_732(jvol_desc->type_m_path_table, jpath_table[2]);
226f0d9efc0Sbeck set_732(jvol_desc->opt_type_m_path_table, jpath_table[3]);
227f0d9efc0Sbeck
228f0d9efc0Sbeck /*
229f0d9efc0Sbeck * Set this one up.
230f0d9efc0Sbeck */
231f0d9efc0Sbeck memcpy(jvol_desc->root_directory_record, &jroot_record,
232*463a9befSmiod sizeof(jvol_desc->root_directory_record));
233f0d9efc0Sbeck
234f0d9efc0Sbeck /*
235f0d9efc0Sbeck * Finally, we have a bunch of strings to convert to Unicode.
236f0d9efc0Sbeck * FIXME(eric) - I don't know how to do this in general, so we will
237f0d9efc0Sbeck * just be really lazy and do a char -> short conversion. We probably
238f0d9efc0Sbeck * will want to filter any characters >= 0x80.
239f0d9efc0Sbeck */
240f0d9efc0Sbeck convert_to_unicode((u_char *)jvol_desc->system_id, sizeof(jvol_desc->system_id), NULL);
241f0d9efc0Sbeck convert_to_unicode((u_char *)jvol_desc->volume_id, sizeof(jvol_desc->volume_id), NULL);
242f0d9efc0Sbeck convert_to_unicode((u_char *)jvol_desc->volume_set_id, sizeof(jvol_desc->volume_set_id), NULL);
243f0d9efc0Sbeck convert_to_unicode((u_char *)jvol_desc->publisher_id, sizeof(jvol_desc->publisher_id), NULL);
244f0d9efc0Sbeck convert_to_unicode((u_char *)jvol_desc->preparer_id, sizeof(jvol_desc->preparer_id), NULL);
245f0d9efc0Sbeck convert_to_unicode((u_char *)jvol_desc->application_id, sizeof(jvol_desc->application_id), NULL);
246f0d9efc0Sbeck convert_to_unicode((u_char *)jvol_desc->copyright_file_id, sizeof(jvol_desc->copyright_file_id), NULL);
247f0d9efc0Sbeck convert_to_unicode((u_char *)jvol_desc->abstract_file_id, sizeof(jvol_desc->abstract_file_id), NULL);
248f0d9efc0Sbeck convert_to_unicode((u_char *)jvol_desc->bibliographic_file_id, sizeof(jvol_desc->bibliographic_file_id), NULL);
249f0d9efc0Sbeck
250f0d9efc0Sbeck
251f0d9efc0Sbeck }
252f0d9efc0Sbeck
FDECL1(assign_joliet_directory_addresses,struct directory *,node)253f0d9efc0Sbeck static void FDECL1(assign_joliet_directory_addresses, struct directory *, node)
254f0d9efc0Sbeck {
255f0d9efc0Sbeck int dir_size;
256f0d9efc0Sbeck struct directory * dpnt;
257f0d9efc0Sbeck
258f0d9efc0Sbeck dpnt = node;
259f0d9efc0Sbeck
260f0d9efc0Sbeck while (dpnt)
261f0d9efc0Sbeck {
262f0d9efc0Sbeck if( (dpnt->dir_flags & INHIBIT_JOLIET_ENTRY) == 0 )
263f0d9efc0Sbeck {
264f0d9efc0Sbeck /*
265f0d9efc0Sbeck * If we already have an extent for this (i.e. it came from
266f0d9efc0Sbeck * a multisession disc), then don't reassign a new extent.
267f0d9efc0Sbeck */
268f0d9efc0Sbeck dpnt->jpath_index = next_jpath_index++;
269f0d9efc0Sbeck if( dpnt->jextent == 0 )
270f0d9efc0Sbeck {
271f0d9efc0Sbeck dpnt->jextent = last_extent;
272f0d9efc0Sbeck dir_size = (dpnt->jsize + (SECTOR_SIZE - 1)) >> 11;
273f0d9efc0Sbeck last_extent += dir_size;
274f0d9efc0Sbeck }
275f0d9efc0Sbeck }
276f0d9efc0Sbeck
277f0d9efc0Sbeck /* skip if hidden - but not for the rr_moved dir */
278f0d9efc0Sbeck if(dpnt->subdir && (!(dpnt->dir_flags & INHIBIT_JOLIET_ENTRY) || dpnt == reloc_dir))
279f0d9efc0Sbeck {
280f0d9efc0Sbeck assign_joliet_directory_addresses(dpnt->subdir);
281f0d9efc0Sbeck }
282f0d9efc0Sbeck dpnt = dpnt->next;
283f0d9efc0Sbeck }
284f0d9efc0Sbeck }
285f0d9efc0Sbeck
286f0d9efc0Sbeck static
FDECL1(build_jpathlist,struct directory *,node)287f0d9efc0Sbeck void FDECL1(build_jpathlist, struct directory *, node)
288f0d9efc0Sbeck {
289f0d9efc0Sbeck struct directory * dpnt;
290f0d9efc0Sbeck
291f0d9efc0Sbeck dpnt = node;
292f0d9efc0Sbeck
293f0d9efc0Sbeck while (dpnt)
294f0d9efc0Sbeck
295f0d9efc0Sbeck {
296f0d9efc0Sbeck if( (dpnt->dir_flags & INHIBIT_JOLIET_ENTRY) == 0 )
297f0d9efc0Sbeck {
298f0d9efc0Sbeck jpathlist[dpnt->jpath_index] = dpnt;
299f0d9efc0Sbeck }
300f0d9efc0Sbeck if(dpnt->subdir) build_jpathlist(dpnt->subdir);
301f0d9efc0Sbeck dpnt = dpnt->next;
302f0d9efc0Sbeck }
303f0d9efc0Sbeck } /* build_jpathlist(... */
304f0d9efc0Sbeck
FDECL2(joliet_compare_paths,void const *,r,void const *,l)305f0d9efc0Sbeck static int FDECL2(joliet_compare_paths, void const *, r, void const *, l)
306f0d9efc0Sbeck {
307f0d9efc0Sbeck struct directory const *ll = *(struct directory * const *)l;
308f0d9efc0Sbeck struct directory const *rr = *(struct directory * const *)r;
309f0d9efc0Sbeck int rparent, lparent;
310f0d9efc0Sbeck
311f0d9efc0Sbeck rparent = rr->parent->jpath_index;
312f0d9efc0Sbeck lparent = ll->parent->jpath_index;
313f0d9efc0Sbeck if( rr->parent == reloc_dir )
314f0d9efc0Sbeck {
315f0d9efc0Sbeck rparent = rr->self->parent_rec->filedir->jpath_index;
316f0d9efc0Sbeck }
317f0d9efc0Sbeck
318f0d9efc0Sbeck if( ll->parent == reloc_dir )
319f0d9efc0Sbeck {
320f0d9efc0Sbeck lparent = ll->self->parent_rec->filedir->jpath_index;
321f0d9efc0Sbeck }
322f0d9efc0Sbeck
323f0d9efc0Sbeck if (rparent < lparent)
324f0d9efc0Sbeck {
325f0d9efc0Sbeck return -1;
326f0d9efc0Sbeck }
327f0d9efc0Sbeck
328f0d9efc0Sbeck if (rparent > lparent)
329f0d9efc0Sbeck {
330f0d9efc0Sbeck return 1;
331f0d9efc0Sbeck }
332f0d9efc0Sbeck #ifdef APPLE_HYB
333f0d9efc0Sbeck /* use Mac name for Joliet name */
334f0d9efc0Sbeck if (USE_MAC_NAME(mac_name, rr->self) && USE_MAC_NAME(mac_name, ll->self))
335f0d9efc0Sbeck return strcmp(rr->self->hfs_ent->name, ll->self->hfs_ent->name);
336f0d9efc0Sbeck else
337f0d9efc0Sbeck #endif /* APPLE_HYB */
338f0d9efc0Sbeck return strcmp(rr->self->name, ll->self->name);
339f0d9efc0Sbeck
340f0d9efc0Sbeck } /* compare_paths(... */
341f0d9efc0Sbeck
generate_joliet_path_tables()342f0d9efc0Sbeck static int generate_joliet_path_tables()
343f0d9efc0Sbeck {
344f0d9efc0Sbeck struct directory_entry * de;
345f0d9efc0Sbeck struct directory * dpnt;
346f0d9efc0Sbeck int fix;
347f0d9efc0Sbeck int j;
348f0d9efc0Sbeck int namelen;
349f0d9efc0Sbeck char * npnt;
350f0d9efc0Sbeck char * npnt1;
351f0d9efc0Sbeck int tablesize;
352f0d9efc0Sbeck
353f0d9efc0Sbeck /*
354f0d9efc0Sbeck * First allocate memory for the tables and initialize the memory
355f0d9efc0Sbeck */
356f0d9efc0Sbeck tablesize = jpath_blocks << 11;
357f0d9efc0Sbeck jpath_table_m = (char *) e_malloc(tablesize);
358f0d9efc0Sbeck jpath_table_l = (char *) e_malloc(tablesize);
359f0d9efc0Sbeck memset(jpath_table_l, 0, tablesize);
360f0d9efc0Sbeck memset(jpath_table_m, 0, tablesize);
361f0d9efc0Sbeck
362f0d9efc0Sbeck if( next_jpath_index > 0xffff )
363f0d9efc0Sbeck {
364f0d9efc0Sbeck fprintf(stderr, "Unable to generate sane path tables - too many directories (%d)\n",
365f0d9efc0Sbeck next_jpath_index);
366f0d9efc0Sbeck exit(1);
367f0d9efc0Sbeck }
368f0d9efc0Sbeck /*
369f0d9efc0Sbeck * Now start filling in the path tables. Start with root directory
370f0d9efc0Sbeck */
371f0d9efc0Sbeck jpath_table_index = 0;
372f0d9efc0Sbeck jpathlist = (struct directory **) e_malloc(sizeof(struct directory *)
373f0d9efc0Sbeck * next_jpath_index);
374f0d9efc0Sbeck memset(jpathlist, 0, sizeof(struct directory *) * next_jpath_index);
375f0d9efc0Sbeck build_jpathlist(root);
376f0d9efc0Sbeck
377f0d9efc0Sbeck do
378f0d9efc0Sbeck {
379f0d9efc0Sbeck fix = 0;
380f0d9efc0Sbeck #ifdef __STDC__
381f0d9efc0Sbeck qsort(&jpathlist[1], next_jpath_index-1, sizeof(struct directory *),
382f0d9efc0Sbeck (int (*)(const void *, const void *))joliet_compare_paths);
383f0d9efc0Sbeck #else
384f0d9efc0Sbeck qsort(&jpathlist[1], next_jpath_index-1, sizeof(struct directory *),
385f0d9efc0Sbeck joliet_compare_paths);
386f0d9efc0Sbeck #endif
387f0d9efc0Sbeck
388f0d9efc0Sbeck for(j=1; j<next_jpath_index; j++)
389f0d9efc0Sbeck {
390f0d9efc0Sbeck if(jpathlist[j]->jpath_index != j)
391f0d9efc0Sbeck {
392f0d9efc0Sbeck jpathlist[j]->jpath_index = j;
393f0d9efc0Sbeck fix++;
394f0d9efc0Sbeck }
395f0d9efc0Sbeck }
396f0d9efc0Sbeck } while(fix);
397f0d9efc0Sbeck
398f0d9efc0Sbeck for(j=1; j<next_jpath_index; j++)
399f0d9efc0Sbeck {
400f0d9efc0Sbeck dpnt = jpathlist[j];
401f0d9efc0Sbeck if(!dpnt)
402f0d9efc0Sbeck {
403f0d9efc0Sbeck fprintf(stderr,"Entry %d not in path tables\n", j);
404f0d9efc0Sbeck exit(1);
405f0d9efc0Sbeck }
406f0d9efc0Sbeck npnt = dpnt->de_name;
407f0d9efc0Sbeck
408f0d9efc0Sbeck npnt1 = strrchr(npnt, PATH_SEPARATOR);
409f0d9efc0Sbeck if(npnt1)
410f0d9efc0Sbeck {
411f0d9efc0Sbeck npnt = npnt1 + 1;
412f0d9efc0Sbeck }
413f0d9efc0Sbeck
414f0d9efc0Sbeck de = dpnt->self;
415f0d9efc0Sbeck if(!de)
416f0d9efc0Sbeck {
417f0d9efc0Sbeck fprintf(stderr,"Fatal goof - directory has amnesia\n");
418f0d9efc0Sbeck exit(1);
419f0d9efc0Sbeck }
420f0d9efc0Sbeck
421f0d9efc0Sbeck #ifdef APPLE_HYB
422f0d9efc0Sbeck if (USE_MAC_NAME(mac_name, de))
423f0d9efc0Sbeck namelen = joliet_strlen(de->hfs_ent->name);
424f0d9efc0Sbeck else
425f0d9efc0Sbeck #endif /* APPLE_HYB */
426f0d9efc0Sbeck namelen = joliet_strlen(de->name);
427f0d9efc0Sbeck
428f0d9efc0Sbeck if( dpnt == root )
429f0d9efc0Sbeck {
430f0d9efc0Sbeck jpath_table_l[jpath_table_index] = 1;
431f0d9efc0Sbeck jpath_table_m[jpath_table_index] = 1;
432f0d9efc0Sbeck }
433f0d9efc0Sbeck else
434f0d9efc0Sbeck {
435f0d9efc0Sbeck jpath_table_l[jpath_table_index] = namelen;
436f0d9efc0Sbeck jpath_table_m[jpath_table_index] = namelen;
437f0d9efc0Sbeck }
438f0d9efc0Sbeck jpath_table_index += 2;
439f0d9efc0Sbeck
440f0d9efc0Sbeck set_731(jpath_table_l + jpath_table_index, dpnt->jextent);
441f0d9efc0Sbeck set_732(jpath_table_m + jpath_table_index, dpnt->jextent);
442f0d9efc0Sbeck jpath_table_index += 4;
443f0d9efc0Sbeck
444f0d9efc0Sbeck if( dpnt->parent != reloc_dir )
445f0d9efc0Sbeck {
446f0d9efc0Sbeck set_721(jpath_table_l + jpath_table_index,
447f0d9efc0Sbeck dpnt->parent->jpath_index);
448f0d9efc0Sbeck set_722(jpath_table_m + jpath_table_index,
449f0d9efc0Sbeck dpnt->parent->jpath_index);
450f0d9efc0Sbeck }
451f0d9efc0Sbeck else
452f0d9efc0Sbeck {
453f0d9efc0Sbeck set_721(jpath_table_l + jpath_table_index,
454f0d9efc0Sbeck dpnt->self->parent_rec->filedir->jpath_index);
455f0d9efc0Sbeck set_722(jpath_table_m + jpath_table_index,
456f0d9efc0Sbeck dpnt->self->parent_rec->filedir->jpath_index);
457f0d9efc0Sbeck }
458f0d9efc0Sbeck
459f0d9efc0Sbeck jpath_table_index += 2;
460f0d9efc0Sbeck
461f0d9efc0Sbeck /*
462f0d9efc0Sbeck * The root directory is still represented in non-unicode fashion.
463f0d9efc0Sbeck */
464f0d9efc0Sbeck if( dpnt == root )
465f0d9efc0Sbeck {
466f0d9efc0Sbeck jpath_table_l[jpath_table_index] = 0;
467f0d9efc0Sbeck jpath_table_m[jpath_table_index] = 0;
468f0d9efc0Sbeck jpath_table_index ++;
469f0d9efc0Sbeck }
470f0d9efc0Sbeck else
471f0d9efc0Sbeck {
472f0d9efc0Sbeck #ifdef APPLE_HYB
473f0d9efc0Sbeck if (USE_MAC_NAME(mac_name , de)) {
474f0d9efc0Sbeck convert_to_unicode((u_char *)jpath_table_l + jpath_table_index,
475f0d9efc0Sbeck namelen, de->hfs_ent->name);
476f0d9efc0Sbeck convert_to_unicode((u_char *)jpath_table_m + jpath_table_index,
477f0d9efc0Sbeck namelen, de->hfs_ent->name);
478f0d9efc0Sbeck }
479f0d9efc0Sbeck else {
480f0d9efc0Sbeck #endif /* APPLE_HYB */
481f0d9efc0Sbeck convert_to_unicode((u_char *)jpath_table_l + jpath_table_index,
482f0d9efc0Sbeck namelen, de->name);
483f0d9efc0Sbeck convert_to_unicode((u_char *)jpath_table_m + jpath_table_index,
484f0d9efc0Sbeck namelen, de->name);
485f0d9efc0Sbeck #ifdef APPLE_HYB
486f0d9efc0Sbeck }
487f0d9efc0Sbeck #endif /* APPLE_HYB */
488f0d9efc0Sbeck
489f0d9efc0Sbeck jpath_table_index += namelen;
490f0d9efc0Sbeck }
491f0d9efc0Sbeck
492f0d9efc0Sbeck if(jpath_table_index & 1)
493f0d9efc0Sbeck {
494f0d9efc0Sbeck jpath_table_index++; /* For odd lengths we pad */
495f0d9efc0Sbeck }
496f0d9efc0Sbeck }
497f0d9efc0Sbeck
498f0d9efc0Sbeck free(jpathlist);
499f0d9efc0Sbeck if(jpath_table_index != jpath_table_size)
500f0d9efc0Sbeck {
501f0d9efc0Sbeck fprintf(stderr,"Joliet path table lengths do not match %d %d\n",
502f0d9efc0Sbeck jpath_table_index,
503f0d9efc0Sbeck jpath_table_size);
504f0d9efc0Sbeck }
505f0d9efc0Sbeck return 0;
506f0d9efc0Sbeck } /* generate_path_tables(... */
507f0d9efc0Sbeck
FDECL2(generate_one_joliet_directory,struct directory *,dpnt,FILE *,outfile)508f0d9efc0Sbeck static void FDECL2(generate_one_joliet_directory, struct directory *, dpnt, FILE *, outfile)
509f0d9efc0Sbeck {
510f0d9efc0Sbeck unsigned int dir_index;
511f0d9efc0Sbeck char * directory_buffer;
512f0d9efc0Sbeck int new_reclen;
513f0d9efc0Sbeck struct directory_entry * s_entry;
514f0d9efc0Sbeck struct directory_entry * s_entry1;
515f0d9efc0Sbeck struct iso_directory_record jrec;
516f0d9efc0Sbeck unsigned int total_size;
517f0d9efc0Sbeck int cvt_len;
518f0d9efc0Sbeck struct directory * finddir;
519f0d9efc0Sbeck
520f0d9efc0Sbeck total_size = (dpnt->jsize + (SECTOR_SIZE - 1)) & ~(SECTOR_SIZE - 1);
521f0d9efc0Sbeck directory_buffer = (char *) e_malloc(total_size);
522f0d9efc0Sbeck memset(directory_buffer, 0, total_size);
523f0d9efc0Sbeck dir_index = 0;
524f0d9efc0Sbeck
525f0d9efc0Sbeck s_entry = dpnt->jcontents;
526f0d9efc0Sbeck while(s_entry)
527f0d9efc0Sbeck {
528f0d9efc0Sbeck if(s_entry->de_flags & INHIBIT_JOLIET_ENTRY) {
529f0d9efc0Sbeck s_entry = s_entry->jnext;
530f0d9efc0Sbeck continue;
531f0d9efc0Sbeck }
532f0d9efc0Sbeck
533f0d9efc0Sbeck /*
534f0d9efc0Sbeck * If this entry was a directory that was relocated, we have a bit
535f0d9efc0Sbeck * of trouble here. We need to dig out the real thing and put it
536f0d9efc0Sbeck * back here. In the Joliet tree, there is no relocated rock
537f0d9efc0Sbeck * ridge, as there are no depth limits to a directory tree.
538f0d9efc0Sbeck */
539f0d9efc0Sbeck if( (s_entry->de_flags & RELOCATED_DIRECTORY) != 0 )
540f0d9efc0Sbeck {
541f0d9efc0Sbeck for(s_entry1 = reloc_dir->contents; s_entry1; s_entry1 = s_entry1->next)
542f0d9efc0Sbeck {
543f0d9efc0Sbeck if( s_entry1->parent_rec == s_entry )
544f0d9efc0Sbeck {
545f0d9efc0Sbeck break;
546f0d9efc0Sbeck }
547f0d9efc0Sbeck }
548f0d9efc0Sbeck if( s_entry1 == NULL )
549f0d9efc0Sbeck {
550f0d9efc0Sbeck /*
551f0d9efc0Sbeck * We got trouble.
552f0d9efc0Sbeck */
553f0d9efc0Sbeck fprintf(stderr, "Unable to locate relocated directory\n");
554f0d9efc0Sbeck exit(1);
555f0d9efc0Sbeck }
556f0d9efc0Sbeck }
557f0d9efc0Sbeck else
558f0d9efc0Sbeck {
559f0d9efc0Sbeck s_entry1 = s_entry;
560f0d9efc0Sbeck }
561f0d9efc0Sbeck
562f0d9efc0Sbeck /*
563f0d9efc0Sbeck * We do not allow directory entries to cross sector boundaries.
564f0d9efc0Sbeck * Simply pad, and then start the next entry at the next sector
565f0d9efc0Sbeck */
566f0d9efc0Sbeck new_reclen = s_entry1->jreclen;
567f0d9efc0Sbeck if( (dir_index & (SECTOR_SIZE - 1)) + new_reclen >= SECTOR_SIZE )
568f0d9efc0Sbeck {
569f0d9efc0Sbeck dir_index = (dir_index + (SECTOR_SIZE - 1)) &
570f0d9efc0Sbeck ~(SECTOR_SIZE - 1);
571f0d9efc0Sbeck }
572f0d9efc0Sbeck
573f0d9efc0Sbeck memcpy(&jrec, &s_entry1->isorec, sizeof(struct iso_directory_record) -
574f0d9efc0Sbeck sizeof(s_entry1->isorec.name));
575f0d9efc0Sbeck
576f0d9efc0Sbeck #ifdef APPLE_HYB
577f0d9efc0Sbeck /* Use the HFS name if it exists */
578f0d9efc0Sbeck if (USE_MAC_NAME(mac_name, s_entry1))
579f0d9efc0Sbeck cvt_len = joliet_strlen(s_entry1->hfs_ent->name);
580f0d9efc0Sbeck else
581f0d9efc0Sbeck #endif /* APPLE_HYB */
582f0d9efc0Sbeck cvt_len = joliet_strlen(s_entry1->name);
583f0d9efc0Sbeck
584f0d9efc0Sbeck /*
585f0d9efc0Sbeck * Fix the record length - this was the non-Joliet version we
586f0d9efc0Sbeck * were seeing.
587f0d9efc0Sbeck */
588f0d9efc0Sbeck jrec.name_len[0] = cvt_len;
589f0d9efc0Sbeck jrec.length[0] = s_entry1->jreclen;
590f0d9efc0Sbeck
591f0d9efc0Sbeck /*
592f0d9efc0Sbeck * If this is a directory, fix the correct size and extent
593f0d9efc0Sbeck * number.
594f0d9efc0Sbeck */
595f0d9efc0Sbeck if( (jrec.flags[0] & 2) != 0 )
596f0d9efc0Sbeck {
597f0d9efc0Sbeck if(strcmp(s_entry1->name,".") == 0)
598f0d9efc0Sbeck {
599f0d9efc0Sbeck jrec.name_len[0] = 1;
600f0d9efc0Sbeck set_733((char *) jrec.extent, dpnt->jextent);
601f0d9efc0Sbeck set_733((char *) jrec.size, ROUND_UP(dpnt->jsize));
602f0d9efc0Sbeck }
603f0d9efc0Sbeck else if(strcmp(s_entry1->name,"..") == 0)
604f0d9efc0Sbeck {
605f0d9efc0Sbeck jrec.name_len[0] = 1;
606f0d9efc0Sbeck if( dpnt->parent == reloc_dir )
607f0d9efc0Sbeck {
608f0d9efc0Sbeck set_733((char *) jrec.extent, dpnt->self->parent_rec->filedir->jextent);
609f0d9efc0Sbeck set_733((char *) jrec.size, ROUND_UP(dpnt->self->parent_rec->filedir->jsize));
610f0d9efc0Sbeck }
611f0d9efc0Sbeck else
612f0d9efc0Sbeck
613f0d9efc0Sbeck {
614f0d9efc0Sbeck set_733((char *) jrec.extent, dpnt->parent->jextent);
615f0d9efc0Sbeck set_733((char *) jrec.size, ROUND_UP(dpnt->parent->jsize));
616f0d9efc0Sbeck }
617f0d9efc0Sbeck }
618f0d9efc0Sbeck else
619f0d9efc0Sbeck {
620f0d9efc0Sbeck if( (s_entry->de_flags & RELOCATED_DIRECTORY) != 0 )
621f0d9efc0Sbeck {
622f0d9efc0Sbeck finddir = reloc_dir->subdir;
623f0d9efc0Sbeck }
624f0d9efc0Sbeck else
625f0d9efc0Sbeck {
626f0d9efc0Sbeck finddir = dpnt->subdir;
627f0d9efc0Sbeck }
628f0d9efc0Sbeck while(1==1)
629f0d9efc0Sbeck {
630f0d9efc0Sbeck if(finddir->self == s_entry1) break;
631f0d9efc0Sbeck finddir = finddir->next;
632f0d9efc0Sbeck if(!finddir)
633f0d9efc0Sbeck {
634f0d9efc0Sbeck fprintf(stderr,"Fatal goof - unable to find directory location\n"); exit(1);
635f0d9efc0Sbeck }
636f0d9efc0Sbeck }
637f0d9efc0Sbeck set_733((char *) jrec.extent, finddir->jextent);
638f0d9efc0Sbeck set_733((char *) jrec.size, ROUND_UP(finddir->jsize));
639f0d9efc0Sbeck }
640f0d9efc0Sbeck }
641f0d9efc0Sbeck
642f0d9efc0Sbeck memcpy(directory_buffer + dir_index, &jrec,
643f0d9efc0Sbeck sizeof(struct iso_directory_record) -
644f0d9efc0Sbeck sizeof(s_entry1->isorec.name));
645f0d9efc0Sbeck
646f0d9efc0Sbeck
647f0d9efc0Sbeck dir_index += sizeof(struct iso_directory_record) -
648f0d9efc0Sbeck sizeof (s_entry1->isorec.name);
649f0d9efc0Sbeck
650f0d9efc0Sbeck /*
651f0d9efc0Sbeck * Finally dump the Unicode version of the filename.
652f0d9efc0Sbeck * Note - . and .. are the same as with non-Joliet discs.
653f0d9efc0Sbeck */
654f0d9efc0Sbeck if( (jrec.flags[0] & 2) != 0
655f0d9efc0Sbeck && strcmp(s_entry1->name, ".") == 0 )
656f0d9efc0Sbeck {
657f0d9efc0Sbeck directory_buffer[dir_index++] = 0;
658f0d9efc0Sbeck }
659f0d9efc0Sbeck else if( (jrec.flags[0] & 2) != 0
660f0d9efc0Sbeck && strcmp(s_entry1->name, "..") == 0 )
661f0d9efc0Sbeck {
662f0d9efc0Sbeck directory_buffer[dir_index++] = 1;
663f0d9efc0Sbeck }
664f0d9efc0Sbeck else
665f0d9efc0Sbeck {
666f0d9efc0Sbeck #ifdef APPLE_HYB
667f0d9efc0Sbeck if (USE_MAC_NAME(mac_name, s_entry1))
668f0d9efc0Sbeck /* Use the HFS name if it exists */
669f0d9efc0Sbeck convert_to_unicode((u_char *)directory_buffer + dir_index,
670f0d9efc0Sbeck cvt_len,
671f0d9efc0Sbeck s_entry1->hfs_ent->name);
672f0d9efc0Sbeck else
673f0d9efc0Sbeck #endif /* APPLE_HYB */
674f0d9efc0Sbeck convert_to_unicode((u_char *)directory_buffer + dir_index,
675f0d9efc0Sbeck cvt_len,
676f0d9efc0Sbeck s_entry1->name);
677f0d9efc0Sbeck dir_index += cvt_len;
678f0d9efc0Sbeck }
679f0d9efc0Sbeck
680f0d9efc0Sbeck if(dir_index & 1)
681f0d9efc0Sbeck {
682f0d9efc0Sbeck directory_buffer[dir_index++] = 0;
683f0d9efc0Sbeck }
684f0d9efc0Sbeck
685f0d9efc0Sbeck s_entry = s_entry->jnext;
686f0d9efc0Sbeck }
687f0d9efc0Sbeck
688f0d9efc0Sbeck if(dpnt->jsize != dir_index)
689f0d9efc0Sbeck {
690f0d9efc0Sbeck fprintf(stderr,"Unexpected joliet directory length %d %d %s\n",dpnt->jsize,
691f0d9efc0Sbeck dir_index, dpnt->de_name);
692f0d9efc0Sbeck }
693f0d9efc0Sbeck
694f0d9efc0Sbeck xfwrite(directory_buffer, 1, total_size, outfile);
695f0d9efc0Sbeck last_extent_written += total_size >> 11;
696f0d9efc0Sbeck free(directory_buffer);
697f0d9efc0Sbeck } /* generate_one_joliet_directory(... */
698f0d9efc0Sbeck
FDECL1(joliet_sort_n_finish,struct directory *,this_dir)699f0d9efc0Sbeck static int FDECL1(joliet_sort_n_finish, struct directory *, this_dir)
700f0d9efc0Sbeck {
701f0d9efc0Sbeck struct directory_entry * s_entry;
702f0d9efc0Sbeck int status = 0;
703f0d9efc0Sbeck
704f0d9efc0Sbeck /* don't want to skip this directory if it's the reloc_dir at the moment */
705f0d9efc0Sbeck if(this_dir != reloc_dir && this_dir->dir_flags & INHIBIT_JOLIET_ENTRY)
706f0d9efc0Sbeck {
707f0d9efc0Sbeck return 0;
708f0d9efc0Sbeck }
709f0d9efc0Sbeck
710f0d9efc0Sbeck for(s_entry = this_dir->contents; s_entry; s_entry = s_entry->next)
711f0d9efc0Sbeck {
712f0d9efc0Sbeck /* skip hidden entries */
713f0d9efc0Sbeck if( (s_entry->de_flags & INHIBIT_JOLIET_ENTRY) != 0 )
714f0d9efc0Sbeck {
715f0d9efc0Sbeck continue;
716f0d9efc0Sbeck }
717f0d9efc0Sbeck
718f0d9efc0Sbeck /*
719f0d9efc0Sbeck * First update the path table sizes for directories.
720f0d9efc0Sbeck *
721f0d9efc0Sbeck * Finally, set the length of the directory entry if Joliet is used.
722f0d9efc0Sbeck * The name is longer, but no Rock Ridge is ever used here, so
723f0d9efc0Sbeck * depending upon the options the entry size might turn out to be about
724f0d9efc0Sbeck * the same. The Unicode name is always a multiple of 2 bytes, so
725f0d9efc0Sbeck * we always add 1 to make it an even number.
726f0d9efc0Sbeck */
727f0d9efc0Sbeck if(s_entry->isorec.flags[0] == 2)
728f0d9efc0Sbeck {
729f0d9efc0Sbeck if (strcmp(s_entry->name,".") && strcmp(s_entry->name,".."))
730f0d9efc0Sbeck {
731f0d9efc0Sbeck #ifdef APPLE_HYB
732f0d9efc0Sbeck if (USE_MAC_NAME(mac_name, s_entry))
733f0d9efc0Sbeck /* Use the HFS name if it exists */
734f0d9efc0Sbeck jpath_table_size += joliet_strlen(s_entry->hfs_ent->name) + sizeof(struct iso_path_table) - 1;
735f0d9efc0Sbeck else
736f0d9efc0Sbeck #endif /* APPLE_HYB */
737f0d9efc0Sbeck jpath_table_size += joliet_strlen(s_entry->name) + sizeof(struct iso_path_table) - 1;
738f0d9efc0Sbeck if (jpath_table_size & 1)
739f0d9efc0Sbeck {
740f0d9efc0Sbeck jpath_table_size++;
741f0d9efc0Sbeck }
742f0d9efc0Sbeck }
743f0d9efc0Sbeck else
744f0d9efc0Sbeck {
745f0d9efc0Sbeck if (this_dir == root && strlen(s_entry->name) == 1)
746f0d9efc0Sbeck {
747f0d9efc0Sbeck jpath_table_size += sizeof(struct iso_path_table);
748f0d9efc0Sbeck if (jpath_table_size & 1) jpath_table_size++;
749f0d9efc0Sbeck }
750f0d9efc0Sbeck }
751f0d9efc0Sbeck }
752f0d9efc0Sbeck
753f0d9efc0Sbeck if (strcmp(s_entry->name,".") && strcmp(s_entry->name,".."))
754f0d9efc0Sbeck {
755f0d9efc0Sbeck #ifdef APPLE_HYB
756f0d9efc0Sbeck if (USE_MAC_NAME(mac_name, s_entry))
757f0d9efc0Sbeck /* Use the HFS name if it exists */
758f0d9efc0Sbeck s_entry->jreclen = sizeof(struct iso_directory_record)
759f0d9efc0Sbeck - sizeof(s_entry->isorec.name)
760f0d9efc0Sbeck + joliet_strlen(s_entry->hfs_ent->name)
761f0d9efc0Sbeck + 1;
762f0d9efc0Sbeck else
763f0d9efc0Sbeck #endif /* APPLE_HYB */
764f0d9efc0Sbeck s_entry->jreclen = sizeof(struct iso_directory_record)
765f0d9efc0Sbeck - sizeof(s_entry->isorec.name)
766f0d9efc0Sbeck + joliet_strlen(s_entry->name)
767f0d9efc0Sbeck + 1;
768f0d9efc0Sbeck }
769f0d9efc0Sbeck else
770f0d9efc0Sbeck {
771f0d9efc0Sbeck /*
772f0d9efc0Sbeck * Special - for '.' and '..' we generate the same records we
773f0d9efc0Sbeck * did for non-Joliet discs.
774f0d9efc0Sbeck */
775f0d9efc0Sbeck s_entry->jreclen = sizeof(struct iso_directory_record)
776f0d9efc0Sbeck - sizeof(s_entry->isorec.name)
777f0d9efc0Sbeck + 1;
778f0d9efc0Sbeck }
779f0d9efc0Sbeck
780f0d9efc0Sbeck
781f0d9efc0Sbeck }
782f0d9efc0Sbeck
783f0d9efc0Sbeck if( (this_dir->dir_flags & INHIBIT_JOLIET_ENTRY) != 0 )
784f0d9efc0Sbeck {
785f0d9efc0Sbeck return 0;
786f0d9efc0Sbeck }
787f0d9efc0Sbeck
788f0d9efc0Sbeck this_dir->jcontents = this_dir->contents;
789f0d9efc0Sbeck status = joliet_sort_directory(&this_dir->jcontents);
790f0d9efc0Sbeck
791f0d9efc0Sbeck /*
792f0d9efc0Sbeck * Now go through the directory and figure out how large this one will be.
793f0d9efc0Sbeck * Do not split a directory entry across a sector boundary
794f0d9efc0Sbeck */
795f0d9efc0Sbeck s_entry = this_dir->jcontents;
796f0d9efc0Sbeck /*
797f0d9efc0Sbeck * XXX Is it ok to comment this out?
798f0d9efc0Sbeck */
799f0d9efc0Sbeck /*XXX JS this_dir->ce_bytes = 0;*/
800f0d9efc0Sbeck for(s_entry = this_dir->jcontents; s_entry; s_entry = s_entry->jnext)
801f0d9efc0Sbeck {
802f0d9efc0Sbeck int jreclen;
803f0d9efc0Sbeck
804f0d9efc0Sbeck if( (s_entry->de_flags & INHIBIT_JOLIET_ENTRY) != 0 )
805f0d9efc0Sbeck {
806f0d9efc0Sbeck continue;
807f0d9efc0Sbeck }
808f0d9efc0Sbeck
809f0d9efc0Sbeck jreclen = s_entry->jreclen;
810f0d9efc0Sbeck
811f0d9efc0Sbeck if ((this_dir->jsize & (SECTOR_SIZE - 1)) + jreclen >= SECTOR_SIZE)
812f0d9efc0Sbeck {
813f0d9efc0Sbeck this_dir->jsize = (this_dir->jsize + (SECTOR_SIZE - 1)) &
814f0d9efc0Sbeck ~(SECTOR_SIZE - 1);
815f0d9efc0Sbeck }
816f0d9efc0Sbeck this_dir->jsize += jreclen;
817f0d9efc0Sbeck }
818f0d9efc0Sbeck return status;
819f0d9efc0Sbeck }
820f0d9efc0Sbeck
821f0d9efc0Sbeck /*
822f0d9efc0Sbeck * Similar to the iso9660 case, except here we perform a full sort based upon the
823f0d9efc0Sbeck * regular name of the file, not the 8.3 version.
824f0d9efc0Sbeck */
FDECL2(joliet_compare_dirs,const void *,rr,const void *,ll)825f0d9efc0Sbeck static int FDECL2(joliet_compare_dirs, const void *, rr, const void *, ll)
826f0d9efc0Sbeck {
827f0d9efc0Sbeck char * rpnt, *lpnt;
828f0d9efc0Sbeck struct directory_entry ** r, **l;
829f0d9efc0Sbeck
830f0d9efc0Sbeck r = (struct directory_entry **) rr;
831f0d9efc0Sbeck l = (struct directory_entry **) ll;
832f0d9efc0Sbeck rpnt = (*r)->name;
833f0d9efc0Sbeck lpnt = (*l)->name;
834f0d9efc0Sbeck
835f0d9efc0Sbeck /*
836f0d9efc0Sbeck * If the entries are the same, this is an error.
837f0d9efc0Sbeck */
838f0d9efc0Sbeck if( strcmp(rpnt, lpnt) == 0 )
839f0d9efc0Sbeck {
840f0d9efc0Sbeck sort_goof++;
841f0d9efc0Sbeck }
842f0d9efc0Sbeck
843f0d9efc0Sbeck /*
844f0d9efc0Sbeck * Put the '.' and '..' entries on the head of the sorted list.
845f0d9efc0Sbeck * For normal ASCII, this always happens to be the case, but out of
846f0d9efc0Sbeck * band characters cause this not to be the case sometimes.
847f0d9efc0Sbeck */
848f0d9efc0Sbeck if( strcmp(rpnt, ".") == 0 ) return -1;
849f0d9efc0Sbeck if( strcmp(lpnt, ".") == 0 ) return 1;
850f0d9efc0Sbeck
851f0d9efc0Sbeck if( strcmp(rpnt, "..") == 0 ) return -1;
852f0d9efc0Sbeck if( strcmp(lpnt, "..") == 0 ) return 1;
853f0d9efc0Sbeck
854f0d9efc0Sbeck while(*rpnt && *lpnt)
855f0d9efc0Sbeck {
856f0d9efc0Sbeck if(*rpnt == ';' && *lpnt != ';') return -1;
857f0d9efc0Sbeck if(*rpnt != ';' && *lpnt == ';') return 1;
858f0d9efc0Sbeck
859f0d9efc0Sbeck if(*rpnt == ';' && *lpnt == ';') return 0;
860f0d9efc0Sbeck
861f0d9efc0Sbeck /*
862f0d9efc0Sbeck * Extensions are not special here. Don't treat the dot as something that
863f0d9efc0Sbeck * must be bumped to the start of the list.
864f0d9efc0Sbeck */
865f0d9efc0Sbeck #if 0
866f0d9efc0Sbeck if(*rpnt == '.' && *lpnt != '.') return -1;
867f0d9efc0Sbeck if(*rpnt != '.' && *lpnt == '.') return 1;
868f0d9efc0Sbeck #endif
869f0d9efc0Sbeck
870f0d9efc0Sbeck if(*rpnt < *lpnt) return -1;
871f0d9efc0Sbeck if(*rpnt > *lpnt) return 1;
872f0d9efc0Sbeck rpnt++; lpnt++;
873f0d9efc0Sbeck }
874f0d9efc0Sbeck if(*rpnt) return 1;
875f0d9efc0Sbeck if(*lpnt) return -1;
876f0d9efc0Sbeck return 0;
877f0d9efc0Sbeck }
878f0d9efc0Sbeck
879f0d9efc0Sbeck
880f0d9efc0Sbeck /*
881f0d9efc0Sbeck * Function: sort_directory
882f0d9efc0Sbeck *
883f0d9efc0Sbeck * Purpose: Sort the directory in the appropriate ISO9660
884f0d9efc0Sbeck * order.
885f0d9efc0Sbeck *
886f0d9efc0Sbeck * Notes: Returns 0 if OK, returns > 0 if an error occurred.
887f0d9efc0Sbeck */
FDECL1(joliet_sort_directory,struct directory_entry **,sort_dir)888f0d9efc0Sbeck static int FDECL1(joliet_sort_directory, struct directory_entry **, sort_dir)
889f0d9efc0Sbeck {
890f0d9efc0Sbeck int dcount = 0;
891f0d9efc0Sbeck int i;
892f0d9efc0Sbeck struct directory_entry * s_entry;
893f0d9efc0Sbeck struct directory_entry ** sortlist;
894f0d9efc0Sbeck
895f0d9efc0Sbeck s_entry = *sort_dir;
896f0d9efc0Sbeck while(s_entry)
897f0d9efc0Sbeck {
898f0d9efc0Sbeck /* skip hidden entries */
899f0d9efc0Sbeck if (!(s_entry->de_flags & INHIBIT_JOLIET_ENTRY))
900f0d9efc0Sbeck dcount++;
901f0d9efc0Sbeck s_entry = s_entry->next;
902f0d9efc0Sbeck }
903f0d9efc0Sbeck
904f0d9efc0Sbeck /*
905f0d9efc0Sbeck * OK, now we know how many there are. Build a vector for sorting.
906f0d9efc0Sbeck */
907f0d9efc0Sbeck sortlist = (struct directory_entry **)
908f0d9efc0Sbeck e_malloc(sizeof(struct directory_entry *) * dcount);
909f0d9efc0Sbeck
910f0d9efc0Sbeck dcount = 0;
911f0d9efc0Sbeck s_entry = *sort_dir;
912f0d9efc0Sbeck while(s_entry)
913f0d9efc0Sbeck {
914f0d9efc0Sbeck /* skip hidden entries */
915f0d9efc0Sbeck if (!(s_entry->de_flags & INHIBIT_JOLIET_ENTRY)) {
916f0d9efc0Sbeck sortlist[dcount] = s_entry;
917f0d9efc0Sbeck dcount++;
918f0d9efc0Sbeck }
919f0d9efc0Sbeck s_entry = s_entry->next;
920f0d9efc0Sbeck }
921f0d9efc0Sbeck
922f0d9efc0Sbeck sort_goof = 0;
923f0d9efc0Sbeck #ifdef __STDC__
924f0d9efc0Sbeck qsort(sortlist, dcount, sizeof(struct directory_entry *),
925f0d9efc0Sbeck (int (*)(const void *, const void *))joliet_compare_dirs);
926f0d9efc0Sbeck #else
927f0d9efc0Sbeck qsort(sortlist, dcount, sizeof(struct directory_entry *),
928f0d9efc0Sbeck joliet_compare_dirs);
929f0d9efc0Sbeck #endif
930f0d9efc0Sbeck
931f0d9efc0Sbeck /*
932f0d9efc0Sbeck * Now reassemble the linked list in the proper sorted order
933f0d9efc0Sbeck */
934f0d9efc0Sbeck for(i=0; i<dcount-1; i++)
935f0d9efc0Sbeck {
936f0d9efc0Sbeck sortlist[i]->jnext = sortlist[i+1];
937f0d9efc0Sbeck }
938f0d9efc0Sbeck
939f0d9efc0Sbeck sortlist[dcount-1]->jnext = NULL;
940f0d9efc0Sbeck *sort_dir = sortlist[0];
941f0d9efc0Sbeck
942f0d9efc0Sbeck free(sortlist);
943f0d9efc0Sbeck return sort_goof;
944f0d9efc0Sbeck }
945f0d9efc0Sbeck
FDECL1(joliet_sort_tree,struct directory *,node)946f0d9efc0Sbeck int FDECL1(joliet_sort_tree, struct directory *, node)
947f0d9efc0Sbeck {
948f0d9efc0Sbeck struct directory * dpnt;
949f0d9efc0Sbeck int ret = 0;
950f0d9efc0Sbeck
951f0d9efc0Sbeck dpnt = node;
952f0d9efc0Sbeck
953f0d9efc0Sbeck while (dpnt){
954f0d9efc0Sbeck ret = joliet_sort_n_finish(dpnt);
955f0d9efc0Sbeck if( ret )
956f0d9efc0Sbeck {
957f0d9efc0Sbeck break;
958f0d9efc0Sbeck }
959f0d9efc0Sbeck if(dpnt->subdir) ret = joliet_sort_tree(dpnt->subdir);
960f0d9efc0Sbeck if( ret )
961f0d9efc0Sbeck {
962f0d9efc0Sbeck break;
963f0d9efc0Sbeck }
964f0d9efc0Sbeck dpnt = dpnt->next;
965f0d9efc0Sbeck }
966f0d9efc0Sbeck return ret;
967f0d9efc0Sbeck }
968f0d9efc0Sbeck
FDECL2(generate_joliet_directories,struct directory *,node,FILE *,outfile)969f0d9efc0Sbeck static void FDECL2(generate_joliet_directories, struct directory *, node, FILE*, outfile){
970f0d9efc0Sbeck struct directory * dpnt;
971f0d9efc0Sbeck
972f0d9efc0Sbeck dpnt = node;
973f0d9efc0Sbeck
974f0d9efc0Sbeck while (dpnt)
975f0d9efc0Sbeck {
976f0d9efc0Sbeck if( (dpnt->dir_flags & INHIBIT_JOLIET_ENTRY) == 0 )
977f0d9efc0Sbeck {
978f0d9efc0Sbeck /*
979f0d9efc0Sbeck * In theory we should never reuse a directory, so this doesn't
980f0d9efc0Sbeck * make much sense.
981f0d9efc0Sbeck */
982f0d9efc0Sbeck if( dpnt->jextent > session_start )
983f0d9efc0Sbeck {
984f0d9efc0Sbeck generate_one_joliet_directory(dpnt, outfile);
985f0d9efc0Sbeck }
986f0d9efc0Sbeck }
987f0d9efc0Sbeck /* skip if hidden - but not for the rr_moved dir */
988f0d9efc0Sbeck if(dpnt->subdir && (!(dpnt->dir_flags & INHIBIT_JOLIET_ENTRY) || dpnt == reloc_dir))
989f0d9efc0Sbeck generate_joliet_directories(dpnt->subdir, outfile);
990f0d9efc0Sbeck dpnt = dpnt->next;
991f0d9efc0Sbeck }
992f0d9efc0Sbeck }
993f0d9efc0Sbeck
994f0d9efc0Sbeck
995f0d9efc0Sbeck /*
996f0d9efc0Sbeck * Function to write the EVD for the disc.
997f0d9efc0Sbeck */
FDECL1(jpathtab_write,FILE *,outfile)998f0d9efc0Sbeck static int FDECL1(jpathtab_write, FILE *, outfile)
999f0d9efc0Sbeck {
1000f0d9efc0Sbeck /*
1001f0d9efc0Sbeck * Next we write the path tables
1002f0d9efc0Sbeck */
1003f0d9efc0Sbeck xfwrite(jpath_table_l, 1, jpath_blocks << 11, outfile);
1004f0d9efc0Sbeck xfwrite(jpath_table_m, 1, jpath_blocks << 11, outfile);
1005f0d9efc0Sbeck last_extent_written += 2*jpath_blocks;
1006f0d9efc0Sbeck free(jpath_table_l);
1007f0d9efc0Sbeck free(jpath_table_m);
1008f0d9efc0Sbeck jpath_table_l = NULL;
1009f0d9efc0Sbeck jpath_table_m = NULL;
1010f0d9efc0Sbeck return 0;
1011f0d9efc0Sbeck }
1012f0d9efc0Sbeck
FDECL1(jdirtree_size,int,starting_extent)1013f0d9efc0Sbeck static int FDECL1(jdirtree_size, int, starting_extent)
1014f0d9efc0Sbeck {
1015f0d9efc0Sbeck assign_joliet_directory_addresses(root);
1016f0d9efc0Sbeck return 0;
1017f0d9efc0Sbeck }
1018f0d9efc0Sbeck
jroot_gen()1019f0d9efc0Sbeck static int jroot_gen()
1020f0d9efc0Sbeck {
1021f0d9efc0Sbeck jroot_record.length[0] = 1 + sizeof(struct iso_directory_record)
1022f0d9efc0Sbeck - sizeof(jroot_record.name);
1023f0d9efc0Sbeck jroot_record.ext_attr_length[0] = 0;
1024f0d9efc0Sbeck set_733((char *) jroot_record.extent, root->jextent);
1025f0d9efc0Sbeck set_733((char *) jroot_record.size, ROUND_UP(root->jsize));
1026f0d9efc0Sbeck iso9660_date(jroot_record.date, root_statbuf.st_mtime);
1027f0d9efc0Sbeck jroot_record.flags[0] = 2;
1028f0d9efc0Sbeck jroot_record.file_unit_size[0] = 0;
1029f0d9efc0Sbeck jroot_record.interleave[0] = 0;
1030f0d9efc0Sbeck set_723(jroot_record.volume_sequence_number, volume_sequence_number);
1031f0d9efc0Sbeck jroot_record.name_len[0] = 1;
1032f0d9efc0Sbeck return 0;
1033f0d9efc0Sbeck }
1034f0d9efc0Sbeck
FDECL1(jdirtree_write,FILE *,outfile)1035f0d9efc0Sbeck static int FDECL1(jdirtree_write, FILE *, outfile)
1036f0d9efc0Sbeck {
1037f0d9efc0Sbeck generate_joliet_directories(root, outfile);
1038f0d9efc0Sbeck return 0;
1039f0d9efc0Sbeck }
1040f0d9efc0Sbeck
1041f0d9efc0Sbeck /*
1042f0d9efc0Sbeck * Function to write the EVD for the disc.
1043f0d9efc0Sbeck */
FDECL1(jvd_write,FILE *,outfile)1044f0d9efc0Sbeck static int FDECL1(jvd_write, FILE *, outfile)
1045f0d9efc0Sbeck {
1046f0d9efc0Sbeck struct iso_primary_descriptor jvol_desc;
1047f0d9efc0Sbeck
1048f0d9efc0Sbeck /*
1049f0d9efc0Sbeck * Next we write out the boot volume descriptor for the disc
1050f0d9efc0Sbeck */
1051f0d9efc0Sbeck jvol_desc = vol_desc;
1052f0d9efc0Sbeck get_joliet_vol_desc(&jvol_desc);
1053f0d9efc0Sbeck xfwrite(&jvol_desc, 1, 2048, outfile);
1054f0d9efc0Sbeck last_extent_written ++;
1055f0d9efc0Sbeck return 0;
1056f0d9efc0Sbeck }
1057f0d9efc0Sbeck
1058f0d9efc0Sbeck /*
1059f0d9efc0Sbeck * Functions to describe padding block at the start of the disc.
1060f0d9efc0Sbeck */
FDECL1(jpathtab_size,int,starting_extent)1061f0d9efc0Sbeck static int FDECL1(jpathtab_size, int, starting_extent)
1062f0d9efc0Sbeck {
1063f0d9efc0Sbeck jpath_table[0] = starting_extent;
1064f0d9efc0Sbeck jpath_table[1] = 0;
1065f0d9efc0Sbeck jpath_table[2] = jpath_table[0] + jpath_blocks;
1066f0d9efc0Sbeck jpath_table[3] = 0;
1067f0d9efc0Sbeck
1068f0d9efc0Sbeck last_extent += 2*jpath_blocks;
1069f0d9efc0Sbeck return 0;
1070f0d9efc0Sbeck }
1071f0d9efc0Sbeck
1072f0d9efc0Sbeck struct output_fragment joliet_desc = {NULL, oneblock_size, jroot_gen,jvd_write};
1073f0d9efc0Sbeck struct output_fragment jpathtable_desc= {NULL, jpathtab_size, generate_joliet_path_tables, jpathtab_write};
1074f0d9efc0Sbeck struct output_fragment jdirtree_desc = {NULL, jdirtree_size, NULL, jdirtree_write};
1075