xref: /openbsd/gnu/usr.sbin/mkhybrid/src/joliet.c (revision 463a9bef)
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