15978408cSSascha Wildner /* $NetBSD: cd9660_conversion.c,v 1.4 2007/03/14 14:11:17 christos Exp $ */
25978408cSSascha Wildner
35978408cSSascha Wildner /*-
45978408cSSascha Wildner * SPDX-License-Identifier: BSD-2-Clause-NetBSD
55978408cSSascha Wildner *
65978408cSSascha Wildner * Copyright (c) 2005 Daniel Watt, Walter Deignan, Ryan Gabrys, Alan
75978408cSSascha Wildner * Perez-Rathke and Ram Vedam. All rights reserved.
85978408cSSascha Wildner *
95978408cSSascha Wildner * This code was written by Daniel Watt, Walter Deignan, Ryan Gabrys,
105978408cSSascha Wildner * Alan Perez-Rathke and Ram Vedam.
115978408cSSascha Wildner *
125978408cSSascha Wildner * Redistribution and use in source and binary forms, with or
135978408cSSascha Wildner * without modification, are permitted provided that the following
145978408cSSascha Wildner * conditions are met:
155978408cSSascha Wildner * 1. Redistributions of source code must retain the above copyright
165978408cSSascha Wildner * notice, this list of conditions and the following disclaimer.
175978408cSSascha Wildner * 2. Redistributions in binary form must reproduce the above
185978408cSSascha Wildner * copyright notice, this list of conditions and the following
195978408cSSascha Wildner * disclaimer in the documentation and/or other materials provided
205978408cSSascha Wildner * with the distribution.
215978408cSSascha Wildner *
225978408cSSascha Wildner * THIS SOFTWARE IS PROVIDED BY DANIEL WATT, WALTER DEIGNAN, RYAN
235978408cSSascha Wildner * GABRYS, ALAN PEREZ-RATHKE AND RAM VEDAM ``AS IS'' AND ANY EXPRESS OR
245978408cSSascha Wildner * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
255978408cSSascha Wildner * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
265978408cSSascha Wildner * DISCLAIMED. IN NO EVENT SHALL DANIEL WATT, WALTER DEIGNAN, RYAN
275978408cSSascha Wildner * GABRYS, ALAN PEREZ-RATHKE AND RAM VEDAM BE LIABLE FOR ANY DIRECT, INDIRECT,
285978408cSSascha Wildner * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
295978408cSSascha Wildner * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
305978408cSSascha Wildner * USE,DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
315978408cSSascha Wildner * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
325978408cSSascha Wildner * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
335978408cSSascha Wildner * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY
345978408cSSascha Wildner * OF SUCH DAMAGE.
35*811c2036SSascha Wildner *
36*811c2036SSascha Wildner * $FreeBSD: head/usr.sbin/makefs/cd9660/cd9660_conversion.c 326276 2017-11-27 15:37:16Z pfg $
375978408cSSascha Wildner */
385978408cSSascha Wildner #include "cd9660.h"
395978408cSSascha Wildner
405978408cSSascha Wildner static char cd9660_compute_gm_offset(time_t);
415978408cSSascha Wildner
425978408cSSascha Wildner #if 0
435978408cSSascha Wildner static inline int
445978408cSSascha Wildner cd9660_pad_even(length)
455978408cSSascha Wildner int length;
465978408cSSascha Wildner {
475978408cSSascha Wildner return length + (length & 0x01);
485978408cSSascha Wildner }
495978408cSSascha Wildner #endif
505978408cSSascha Wildner
515978408cSSascha Wildner /*
525978408cSSascha Wildner * These can probably be implemented using a macro
535978408cSSascha Wildner */
545978408cSSascha Wildner
555978408cSSascha Wildner /* Little endian */
565978408cSSascha Wildner void
cd9660_721(uint16_t w,unsigned char * twochar)575978408cSSascha Wildner cd9660_721(uint16_t w, unsigned char *twochar)
585978408cSSascha Wildner {
595978408cSSascha Wildner #if BYTE_ORDER == BIG_ENDIAN
605978408cSSascha Wildner w = bswap16(w);
615978408cSSascha Wildner #endif
625978408cSSascha Wildner memcpy(twochar,&w,2);
635978408cSSascha Wildner }
645978408cSSascha Wildner
655978408cSSascha Wildner void
cd9660_731(uint32_t w,unsigned char * fourchar)665978408cSSascha Wildner cd9660_731(uint32_t w, unsigned char *fourchar)
675978408cSSascha Wildner {
685978408cSSascha Wildner #if BYTE_ORDER == BIG_ENDIAN
695978408cSSascha Wildner w = bswap32(w);
705978408cSSascha Wildner #endif
715978408cSSascha Wildner memcpy(fourchar,&w,4);
725978408cSSascha Wildner }
735978408cSSascha Wildner
745978408cSSascha Wildner /* Big endian */
755978408cSSascha Wildner void
cd9660_722(uint16_t w,unsigned char * twochar)765978408cSSascha Wildner cd9660_722(uint16_t w, unsigned char *twochar)
775978408cSSascha Wildner {
785978408cSSascha Wildner #if BYTE_ORDER == LITTLE_ENDIAN
795978408cSSascha Wildner w = bswap16(w);
805978408cSSascha Wildner #endif
815978408cSSascha Wildner memcpy(twochar,&w,2);
825978408cSSascha Wildner }
835978408cSSascha Wildner
845978408cSSascha Wildner void
cd9660_732(uint32_t w,unsigned char * fourchar)855978408cSSascha Wildner cd9660_732(uint32_t w, unsigned char *fourchar)
865978408cSSascha Wildner {
875978408cSSascha Wildner #if BYTE_ORDER == LITTLE_ENDIAN
885978408cSSascha Wildner w = bswap32(w);
895978408cSSascha Wildner #endif
905978408cSSascha Wildner memcpy(fourchar,&w,4);
915978408cSSascha Wildner }
925978408cSSascha Wildner
935978408cSSascha Wildner /**
945978408cSSascha Wildner * Convert a dword into a double endian string of eight characters
955978408cSSascha Wildner * @param int The double word to convert
965978408cSSascha Wildner * @param char* The string to write the both endian double word to - It is assumed this is allocated and at least
975978408cSSascha Wildner * eight characters long
985978408cSSascha Wildner */
995978408cSSascha Wildner void
cd9660_bothendian_dword(uint32_t dw,unsigned char * eightchar)1005978408cSSascha Wildner cd9660_bothendian_dword(uint32_t dw, unsigned char *eightchar)
1015978408cSSascha Wildner {
1025978408cSSascha Wildner uint32_t le, be;
1035978408cSSascha Wildner #if BYTE_ORDER == LITTLE_ENDIAN
1045978408cSSascha Wildner le = dw;
1055978408cSSascha Wildner be = bswap32(dw);
1065978408cSSascha Wildner #endif
1075978408cSSascha Wildner #if BYTE_ORDER == BIG_ENDIAN
1085978408cSSascha Wildner be = dw;
1095978408cSSascha Wildner le = bswap32(dw);
1105978408cSSascha Wildner #endif
1115978408cSSascha Wildner memcpy(eightchar, &le, 4);
1125978408cSSascha Wildner memcpy((eightchar+4), &be, 4);
1135978408cSSascha Wildner }
1145978408cSSascha Wildner
1155978408cSSascha Wildner /**
1165978408cSSascha Wildner * Convert a word into a double endian string of four characters
1175978408cSSascha Wildner * @param int The word to convert
1185978408cSSascha Wildner * @param char* The string to write the both endian word to - It is assumed this is allocated and at least
1195978408cSSascha Wildner * four characters long
1205978408cSSascha Wildner */
1215978408cSSascha Wildner void
cd9660_bothendian_word(uint16_t dw,unsigned char * fourchar)1225978408cSSascha Wildner cd9660_bothendian_word(uint16_t dw, unsigned char *fourchar)
1235978408cSSascha Wildner {
1245978408cSSascha Wildner uint16_t le, be;
1255978408cSSascha Wildner #if BYTE_ORDER == LITTLE_ENDIAN
1265978408cSSascha Wildner le = dw;
1275978408cSSascha Wildner be = bswap16(dw);
1285978408cSSascha Wildner #endif
1295978408cSSascha Wildner #if BYTE_ORDER == BIG_ENDIAN
1305978408cSSascha Wildner be = dw;
1315978408cSSascha Wildner le = bswap16(dw);
1325978408cSSascha Wildner #endif
1335978408cSSascha Wildner memcpy(fourchar, &le, 2);
1345978408cSSascha Wildner memcpy((fourchar+2), &be, 2);
1355978408cSSascha Wildner }
1365978408cSSascha Wildner
1375978408cSSascha Wildner void
cd9660_pad_string_spaces(char * str,int len)1385978408cSSascha Wildner cd9660_pad_string_spaces(char *str, int len)
1395978408cSSascha Wildner {
1405978408cSSascha Wildner int i;
1415978408cSSascha Wildner
1425978408cSSascha Wildner for (i = 0; i < len; i ++) {
1435978408cSSascha Wildner if (str[i] == '\0')
1445978408cSSascha Wildner str[i] = 0x20;
1455978408cSSascha Wildner }
1465978408cSSascha Wildner }
1475978408cSSascha Wildner
1485978408cSSascha Wildner static char
cd9660_compute_gm_offset(time_t tim)1495978408cSSascha Wildner cd9660_compute_gm_offset(time_t tim)
1505978408cSSascha Wildner {
1515978408cSSascha Wildner struct tm t, gm;
1525978408cSSascha Wildner
1535978408cSSascha Wildner (void)localtime_r(&tim, &t);
1545978408cSSascha Wildner (void)gmtime_r(&tim, &gm);
1555978408cSSascha Wildner gm.tm_year -= t.tm_year;
1565978408cSSascha Wildner gm.tm_yday -= t.tm_yday;
1575978408cSSascha Wildner gm.tm_hour -= t.tm_hour;
1585978408cSSascha Wildner gm.tm_min -= t.tm_min;
1595978408cSSascha Wildner if (gm.tm_year < 0)
1605978408cSSascha Wildner gm.tm_yday = -1;
1615978408cSSascha Wildner else if (gm.tm_year > 0)
1625978408cSSascha Wildner gm.tm_yday = 1;
1635978408cSSascha Wildner
1645978408cSSascha Wildner return (char)(-(gm.tm_min + 60* (24 * gm.tm_yday + gm.tm_hour)) / 15);
1655978408cSSascha Wildner }
1665978408cSSascha Wildner
1675978408cSSascha Wildner /* Long dates: 17 characters */
1685978408cSSascha Wildner void
cd9660_time_8426(unsigned char * buf,time_t tim)1695978408cSSascha Wildner cd9660_time_8426(unsigned char *buf, time_t tim)
1705978408cSSascha Wildner {
1715978408cSSascha Wildner struct tm t;
1725978408cSSascha Wildner char temp[18];
1735978408cSSascha Wildner
1745978408cSSascha Wildner (void)localtime_r(&tim, &t);
1755978408cSSascha Wildner (void)snprintf(temp, sizeof(temp), "%04i%02i%02i%02i%02i%02i%02i",
1765978408cSSascha Wildner 1900+(int)t.tm_year,
1775978408cSSascha Wildner (int)t.tm_mon+1,
1785978408cSSascha Wildner (int)t.tm_mday,
1795978408cSSascha Wildner (int)t.tm_hour,
1805978408cSSascha Wildner (int)t.tm_min,
1815978408cSSascha Wildner (int)t.tm_sec,
1825978408cSSascha Wildner 0);
1835978408cSSascha Wildner (void)memcpy(buf, temp, 16);
1845978408cSSascha Wildner buf[16] = cd9660_compute_gm_offset(tim);
1855978408cSSascha Wildner }
1865978408cSSascha Wildner
1875978408cSSascha Wildner /* Short dates: 7 characters */
1885978408cSSascha Wildner void
cd9660_time_915(unsigned char * buf,time_t tim)1895978408cSSascha Wildner cd9660_time_915(unsigned char *buf, time_t tim)
1905978408cSSascha Wildner {
1915978408cSSascha Wildner struct tm t;
1925978408cSSascha Wildner
1935978408cSSascha Wildner (void)localtime_r(&tim, &t);
1945978408cSSascha Wildner buf[0] = t.tm_year;
1955978408cSSascha Wildner buf[1] = t.tm_mon+1;
1965978408cSSascha Wildner buf[2] = t.tm_mday;
1975978408cSSascha Wildner buf[3] = t.tm_hour;
1985978408cSSascha Wildner buf[4] = t.tm_min;
1995978408cSSascha Wildner buf[5] = t.tm_sec;
2005978408cSSascha Wildner buf[6] = cd9660_compute_gm_offset(tim);
2015978408cSSascha Wildner }
202