15978408cSSascha Wildner /* $NetBSD: cd9660_archimedes.c,v 1.1 2009/01/10 22:06:29 bjh21 Exp $ */
25978408cSSascha Wildner
35978408cSSascha Wildner /*-
45978408cSSascha Wildner * SPDX-License-Identifier: BSD-3-Clause
55978408cSSascha Wildner *
65978408cSSascha Wildner * Copyright (c) 1998, 2009 Ben Harris
75978408cSSascha Wildner * All rights reserved.
85978408cSSascha Wildner *
95978408cSSascha Wildner * Redistribution and use in source and binary forms, with or without
105978408cSSascha Wildner * modification, are permitted provided that the following conditions
115978408cSSascha Wildner * are met:
125978408cSSascha Wildner * 1. Redistributions of source code must retain the above copyright
135978408cSSascha Wildner * notice, this list of conditions and the following disclaimer.
145978408cSSascha Wildner * 2. Redistributions in binary form must reproduce the above copyright
155978408cSSascha Wildner * notice, this list of conditions and the following disclaimer in the
165978408cSSascha Wildner * documentation and/or other materials provided with the distribution.
175978408cSSascha Wildner * 3. The name of the author may not be used to endorse or promote products
185978408cSSascha Wildner * derived from this software without specific prior written permission.
195978408cSSascha Wildner *
205978408cSSascha Wildner * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
215978408cSSascha Wildner * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
225978408cSSascha Wildner * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
235978408cSSascha Wildner * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
245978408cSSascha Wildner * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
255978408cSSascha Wildner * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
265978408cSSascha Wildner * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
275978408cSSascha Wildner * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
285978408cSSascha Wildner * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
295978408cSSascha Wildner * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30*811c2036SSascha Wildner *
31*811c2036SSascha Wildner * $FreeBSD: head/usr.sbin/makefs/cd9660/cd9660_archimedes.c 326276 2017-11-27 15:37:16Z pfg $
325978408cSSascha Wildner */
335978408cSSascha Wildner /*
345978408cSSascha Wildner * cd9660_archimedes.c - support for RISC OS "ARCHIMEDES" extension
355978408cSSascha Wildner *
365978408cSSascha Wildner * RISC OS CDFS looks for a special block at the end of the System Use
375978408cSSascha Wildner * Field for each file. If present, this contains the RISC OS load
385978408cSSascha Wildner * and exec address (used to hold the file timestamp and type), the
395978408cSSascha Wildner * file attributes, and a flag indicating whether the first character
405978408cSSascha Wildner * of the filename should be replaced with '!' (since many special
415978408cSSascha Wildner * RISC OS filenames do).
425978408cSSascha Wildner */
435978408cSSascha Wildner
445978408cSSascha Wildner #include <assert.h>
455978408cSSascha Wildner #include <stdint.h>
465978408cSSascha Wildner #include <stdio.h>
475978408cSSascha Wildner #include <string.h>
485978408cSSascha Wildner #include <util.h>
495978408cSSascha Wildner
505978408cSSascha Wildner #include "makefs.h"
515978408cSSascha Wildner #include "cd9660.h"
525978408cSSascha Wildner #include "cd9660_archimedes.h"
535978408cSSascha Wildner
545978408cSSascha Wildner /*
555978408cSSascha Wildner * Convert a Unix time_t (non-leap seconds since 1970-01-01) to a RISC
565978408cSSascha Wildner * OS time (non-leap(?) centiseconds since 1900-01-01(?)).
575978408cSSascha Wildner */
585978408cSSascha Wildner
595978408cSSascha Wildner static u_int64_t
riscos_date(time_t unixtime)605978408cSSascha Wildner riscos_date(time_t unixtime)
615978408cSSascha Wildner {
625978408cSSascha Wildner u_int64_t base;
635978408cSSascha Wildner
645978408cSSascha Wildner base = 31536000ULL * 70 + 86400 * 17;
655978408cSSascha Wildner return (((u_int64_t)unixtime) + base)*100;
665978408cSSascha Wildner }
675978408cSSascha Wildner
685978408cSSascha Wildner /*
695978408cSSascha Wildner * Add "ARCHIMEDES" metadata to a node if that seems appropriate.
705978408cSSascha Wildner *
715978408cSSascha Wildner * We touch regular files with names matching /,[0-9a-f]{3}$/ and
725978408cSSascha Wildner * directories matching /^!/.
735978408cSSascha Wildner */
745978408cSSascha Wildner static void
archimedes_convert_node(cd9660node * node)755978408cSSascha Wildner archimedes_convert_node(cd9660node *node)
765978408cSSascha Wildner {
775978408cSSascha Wildner struct ISO_ARCHIMEDES *arc;
785978408cSSascha Wildner size_t len;
795978408cSSascha Wildner int type = -1;
805978408cSSascha Wildner uint64_t stamp;
815978408cSSascha Wildner
825978408cSSascha Wildner if (node->su_tail_data != NULL)
835978408cSSascha Wildner /* Something else already has the tail. */
845978408cSSascha Wildner return;
855978408cSSascha Wildner
865978408cSSascha Wildner len = strlen(node->node->name);
875978408cSSascha Wildner if (len < 1) return;
885978408cSSascha Wildner
895978408cSSascha Wildner if (len >= 4 && node->node->name[len-4] == ',')
905978408cSSascha Wildner /* XXX should support ,xxx and ,lxa */
915978408cSSascha Wildner type = strtoul(node->node->name + len - 3, NULL, 16);
925978408cSSascha Wildner if (type == -1 && node->node->name[0] != '!')
935978408cSSascha Wildner return;
945978408cSSascha Wildner if (type == -1) type = 0;
955978408cSSascha Wildner
965978408cSSascha Wildner assert(sizeof(*arc) == 32);
975978408cSSascha Wildner arc = ecalloc(1, sizeof(*arc));
985978408cSSascha Wildner
995978408cSSascha Wildner stamp = riscos_date(node->node->inode->st.st_mtime);
1005978408cSSascha Wildner
1015978408cSSascha Wildner memcpy(arc->magic, "ARCHIMEDES", 10);
1025978408cSSascha Wildner cd9660_731(0xfff00000 | (type << 8) | (stamp >> 32), arc->loadaddr);
1035978408cSSascha Wildner cd9660_731(stamp & 0x00ffffffffULL, arc->execaddr);
1045978408cSSascha Wildner arc->ro_attr = RO_ACCESS_UR | RO_ACCESS_OR;
1055978408cSSascha Wildner arc->cdfs_attr = node->node->name[0] == '!' ? CDFS_PLING : 0;
1065978408cSSascha Wildner node->su_tail_data = (void *)arc;
1075978408cSSascha Wildner node->su_tail_size = sizeof(*arc);
1085978408cSSascha Wildner }
1095978408cSSascha Wildner
1105978408cSSascha Wildner /*
1115978408cSSascha Wildner * Add "ARCHIMEDES" metadata to an entire tree recursively.
1125978408cSSascha Wildner */
1135978408cSSascha Wildner void
archimedes_convert_tree(cd9660node * node)1145978408cSSascha Wildner archimedes_convert_tree(cd9660node *node)
1155978408cSSascha Wildner {
1165978408cSSascha Wildner cd9660node *cn;
1175978408cSSascha Wildner
1185978408cSSascha Wildner assert(node != NULL);
1195978408cSSascha Wildner
1205978408cSSascha Wildner archimedes_convert_node(node);
1215978408cSSascha Wildner
1225978408cSSascha Wildner /* Recurse on children. */
1235978408cSSascha Wildner TAILQ_FOREACH(cn, &node->cn_children, cn_next_child)
1245978408cSSascha Wildner archimedes_convert_tree(cn);
1255978408cSSascha Wildner }
126