xref: /openbsd-src/usr.sbin/makefs/cd9660/cd9660_conversion.c (revision 9a86cdb6ba5369e6dd43374dabd04379f4952a21)
1*9a86cdb6Snatano /*	$OpenBSD: cd9660_conversion.c,v 1.2 2016/10/16 20:26:56 natano Exp $	*/
26163fc9cSnatano /*	$NetBSD: cd9660_conversion.c,v 1.4 2007/03/14 14:11:17 christos Exp $	*/
36163fc9cSnatano 
46163fc9cSnatano /*
56163fc9cSnatano  * Copyright (c) 2005 Daniel Watt, Walter Deignan, Ryan Gabrys, Alan
66163fc9cSnatano  * Perez-Rathke and Ram Vedam.  All rights reserved.
76163fc9cSnatano  *
86163fc9cSnatano  * This code was written by Daniel Watt, Walter Deignan, Ryan Gabrys,
96163fc9cSnatano  * Alan Perez-Rathke and Ram Vedam.
106163fc9cSnatano  *
116163fc9cSnatano  * Redistribution and use in source and binary forms, with or
126163fc9cSnatano  * without modification, are permitted provided that the following
136163fc9cSnatano  * conditions are met:
146163fc9cSnatano  * 1. Redistributions of source code must retain the above copyright
156163fc9cSnatano  *    notice, this list of conditions and the following disclaimer.
166163fc9cSnatano  * 2. Redistributions in binary form must reproduce the above
176163fc9cSnatano  *    copyright notice, this list of conditions and the following
186163fc9cSnatano  *    disclaimer in the documentation and/or other materials provided
196163fc9cSnatano  *    with the distribution.
206163fc9cSnatano  *
216163fc9cSnatano  * THIS SOFTWARE IS PROVIDED BY DANIEL WATT, WALTER DEIGNAN, RYAN
226163fc9cSnatano  * GABRYS, ALAN PEREZ-RATHKE AND RAM VEDAM ``AS IS'' AND ANY EXPRESS OR
236163fc9cSnatano  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
246163fc9cSnatano  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
256163fc9cSnatano  * DISCLAIMED.  IN NO EVENT SHALL DANIEL WATT, WALTER DEIGNAN, RYAN
266163fc9cSnatano  * GABRYS, ALAN PEREZ-RATHKE AND RAM VEDAM BE LIABLE FOR ANY DIRECT, INDIRECT,
276163fc9cSnatano  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
286163fc9cSnatano  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
296163fc9cSnatano  * USE,DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
306163fc9cSnatano  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
316163fc9cSnatano  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
326163fc9cSnatano  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY
336163fc9cSnatano  * OF SUCH DAMAGE.
346163fc9cSnatano  */
356163fc9cSnatano #include "cd9660.h"
366163fc9cSnatano 
376163fc9cSnatano #define bswap16 swap16
386163fc9cSnatano #define bswap32 swap32
396163fc9cSnatano 
406163fc9cSnatano 
416163fc9cSnatano static char cd9660_compute_gm_offset(time_t);
426163fc9cSnatano 
436163fc9cSnatano #if 0
446163fc9cSnatano static inline int
456163fc9cSnatano cd9660_pad_even(length)
466163fc9cSnatano int length;
476163fc9cSnatano {
486163fc9cSnatano 	return length + (length & 0x01);
496163fc9cSnatano }
506163fc9cSnatano #endif
516163fc9cSnatano 
526163fc9cSnatano /*
536163fc9cSnatano * These can probably be implemented using a macro
546163fc9cSnatano */
556163fc9cSnatano 
566163fc9cSnatano /* Little endian */
576163fc9cSnatano void
cd9660_721(uint16_t w,unsigned char * twochar)586163fc9cSnatano cd9660_721(uint16_t w, unsigned char *twochar)
596163fc9cSnatano {
606163fc9cSnatano #if BYTE_ORDER == BIG_ENDIAN
616163fc9cSnatano 	w = bswap16(w);
626163fc9cSnatano #endif
636163fc9cSnatano 	memcpy(twochar,&w,2);
646163fc9cSnatano }
656163fc9cSnatano 
666163fc9cSnatano void
cd9660_731(uint32_t w,unsigned char * fourchar)676163fc9cSnatano cd9660_731(uint32_t w, unsigned char *fourchar)
686163fc9cSnatano {
696163fc9cSnatano #if BYTE_ORDER == BIG_ENDIAN
706163fc9cSnatano 	w = bswap32(w);
716163fc9cSnatano #endif
726163fc9cSnatano 	memcpy(fourchar,&w,4);
736163fc9cSnatano }
746163fc9cSnatano 
756163fc9cSnatano /* Big endian */
766163fc9cSnatano void
cd9660_722(uint16_t w,unsigned char * twochar)776163fc9cSnatano cd9660_722(uint16_t w, unsigned char *twochar)
786163fc9cSnatano {
796163fc9cSnatano #if BYTE_ORDER == LITTLE_ENDIAN
806163fc9cSnatano 	w = bswap16(w);
816163fc9cSnatano #endif
826163fc9cSnatano 	memcpy(twochar,&w,2);
836163fc9cSnatano }
846163fc9cSnatano 
856163fc9cSnatano void
cd9660_732(uint32_t w,unsigned char * fourchar)866163fc9cSnatano cd9660_732(uint32_t w, unsigned char *fourchar)
876163fc9cSnatano {
886163fc9cSnatano #if BYTE_ORDER == LITTLE_ENDIAN
896163fc9cSnatano 	w = bswap32(w);
906163fc9cSnatano #endif
916163fc9cSnatano 	memcpy(fourchar,&w,4);
926163fc9cSnatano }
936163fc9cSnatano 
946163fc9cSnatano /**
956163fc9cSnatano * Convert a dword into a double endian string of eight characters
966163fc9cSnatano * @param int The double word to convert
976163fc9cSnatano * @param char* The string to write the both endian double word to - It is assumed this is allocated and at least
986163fc9cSnatano *		eight characters long
996163fc9cSnatano */
1006163fc9cSnatano void
cd9660_bothendian_dword(uint32_t dw,unsigned char * eightchar)1016163fc9cSnatano cd9660_bothendian_dword(uint32_t dw, unsigned char *eightchar)
1026163fc9cSnatano {
1036163fc9cSnatano 	uint32_t le, be;
1046163fc9cSnatano #if BYTE_ORDER == LITTLE_ENDIAN
1056163fc9cSnatano 	le = dw;
1066163fc9cSnatano 	be = bswap32(dw);
1076163fc9cSnatano #endif
1086163fc9cSnatano #if BYTE_ORDER == BIG_ENDIAN
1096163fc9cSnatano 	be = dw;
1106163fc9cSnatano 	le = bswap32(dw);
1116163fc9cSnatano #endif
1126163fc9cSnatano 	memcpy(eightchar, &le, 4);
1136163fc9cSnatano 	memcpy((eightchar+4), &be, 4);
1146163fc9cSnatano }
1156163fc9cSnatano 
1166163fc9cSnatano /**
1176163fc9cSnatano * Convert a word into a double endian string of four characters
1186163fc9cSnatano * @param int The word to convert
1196163fc9cSnatano * @param char* The string to write the both endian word to - It is assumed this is allocated and at least
1206163fc9cSnatano *		four characters long
1216163fc9cSnatano */
1226163fc9cSnatano void
cd9660_bothendian_word(uint16_t dw,unsigned char * fourchar)1236163fc9cSnatano cd9660_bothendian_word(uint16_t dw, unsigned char *fourchar)
1246163fc9cSnatano {
1256163fc9cSnatano 	uint16_t le, be;
1266163fc9cSnatano #if BYTE_ORDER == LITTLE_ENDIAN
1276163fc9cSnatano 	le = dw;
1286163fc9cSnatano 	be = bswap16(dw);
1296163fc9cSnatano #endif
1306163fc9cSnatano #if BYTE_ORDER == BIG_ENDIAN
1316163fc9cSnatano 	be = dw;
1326163fc9cSnatano 	le = bswap16(dw);
1336163fc9cSnatano #endif
1346163fc9cSnatano 	memcpy(fourchar, &le, 2);
1356163fc9cSnatano 	memcpy((fourchar+2), &be, 2);
1366163fc9cSnatano }
1376163fc9cSnatano 
1386163fc9cSnatano void
cd9660_pad_string_spaces(char * str,int len)1396163fc9cSnatano cd9660_pad_string_spaces(char *str, int len)
1406163fc9cSnatano {
1416163fc9cSnatano 	int i;
1426163fc9cSnatano 
1436163fc9cSnatano 	for (i = 0; i < len; i ++) {
1446163fc9cSnatano 		if (str[i] == '\0')
1456163fc9cSnatano 			str[i] = 0x20;
1466163fc9cSnatano 	}
1476163fc9cSnatano }
1486163fc9cSnatano 
1496163fc9cSnatano static char
cd9660_compute_gm_offset(time_t tim)1506163fc9cSnatano cd9660_compute_gm_offset(time_t tim)
1516163fc9cSnatano {
1526163fc9cSnatano 	struct tm t, gm;
1536163fc9cSnatano 
1546163fc9cSnatano 	(void)localtime_r(&tim, &t);
1556163fc9cSnatano 	(void)gmtime_r(&tim, &gm);
1566163fc9cSnatano 	gm.tm_year -= t.tm_year;
1576163fc9cSnatano 	gm.tm_yday -= t.tm_yday;
1586163fc9cSnatano 	gm.tm_hour -= t.tm_hour;
1596163fc9cSnatano 	gm.tm_min  -= t.tm_min;
1606163fc9cSnatano 	if (gm.tm_year < 0)
1616163fc9cSnatano 		gm.tm_yday = -1;
1626163fc9cSnatano 	else if (gm.tm_year > 0)
1636163fc9cSnatano 		gm.tm_yday = 1;
1646163fc9cSnatano 
1656163fc9cSnatano 	return (char)(-(gm.tm_min + 60* (24 * gm.tm_yday + gm.tm_hour)) / 15);
1666163fc9cSnatano }
1676163fc9cSnatano 
1686163fc9cSnatano /* Long dates: 17 characters */
1696163fc9cSnatano void
cd9660_time_8426(unsigned char * buf,time_t tim)1706163fc9cSnatano cd9660_time_8426(unsigned char *buf, time_t tim)
1716163fc9cSnatano {
1726163fc9cSnatano 	struct tm t;
1736163fc9cSnatano 	char temp[18];
1746163fc9cSnatano 
1756163fc9cSnatano 	(void)localtime_r(&tim, &t);
1766163fc9cSnatano 	(void)snprintf(temp, sizeof(temp), "%04i%02i%02i%02i%02i%02i%02i",
1776163fc9cSnatano 		1900+(int)t.tm_year,
1786163fc9cSnatano 		(int)t.tm_mon+1,
1796163fc9cSnatano 		(int)t.tm_mday,
1806163fc9cSnatano 		(int)t.tm_hour,
1816163fc9cSnatano 		(int)t.tm_min,
1826163fc9cSnatano 		(int)t.tm_sec,
1836163fc9cSnatano 		0);
1846163fc9cSnatano 	(void)memcpy(buf, temp, 16);
1856163fc9cSnatano 	buf[16] = cd9660_compute_gm_offset(tim);
1866163fc9cSnatano }
1876163fc9cSnatano 
1886163fc9cSnatano /* Short dates: 7 characters */
1896163fc9cSnatano void
cd9660_time_915(unsigned char * buf,time_t tim)1906163fc9cSnatano cd9660_time_915(unsigned char *buf, time_t tim)
1916163fc9cSnatano {
1926163fc9cSnatano 	struct tm t;
1936163fc9cSnatano 
1946163fc9cSnatano 	(void)localtime_r(&tim, &t);
1956163fc9cSnatano 	buf[0] = t.tm_year;
1966163fc9cSnatano 	buf[1] = t.tm_mon+1;
1976163fc9cSnatano 	buf[2] = t.tm_mday;
1986163fc9cSnatano 	buf[3] = t.tm_hour;
1996163fc9cSnatano 	buf[4] = t.tm_min;
2006163fc9cSnatano 	buf[5] = t.tm_sec;
2016163fc9cSnatano 	buf[6] = cd9660_compute_gm_offset(tim);
2026163fc9cSnatano }
203