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