1/*	$NetBSD: cd9660_conversion.c,v 1.5 2017/02/08 21:33:12 christos Exp $	*/
2
3/*
4 * Copyright (c) 2005 Daniel Watt, Walter Deignan, Ryan Gabrys, Alan
5 * Perez-Rathke and Ram Vedam.  All rights reserved.
6 *
7 * This code was written by Daniel Watt, Walter Deignan, Ryan Gabrys,
8 * Alan Perez-Rathke and Ram Vedam.
9 *
10 * Redistribution and use in source and binary forms, with or
11 * without modification, are permitted provided that the following
12 * conditions are met:
13 * 1. Redistributions of source code must retain the above copyright
14 *    notice, this list of conditions and the following disclaimer.
15 * 2. Redistributions in binary form must reproduce the above
16 *    copyright notice, this list of conditions and the following
17 *    disclaimer in the documentation and/or other materials provided
18 *    with the distribution.
19 *
20 * THIS SOFTWARE IS PROVIDED BY DANIEL WATT, WALTER DEIGNAN, RYAN
21 * GABRYS, ALAN PEREZ-RATHKE AND RAM VEDAM ``AS IS'' AND ANY EXPRESS OR
22 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
23 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
24 * DISCLAIMED.  IN NO EVENT SHALL DANIEL WATT, WALTER DEIGNAN, RYAN
25 * GABRYS, ALAN PEREZ-RATHKE AND RAM VEDAM BE LIABLE FOR ANY DIRECT, INDIRECT,
26 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
27 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
28 * USE,DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
29 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
30 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY
32 * OF SUCH DAMAGE.
33 */
34#include "cd9660.h"
35
36#include <sys/cdefs.h>
37#if defined(__RCSID) && !defined(__lint)
38__RCSID("$NetBSD: cd9660_conversion.c,v 1.5 2017/02/08 21:33:12 christos Exp $");
39#endif  /* !__lint */
40
41
42static char cd9660_compute_gm_offset(time_t);
43
44#if 0
45static inline int
46cd9660_pad_even(length)
47int length;
48{
49	return length + (length & 0x01);
50}
51#endif
52
53/*
54* These can probably be implemented using a macro
55*/
56
57/* Little endian */
58void
59cd9660_721(uint16_t w, unsigned char *twochar)
60{
61#if BYTE_ORDER == BIG_ENDIAN
62	w = bswap16(w);
63#endif
64	memcpy(twochar,&w,2);
65}
66
67void
68cd9660_731(uint32_t w, unsigned char *fourchar)
69{
70#if BYTE_ORDER == BIG_ENDIAN
71	w = bswap32(w);
72#endif
73	memcpy(fourchar,&w,4);
74}
75
76/* Big endian */
77void
78cd9660_722(uint16_t w, unsigned char *twochar)
79{
80#if BYTE_ORDER == LITTLE_ENDIAN
81	w = bswap16(w);
82#endif
83	memcpy(twochar,&w,2);
84}
85
86void
87cd9660_732(uint32_t w, unsigned char *fourchar)
88{
89#if BYTE_ORDER == LITTLE_ENDIAN
90	w = bswap32(w);
91#endif
92	memcpy(fourchar,&w,4);
93}
94
95/**
96* Convert a dword into a double endian string of eight characters
97* @param int The double word to convert
98* @param char* The string to write the both endian double word to - It is assumed this is allocated and at least
99*		eight characters long
100*/
101void
102cd9660_bothendian_dword(uint32_t dw, unsigned char *eightchar)
103{
104	uint32_t le, be;
105#if BYTE_ORDER == LITTLE_ENDIAN
106	le = dw;
107	be = bswap32(dw);
108#endif
109#if BYTE_ORDER == BIG_ENDIAN
110	be = dw;
111	le = bswap32(dw);
112#endif
113	memcpy(eightchar, &le, 4);
114	memcpy((eightchar+4), &be, 4);
115}
116
117/**
118* Convert a word into a double endian string of four characters
119* @param int The word to convert
120* @param char* The string to write the both endian word to - It is assumed this is allocated and at least
121*		four characters long
122*/
123void
124cd9660_bothendian_word(uint16_t dw, unsigned char *fourchar)
125{
126	uint16_t le, be;
127#if BYTE_ORDER == LITTLE_ENDIAN
128	le = dw;
129	be = bswap16(dw);
130#endif
131#if BYTE_ORDER == BIG_ENDIAN
132	be = dw;
133	le = bswap16(dw);
134#endif
135	memcpy(fourchar, &le, 2);
136	memcpy((fourchar+2), &be, 2);
137}
138
139void
140cd9660_pad_string_spaces(char *str, int len)
141{
142	int i;
143
144	for (i = 0; i < len; i ++) {
145		if (str[i] == '\0')
146			str[i] = 0x20;
147	}
148}
149
150static char
151cd9660_compute_gm_offset(time_t tim)
152{
153	if (stampst.st_ino)
154		return 0;
155
156	struct tm t, gm;
157
158	(void)localtime_r(&tim, &t);
159	(void)gmtime_r(&tim, &gm);
160	gm.tm_year -= t.tm_year;
161	gm.tm_yday -= t.tm_yday;
162	gm.tm_hour -= t.tm_hour;
163	gm.tm_min  -= t.tm_min;
164	if (gm.tm_year < 0)
165		gm.tm_yday = -1;
166	else if (gm.tm_year > 0)
167		gm.tm_yday = 1;
168
169	return (char)(-(gm.tm_min + 60* (24 * gm.tm_yday + gm.tm_hour)) / 15);
170}
171
172/* Long dates: 17 characters */
173void
174cd9660_time_8426(unsigned char *buf, time_t tim)
175{
176	struct tm t;
177	char temp[18];
178
179	if (stampst.st_ino)
180		(void)gmtime_r(&tim, &t);
181	else
182		(void)localtime_r(&tim, &t);
183	(void)snprintf(temp, sizeof(temp), "%04i%02i%02i%02i%02i%02i%02i",
184		1900+(int)t.tm_year,
185		(int)t.tm_mon+1,
186		(int)t.tm_mday,
187		(int)t.tm_hour,
188		(int)t.tm_min,
189		(int)t.tm_sec,
190		0);
191	(void)memcpy(buf, temp, 16);
192	buf[16] = cd9660_compute_gm_offset(tim);
193}
194
195/* Short dates: 7 characters */
196void
197cd9660_time_915(unsigned char *buf, time_t tim)
198{
199	struct tm t;
200
201	if (stampst.st_ino)
202		(void)gmtime_r(&tim, &t);
203	else
204		(void)localtime_r(&tim, &t);
205	buf[0] = t.tm_year;
206	buf[1] = t.tm_mon+1;
207	buf[2] = t.tm_mday;
208	buf[3] = t.tm_hour;
209	buf[4] = t.tm_min;
210	buf[5] = t.tm_sec;
211	buf[6] = cd9660_compute_gm_offset(tim);
212}
213