xref: /dflybsd-src/usr.sbin/makefs/hammer2/hammer2_subr.c (revision 6da97f51504a575acd558f21089e0b2bcfbe8b37)
12d60b848STomohiro Kusumi /*
22d60b848STomohiro Kusumi  * SPDX-License-Identifier: BSD-3-Clause
32d60b848STomohiro Kusumi  *
42d60b848STomohiro Kusumi  * Copyright (c) 2022 Tomohiro Kusumi <tkusumi@netbsd.org>
52d60b848STomohiro Kusumi  * Copyright (c) 2011-2022 The DragonFly Project.  All rights reserved.
62d60b848STomohiro Kusumi  *
72d60b848STomohiro Kusumi  * This code is derived from software contributed to The DragonFly Project
82d60b848STomohiro Kusumi  * by Matthew Dillon <dillon@dragonflybsd.org>
92d60b848STomohiro Kusumi  *
102d60b848STomohiro Kusumi  * Redistribution and use in source and binary forms, with or without
112d60b848STomohiro Kusumi  * modification, are permitted provided that the following conditions
122d60b848STomohiro Kusumi  * are met:
132d60b848STomohiro Kusumi  *
142d60b848STomohiro Kusumi  * 1. Redistributions of source code must retain the above copyright
152d60b848STomohiro Kusumi  *    notice, this list of conditions and the following disclaimer.
162d60b848STomohiro Kusumi  * 2. Redistributions in binary form must reproduce the above copyright
172d60b848STomohiro Kusumi  *    notice, this list of conditions and the following disclaimer in
182d60b848STomohiro Kusumi  *    the documentation and/or other materials provided with the
192d60b848STomohiro Kusumi  *    distribution.
202d60b848STomohiro Kusumi  * 3. Neither the name of The DragonFly Project nor the names of its
212d60b848STomohiro Kusumi  *    contributors may be used to endorse or promote products derived
222d60b848STomohiro Kusumi  *    from this software without specific, prior written permission.
232d60b848STomohiro Kusumi  *
242d60b848STomohiro Kusumi  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
252d60b848STomohiro Kusumi  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
262d60b848STomohiro Kusumi  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
272d60b848STomohiro Kusumi  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE
282d60b848STomohiro Kusumi  * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
292d60b848STomohiro Kusumi  * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
302d60b848STomohiro Kusumi  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
312d60b848STomohiro Kusumi  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
322d60b848STomohiro Kusumi  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
332d60b848STomohiro Kusumi  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
342d60b848STomohiro Kusumi  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
352d60b848STomohiro Kusumi  * SUCH DAMAGE.
362d60b848STomohiro Kusumi  */
372d60b848STomohiro Kusumi /*
382d60b848STomohiro Kusumi #include <sys/cdefs.h>
392d60b848STomohiro Kusumi #include <sys/param.h>
402d60b848STomohiro Kusumi #include <sys/systm.h>
412d60b848STomohiro Kusumi #include <sys/types.h>
422d60b848STomohiro Kusumi #include <sys/uuid.h>
432d60b848STomohiro Kusumi */
44*6da97f51STomohiro Kusumi #include <sys/time.h>
452d60b848STomohiro Kusumi #include <sys/dirent.h>
462d60b848STomohiro Kusumi 
472d60b848STomohiro Kusumi #include "hammer2.h"
48aa86710aSTomohiro Kusumi #include "makefs.h"
492d60b848STomohiro Kusumi 
502d60b848STomohiro Kusumi /*
512d60b848STomohiro Kusumi  * Return the directory entry type for an inode.
522d60b848STomohiro Kusumi  */
532d60b848STomohiro Kusumi int
hammer2_get_dtype(uint8_t type)542d60b848STomohiro Kusumi hammer2_get_dtype(uint8_t type)
552d60b848STomohiro Kusumi {
562d60b848STomohiro Kusumi 	switch(type) {
572d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_UNKNOWN:
582d60b848STomohiro Kusumi 		return (DT_UNKNOWN);
592d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_DIRECTORY:
602d60b848STomohiro Kusumi 		return (DT_DIR);
612d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_REGFILE:
622d60b848STomohiro Kusumi 		return (DT_REG);
632d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_FIFO:
642d60b848STomohiro Kusumi 		return (DT_FIFO);
652d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_CDEV:
662d60b848STomohiro Kusumi 		return (DT_CHR);
672d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_BDEV:
682d60b848STomohiro Kusumi 		return (DT_BLK);
692d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_SOFTLINK:
702d60b848STomohiro Kusumi 		return (DT_LNK);
712d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_SOCKET:
722d60b848STomohiro Kusumi 		return (DT_SOCK);
732d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_WHITEOUT:	/* not supported */
742d60b848STomohiro Kusumi 		return (DT_UNKNOWN);
752d60b848STomohiro Kusumi 	default:
762d60b848STomohiro Kusumi 		return (DT_UNKNOWN);
772d60b848STomohiro Kusumi 	}
782d60b848STomohiro Kusumi 	/* not reached */
792d60b848STomohiro Kusumi }
802d60b848STomohiro Kusumi 
812d60b848STomohiro Kusumi /*
822d60b848STomohiro Kusumi  * Return the directory entry type for an inode
832d60b848STomohiro Kusumi  */
842d60b848STomohiro Kusumi int
hammer2_get_vtype(uint8_t type)852d60b848STomohiro Kusumi hammer2_get_vtype(uint8_t type)
862d60b848STomohiro Kusumi {
872d60b848STomohiro Kusumi 	switch(type) {
882d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_UNKNOWN:
892d60b848STomohiro Kusumi 		return (VBAD);
902d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_DIRECTORY:
912d60b848STomohiro Kusumi 		return (VDIR);
922d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_REGFILE:
932d60b848STomohiro Kusumi 		return (VREG);
942d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_FIFO:
952d60b848STomohiro Kusumi 		return (VFIFO);
962d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_CDEV:
972d60b848STomohiro Kusumi 		return (VCHR);
982d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_BDEV:
992d60b848STomohiro Kusumi 		return (VBLK);
1002d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_SOFTLINK:
1012d60b848STomohiro Kusumi 		return (VLNK);
1022d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_SOCKET:
1032d60b848STomohiro Kusumi 		return (VSOCK);
1042d60b848STomohiro Kusumi 	case HAMMER2_OBJTYPE_WHITEOUT:	/* not supported */
1052d60b848STomohiro Kusumi 		return (VBAD);
1062d60b848STomohiro Kusumi 	default:
1072d60b848STomohiro Kusumi 		return (VBAD);
1082d60b848STomohiro Kusumi 	}
1092d60b848STomohiro Kusumi 	/* not reached */
1102d60b848STomohiro Kusumi }
1112d60b848STomohiro Kusumi 
1122d60b848STomohiro Kusumi uint8_t
hammer2_get_obj_type(enum vtype vtype)1132d60b848STomohiro Kusumi hammer2_get_obj_type(enum vtype vtype)
1142d60b848STomohiro Kusumi {
1152d60b848STomohiro Kusumi 	switch(vtype) {
1162d60b848STomohiro Kusumi 	case VDIR:
1172d60b848STomohiro Kusumi 		return(HAMMER2_OBJTYPE_DIRECTORY);
1182d60b848STomohiro Kusumi 	case VREG:
1192d60b848STomohiro Kusumi 		return(HAMMER2_OBJTYPE_REGFILE);
1202d60b848STomohiro Kusumi 	case VFIFO:
1212d60b848STomohiro Kusumi 		return(HAMMER2_OBJTYPE_FIFO);
1222d60b848STomohiro Kusumi 	case VSOCK:
1232d60b848STomohiro Kusumi 		return(HAMMER2_OBJTYPE_SOCKET);
1242d60b848STomohiro Kusumi 	case VCHR:
1252d60b848STomohiro Kusumi 		return(HAMMER2_OBJTYPE_CDEV);
1262d60b848STomohiro Kusumi 	case VBLK:
1272d60b848STomohiro Kusumi 		return(HAMMER2_OBJTYPE_BDEV);
1282d60b848STomohiro Kusumi 	case VLNK:
1292d60b848STomohiro Kusumi 		return(HAMMER2_OBJTYPE_SOFTLINK);
1302d60b848STomohiro Kusumi 	default:
1312d60b848STomohiro Kusumi 		return(HAMMER2_OBJTYPE_UNKNOWN);
1322d60b848STomohiro Kusumi 	}
1332d60b848STomohiro Kusumi 	/* not reached */
1342d60b848STomohiro Kusumi }
1352d60b848STomohiro Kusumi 
1362d60b848STomohiro Kusumi /*
1372d60b848STomohiro Kusumi  * Convert a hammer2 64-bit time to a timespec.
1382d60b848STomohiro Kusumi  */
1392d60b848STomohiro Kusumi void
hammer2_time_to_timespec(uint64_t xtime,struct timespec * ts)1402d60b848STomohiro Kusumi hammer2_time_to_timespec(uint64_t xtime, struct timespec *ts)
1412d60b848STomohiro Kusumi {
1422d60b848STomohiro Kusumi 	ts->tv_sec = (unsigned long)(xtime / 1000000);
1432d60b848STomohiro Kusumi 	ts->tv_nsec = (unsigned int)(xtime % 1000000) * 1000L;
1442d60b848STomohiro Kusumi }
1452d60b848STomohiro Kusumi 
1462d60b848STomohiro Kusumi uint64_t
hammer2_timespec_to_time(const struct timespec * ts)1472d60b848STomohiro Kusumi hammer2_timespec_to_time(const struct timespec *ts)
1482d60b848STomohiro Kusumi {
1492d60b848STomohiro Kusumi 	uint64_t xtime;
1502d60b848STomohiro Kusumi 
1512d60b848STomohiro Kusumi 	xtime = (unsigned)(ts->tv_nsec / 1000) +
1522d60b848STomohiro Kusumi 		(unsigned long)ts->tv_sec * 1000000ULL;
1532d60b848STomohiro Kusumi 	return(xtime);
1542d60b848STomohiro Kusumi }
1552d60b848STomohiro Kusumi 
1562d60b848STomohiro Kusumi /*
157*6da97f51STomohiro Kusumi  * Convert a hammer2 64-bit time to a timeval.
158*6da97f51STomohiro Kusumi  */
159*6da97f51STomohiro Kusumi void
hammer2_time_to_timeval(uint64_t xtime,struct timeval * tv)160*6da97f51STomohiro Kusumi hammer2_time_to_timeval(uint64_t xtime, struct timeval *tv)
161*6da97f51STomohiro Kusumi {
162*6da97f51STomohiro Kusumi 	tv->tv_sec = (unsigned long)(xtime / 1000000);
163*6da97f51STomohiro Kusumi 	tv->tv_usec = (unsigned int)(xtime % 1000000);
164*6da97f51STomohiro Kusumi }
165*6da97f51STomohiro Kusumi 
166*6da97f51STomohiro Kusumi /*
1672d60b848STomohiro Kusumi  * Convert a uuid to a unix uid or gid
1682d60b848STomohiro Kusumi  */
1692d60b848STomohiro Kusumi uint32_t
hammer2_to_unix_xid(const uuid_t * uuid)1702d60b848STomohiro Kusumi hammer2_to_unix_xid(const uuid_t *uuid)
1712d60b848STomohiro Kusumi {
1722d60b848STomohiro Kusumi 	return(*(const uint32_t *)&uuid->node[2]);
1732d60b848STomohiro Kusumi }
1742d60b848STomohiro Kusumi 
1752d60b848STomohiro Kusumi void
hammer2_guid_to_uuid(uuid_t * uuid,uint32_t guid)1762d60b848STomohiro Kusumi hammer2_guid_to_uuid(uuid_t *uuid, uint32_t guid)
1772d60b848STomohiro Kusumi {
1782d60b848STomohiro Kusumi 	bzero(uuid, sizeof(*uuid));
1792d60b848STomohiro Kusumi 	*(uint32_t *)&uuid->node[2] = guid;
1802d60b848STomohiro Kusumi }
1812d60b848STomohiro Kusumi 
1822d60b848STomohiro Kusumi /*
1832d60b848STomohiro Kusumi  * Borrow HAMMER1's directory hash algorithm #1 with a few modifications.
1842d60b848STomohiro Kusumi  * The filename is split into fields which are hashed separately and then
1852d60b848STomohiro Kusumi  * added together.
1862d60b848STomohiro Kusumi  *
1872d60b848STomohiro Kusumi  * Differences include: bit 63 must be set to 1 for HAMMER2 (HAMMER1 sets
1882d60b848STomohiro Kusumi  * it to 0), this is because bit63=0 is used for hidden hardlinked inodes.
1892d60b848STomohiro Kusumi  * (This means we do not need to do a 0-check/or-with-0x100000000 either).
1902d60b848STomohiro Kusumi  *
1912d60b848STomohiro Kusumi  * Also, the iscsi crc code is used instead of the old crc32 code.
1922d60b848STomohiro Kusumi  */
1932d60b848STomohiro Kusumi hammer2_key_t
hammer2_dirhash(const char * aname,size_t len)194e49cd535STomohiro Kusumi hammer2_dirhash(const char *aname, size_t len)
1952d60b848STomohiro Kusumi {
1962d60b848STomohiro Kusumi 	uint32_t crcx;
1972d60b848STomohiro Kusumi 	uint64_t key;
1982d60b848STomohiro Kusumi 	size_t i;
1992d60b848STomohiro Kusumi 	size_t j;
2002d60b848STomohiro Kusumi 
2012d60b848STomohiro Kusumi 	key = 0;
2022d60b848STomohiro Kusumi 
2032d60b848STomohiro Kusumi 	/*
2042d60b848STomohiro Kusumi 	 * m32
2052d60b848STomohiro Kusumi 	 */
2062d60b848STomohiro Kusumi 	crcx = 0;
2072d60b848STomohiro Kusumi 	for (i = j = 0; i < len; ++i) {
2082d60b848STomohiro Kusumi 		if (aname[i] == '.' ||
2092d60b848STomohiro Kusumi 		    aname[i] == '-' ||
2102d60b848STomohiro Kusumi 		    aname[i] == '_' ||
2112d60b848STomohiro Kusumi 		    aname[i] == '~') {
2122d60b848STomohiro Kusumi 			if (i != j)
2132d60b848STomohiro Kusumi 				crcx += hammer2_icrc32(aname + j, i - j);
2142d60b848STomohiro Kusumi 			j = i + 1;
2152d60b848STomohiro Kusumi 		}
2162d60b848STomohiro Kusumi 	}
2172d60b848STomohiro Kusumi 	if (i != j)
2182d60b848STomohiro Kusumi 		crcx += hammer2_icrc32(aname + j, i - j);
2192d60b848STomohiro Kusumi 
2202d60b848STomohiro Kusumi 	/*
2212d60b848STomohiro Kusumi 	 * The directory hash utilizes the top 32 bits of the 64-bit key.
2222d60b848STomohiro Kusumi 	 * Bit 63 must be set to 1.
2232d60b848STomohiro Kusumi 	 */
2242d60b848STomohiro Kusumi 	crcx |= 0x80000000U;
2252d60b848STomohiro Kusumi 	key |= (uint64_t)crcx << 32;
2262d60b848STomohiro Kusumi 
2272d60b848STomohiro Kusumi 	/*
2282d60b848STomohiro Kusumi 	 * l16 - crc of entire filename
2292d60b848STomohiro Kusumi 	 *
23000e4ae5cSTomohiro Kusumi 	 * This crc reduces degenerate hash collision conditions.
2312d60b848STomohiro Kusumi 	 */
2322d60b848STomohiro Kusumi 	crcx = hammer2_icrc32(aname, len);
2332d60b848STomohiro Kusumi 	crcx = crcx ^ (crcx << 16);
2342d60b848STomohiro Kusumi 	key |= crcx & 0xFFFF0000U;
2352d60b848STomohiro Kusumi 
2362d60b848STomohiro Kusumi 	/*
2372d60b848STomohiro Kusumi 	 * Set bit 15.  This allows readdir to strip bit 63 so a positive
2382d60b848STomohiro Kusumi 	 * 64-bit cookie/offset can always be returned, and still guarantee
2392d60b848STomohiro Kusumi 	 * that the values 0x0000-0x7FFF are available for artificial entries.
2402d60b848STomohiro Kusumi 	 * ('.' and '..').
2412d60b848STomohiro Kusumi 	 */
2422d60b848STomohiro Kusumi 	key |= 0x8000U;
2432d60b848STomohiro Kusumi 
2442d60b848STomohiro Kusumi 	return (key);
2452d60b848STomohiro Kusumi }
2462d60b848STomohiro Kusumi 
2472d60b848STomohiro Kusumi /*
2482d60b848STomohiro Kusumi  * Convert bytes to radix with no limitations.
2492d60b848STomohiro Kusumi  *
2502d60b848STomohiro Kusumi  * 0 bytes is special-cased to a radix of zero (which would normally
2512d60b848STomohiro Kusumi  * translate to (1 << 0) == 1).
2522d60b848STomohiro Kusumi  */
2532d60b848STomohiro Kusumi int
hammer2_getradix(size_t bytes)2542d60b848STomohiro Kusumi hammer2_getradix(size_t bytes)
2552d60b848STomohiro Kusumi {
2562d60b848STomohiro Kusumi 	int radix;
2572d60b848STomohiro Kusumi 
2582d60b848STomohiro Kusumi 	/*
25901a24a93STomohiro Kusumi 	 * Optimize the iteration by pre-checking commonly used radixes.
2602d60b848STomohiro Kusumi 	 */
2612d60b848STomohiro Kusumi 	if (bytes == HAMMER2_PBUFSIZE)
2622d60b848STomohiro Kusumi 		radix = HAMMER2_PBUFRADIX;
2632d60b848STomohiro Kusumi 	else if (bytes >= HAMMER2_LBUFSIZE)
2642d60b848STomohiro Kusumi 		radix = HAMMER2_LBUFRADIX;
2652d60b848STomohiro Kusumi 	else if (bytes >= HAMMER2_ALLOC_MIN)	/* clamp */
2662d60b848STomohiro Kusumi 		radix = HAMMER2_RADIX_MIN;
2672d60b848STomohiro Kusumi 	else
2682d60b848STomohiro Kusumi 		radix = 0;
2692d60b848STomohiro Kusumi 
2702d60b848STomohiro Kusumi 	/*
2712d60b848STomohiro Kusumi 	 * Iterate as needed.  Note that bytes == 0 is expected to return
2722d60b848STomohiro Kusumi 	 * a radix of 0 as a special case.
2732d60b848STomohiro Kusumi 	 */
2742d60b848STomohiro Kusumi 	while (((size_t)1 << radix) < bytes)
2752d60b848STomohiro Kusumi 		++radix;
2762d60b848STomohiro Kusumi 	return (radix);
2772d60b848STomohiro Kusumi }
2782d60b848STomohiro Kusumi 
2792d60b848STomohiro Kusumi /*
2802d60b848STomohiro Kusumi  * The logical block size is currently always PBUFSIZE.
2812d60b848STomohiro Kusumi  */
2822d60b848STomohiro Kusumi int
hammer2_calc_logical(hammer2_inode_t * ip,hammer2_off_t uoff,hammer2_key_t * lbasep,hammer2_key_t * leofp)2832d60b848STomohiro Kusumi hammer2_calc_logical(hammer2_inode_t *ip, hammer2_off_t uoff,
2842d60b848STomohiro Kusumi 		     hammer2_key_t *lbasep, hammer2_key_t *leofp)
2852d60b848STomohiro Kusumi {
2862d60b848STomohiro Kusumi 	if (lbasep)
2872d60b848STomohiro Kusumi 		*lbasep = uoff & ~HAMMER2_PBUFMASK64;
2882d60b848STomohiro Kusumi 	if (leofp) {
2892d60b848STomohiro Kusumi 		*leofp = (ip->meta.size + HAMMER2_PBUFMASK64) &
2902d60b848STomohiro Kusumi 			 ~HAMMER2_PBUFMASK64;
2912d60b848STomohiro Kusumi 	}
2922d60b848STomohiro Kusumi 	return (HAMMER2_PBUFSIZE);
2932d60b848STomohiro Kusumi }
2942d60b848STomohiro Kusumi 
2952d60b848STomohiro Kusumi /*
2962d60b848STomohiro Kusumi  * Calculate the physical block size.  pblksize <= lblksize.  Primarily
2972d60b848STomohiro Kusumi  * used to calculate a smaller physical block for the logical block
2982d60b848STomohiro Kusumi  * containing the file EOF.
2992d60b848STomohiro Kusumi  *
3002d60b848STomohiro Kusumi  * Returns 0 if the requested base offset is beyond the file EOF.
3012d60b848STomohiro Kusumi  */
3022d60b848STomohiro Kusumi int
hammer2_calc_physical(hammer2_inode_t * ip,hammer2_key_t lbase)3032d60b848STomohiro Kusumi hammer2_calc_physical(hammer2_inode_t *ip, hammer2_key_t lbase)
3042d60b848STomohiro Kusumi {
3052d60b848STomohiro Kusumi 	int lblksize;
3062d60b848STomohiro Kusumi 	int pblksize;
3072d60b848STomohiro Kusumi 	int eofbytes;
3082d60b848STomohiro Kusumi 
3092d60b848STomohiro Kusumi 	lblksize = hammer2_calc_logical(ip, lbase, NULL, NULL);
3102d60b848STomohiro Kusumi 	if (lbase + lblksize <= ip->meta.size)
3112d60b848STomohiro Kusumi 		return (lblksize);
3122d60b848STomohiro Kusumi 	if (lbase >= ip->meta.size)
3132d60b848STomohiro Kusumi 		return (0);
3142d60b848STomohiro Kusumi 	eofbytes = (int)(ip->meta.size - lbase);
3152d60b848STomohiro Kusumi 	pblksize = lblksize;
3162d60b848STomohiro Kusumi 	while (pblksize >= eofbytes && pblksize >= HAMMER2_ALLOC_MIN)
3172d60b848STomohiro Kusumi 		pblksize >>= 1;
3182d60b848STomohiro Kusumi 	pblksize <<= 1;
3192d60b848STomohiro Kusumi 
3202d60b848STomohiro Kusumi 	return (pblksize);
3212d60b848STomohiro Kusumi }
3222d60b848STomohiro Kusumi 
323aa86710aSTomohiro Kusumi extern fsnode *hammer2_curnode;
324aa86710aSTomohiro Kusumi 
3253999233bSTomohiro Kusumi static void
hammer2_get_curtime(uint64_t * timep)3263999233bSTomohiro Kusumi hammer2_get_curtime(uint64_t *timep)
3273999233bSTomohiro Kusumi {
3283999233bSTomohiro Kusumi 	struct timeval ts;
3293999233bSTomohiro Kusumi 	int error;
3303999233bSTomohiro Kusumi 
3313999233bSTomohiro Kusumi 	error = gettimeofday(&ts, NULL);
3323999233bSTomohiro Kusumi 	KKASSERT(error == 0);
3333999233bSTomohiro Kusumi 	*timep = (unsigned long)ts.tv_sec * 1000000 + ts.tv_usec;
3343999233bSTomohiro Kusumi }
3353999233bSTomohiro Kusumi 
3362d60b848STomohiro Kusumi void
hammer2_update_time(uint64_t * timep,bool is_mtime)337ddd1d3d1STomohiro Kusumi hammer2_update_time(uint64_t *timep, bool is_mtime)
3382d60b848STomohiro Kusumi {
339ddd1d3d1STomohiro Kusumi 	struct timespec *ts;
340ddd1d3d1STomohiro Kusumi 	struct stat *st;
3412d60b848STomohiro Kusumi 
3423999233bSTomohiro Kusumi 	/* HAMMER2 ioctl commands */
3433999233bSTomohiro Kusumi 	if (hammer2_curnode == NULL) {
3443999233bSTomohiro Kusumi 		hammer2_get_curtime(timep);
3453999233bSTomohiro Kusumi 		return;
3463999233bSTomohiro Kusumi 	}
3473999233bSTomohiro Kusumi 
348ddd1d3d1STomohiro Kusumi 	st = stampst.st_ino != 0 ? &stampst : &hammer2_curnode->inode->st;
349ddd1d3d1STomohiro Kusumi 	ts = is_mtime ? &st->st_mtim : &st->st_ctim;
350ddd1d3d1STomohiro Kusumi 
351ddd1d3d1STomohiro Kusumi 	*timep = (uint64_t)ts->tv_sec * 1000000 + ts->tv_nsec / 1000;
3522d60b848STomohiro Kusumi }
3532d60b848STomohiro Kusumi 
3542d60b848STomohiro Kusumi void
hammer2_adjreadcounter(int btype,size_t bytes)3552d60b848STomohiro Kusumi hammer2_adjreadcounter(int btype, size_t bytes)
3562d60b848STomohiro Kusumi {
3572d60b848STomohiro Kusumi 	long *counterp;
3582d60b848STomohiro Kusumi 
3592d60b848STomohiro Kusumi 	switch(btype) {
3602d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_DATA:
3612d60b848STomohiro Kusumi 		counterp = &hammer2_iod_file_read;
3622d60b848STomohiro Kusumi 		break;
3632d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_DIRENT:
3642d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_INODE:
3652d60b848STomohiro Kusumi 		counterp = &hammer2_iod_meta_read;
3662d60b848STomohiro Kusumi 		break;
3672d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_INDIRECT:
3682d60b848STomohiro Kusumi 		counterp = &hammer2_iod_indr_read;
3692d60b848STomohiro Kusumi 		break;
3702d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_FREEMAP_NODE:
3712d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_FREEMAP_LEAF:
3722d60b848STomohiro Kusumi 		counterp = &hammer2_iod_fmap_read;
3732d60b848STomohiro Kusumi 		break;
3742d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_FREEMAP:
3752d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_VOLUME:
3762d60b848STomohiro Kusumi 		counterp = &hammer2_iod_volu_read;
3772d60b848STomohiro Kusumi 		break;
3782d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_EMPTY:
3792d60b848STomohiro Kusumi 	default:
3802d60b848STomohiro Kusumi 		return;
3812d60b848STomohiro Kusumi 	}
3822d60b848STomohiro Kusumi 	*counterp += bytes;
3832d60b848STomohiro Kusumi }
3842d60b848STomohiro Kusumi 
3852d60b848STomohiro Kusumi void
hammer2_adjwritecounter(int btype,size_t bytes)3862d60b848STomohiro Kusumi hammer2_adjwritecounter(int btype, size_t bytes)
3872d60b848STomohiro Kusumi {
3882d60b848STomohiro Kusumi 	long *counterp;
3892d60b848STomohiro Kusumi 
3902d60b848STomohiro Kusumi 	switch(btype) {
3912d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_DATA:
3922d60b848STomohiro Kusumi 		counterp = &hammer2_iod_file_write;
3932d60b848STomohiro Kusumi 		break;
3942d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_DIRENT:
3952d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_INODE:
3962d60b848STomohiro Kusumi 		counterp = &hammer2_iod_meta_write;
3972d60b848STomohiro Kusumi 		break;
3982d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_INDIRECT:
3992d60b848STomohiro Kusumi 		counterp = &hammer2_iod_indr_write;
4002d60b848STomohiro Kusumi 		break;
4012d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_FREEMAP_NODE:
4022d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_FREEMAP_LEAF:
4032d60b848STomohiro Kusumi 		counterp = &hammer2_iod_fmap_write;
4042d60b848STomohiro Kusumi 		break;
4052d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_FREEMAP:
4062d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_VOLUME:
4072d60b848STomohiro Kusumi 		counterp = &hammer2_iod_volu_write;
4082d60b848STomohiro Kusumi 		break;
4092d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_EMPTY:
4102d60b848STomohiro Kusumi 	default:
4112d60b848STomohiro Kusumi 		return;
4122d60b848STomohiro Kusumi 	}
4132d60b848STomohiro Kusumi 	*counterp += bytes;
4142d60b848STomohiro Kusumi }
4152d60b848STomohiro Kusumi 
4162d60b848STomohiro Kusumi /*
4172d60b848STomohiro Kusumi  * Check for pending signal to allow interruption.  This function will
4182d60b848STomohiro Kusumi  * return immediately if the calling thread is a kernel thread and not
4192d60b848STomohiro Kusumi  * a user thread.
4202d60b848STomohiro Kusumi  */
4212d60b848STomohiro Kusumi int
hammer2_signal_check(time_t * timep)4222d60b848STomohiro Kusumi hammer2_signal_check(time_t *timep)
4232d60b848STomohiro Kusumi {
424a63188c8STomohiro Kusumi #if 0
4252d60b848STomohiro Kusumi 	thread_t td = curthread;
4262d60b848STomohiro Kusumi 	int error = 0;
4272d60b848STomohiro Kusumi 
4282d60b848STomohiro Kusumi 	if (td->td_lwp) {
4292d60b848STomohiro Kusumi 		lwkt_user_yield();
4302d60b848STomohiro Kusumi 		if (*timep != time_second) {
4312d60b848STomohiro Kusumi 			*timep = time_second;
4322d60b848STomohiro Kusumi 			if (CURSIG_NOBLOCK(curthread->td_lwp) != 0)
4332d60b848STomohiro Kusumi 				error = HAMMER2_ERROR_ABORTED;
4342d60b848STomohiro Kusumi 		}
4352d60b848STomohiro Kusumi 	} else {
4362d60b848STomohiro Kusumi 		lwkt_yield();
4372d60b848STomohiro Kusumi 	}
4382d60b848STomohiro Kusumi 	return error;
4392d60b848STomohiro Kusumi #endif
440a63188c8STomohiro Kusumi 	return 0;
441a63188c8STomohiro Kusumi }
4422d60b848STomohiro Kusumi 
4432d60b848STomohiro Kusumi const char *
hammer2_error_str(int error)4442d60b848STomohiro Kusumi hammer2_error_str(int error)
4452d60b848STomohiro Kusumi {
4462d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_EIO)
4472d60b848STomohiro Kusumi 		return("I/O Error");
4482d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_CHECK)
4492d60b848STomohiro Kusumi 		return("Check Error");
4502d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_INCOMPLETE)
4512d60b848STomohiro Kusumi 		return("Cluster Quorum Error");
4522d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_DEPTH)
4532d60b848STomohiro Kusumi 		return("Chain Depth Error");
4542d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_BADBREF)
4552d60b848STomohiro Kusumi 		return("Bad Blockref Error");
4562d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_ENOSPC)
4572d60b848STomohiro Kusumi 		return("No Space on Device");
4582d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_ENOENT)
4592d60b848STomohiro Kusumi 		return("Entry Not Found");
4602d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_ENOTEMPTY)
4612d60b848STomohiro Kusumi 		return("Directory Not Empty");
4622d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_EAGAIN)
4632d60b848STomohiro Kusumi 		return("EAGAIN");
4642d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_ENOTDIR)
4652d60b848STomohiro Kusumi 		return("Not a Directory");
4662d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_EISDIR)
4672d60b848STomohiro Kusumi 		return("Is a Directory");
4682d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_EINPROGRESS)
4692d60b848STomohiro Kusumi 		return("Operation in Progress");
4702d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_ABORTED)
4712d60b848STomohiro Kusumi 		return("Operation Aborted");
4722d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_EOF)
4732d60b848STomohiro Kusumi 		return("Operation Complete");
4742d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_EINVAL)
4752d60b848STomohiro Kusumi 		return("Invalid Operation");
4762d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_EEXIST)
4772d60b848STomohiro Kusumi 		return("Object Exists");
4782d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_EDEADLK)
4792d60b848STomohiro Kusumi 		return("Deadlock Detected");
4802d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_ESRCH)
4812d60b848STomohiro Kusumi 		return("Object Not Found");
4822d60b848STomohiro Kusumi 	if (error & HAMMER2_ERROR_ETIMEDOUT)
4832d60b848STomohiro Kusumi 		return("Timeout");
4842d60b848STomohiro Kusumi 	return("Unknown Error");
4852d60b848STomohiro Kusumi }
4862d60b848STomohiro Kusumi 
4872d60b848STomohiro Kusumi const char *
hammer2_bref_type_str(int btype)4882d60b848STomohiro Kusumi hammer2_bref_type_str(int btype)
4892d60b848STomohiro Kusumi {
4902d60b848STomohiro Kusumi 	switch(btype) {
4912d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_EMPTY:
4922d60b848STomohiro Kusumi 		return("empty");
4932d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_INODE:
4942d60b848STomohiro Kusumi 		return("inode");
4952d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_INDIRECT:
4962d60b848STomohiro Kusumi 		return("indirect");
4972d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_DATA:
4982d60b848STomohiro Kusumi 		return("data");
4992d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_DIRENT:
5002d60b848STomohiro Kusumi 		return("dirent");
5012d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_FREEMAP_NODE:
5022d60b848STomohiro Kusumi 		return("freemap_node");
5032d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_FREEMAP_LEAF:
5042d60b848STomohiro Kusumi 		return("freemap_leaf");
5052d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_INVALID:
5062d60b848STomohiro Kusumi 		return("invalid");
5072d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_FREEMAP:
5082d60b848STomohiro Kusumi 		return("freemap");
5092d60b848STomohiro Kusumi 	case HAMMER2_BREF_TYPE_VOLUME:
5102d60b848STomohiro Kusumi 		return("volume");
5112d60b848STomohiro Kusumi 	default:
5122d60b848STomohiro Kusumi 		return("unknown");
5132d60b848STomohiro Kusumi 	}
5142d60b848STomohiro Kusumi }
515