/* $NetBSD: udf.h,v 1.44 2011/09/27 01:13:16 christos Exp $ */ /* * Copyright (c) 2006, 2008 Reinoud Zandijk * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * */ #ifndef _FS_UDF_UDF_H_ #define _FS_UDF_UDF_H_ #include #include #include #include #include "udf_osta.h" #include "ecma167-udf.h" #include #include #include #include #include /* debug section */ extern int udf_verbose; /* undefine UDF_COMPLETE_DELETE to need `purge'; but purge is not implemented */ #define UDF_COMPLETE_DELETE /* debug categories */ #define UDF_DEBUG_VOLUMES 0x0000001 #define UDF_DEBUG_LOCKING 0x0000002 #define UDF_DEBUG_NODE 0x0000004 #define UDF_DEBUG_LOOKUP 0x0000008 #define UDF_DEBUG_READDIR 0x0000010 #define UDF_DEBUG_FIDS 0x0000020 #define UDF_DEBUG_DESCRIPTOR 0x0000040 #define UDF_DEBUG_TRANSLATE 0x0000080 #define UDF_DEBUG_STRATEGY 0x0000100 #define UDF_DEBUG_READ 0x0000200 #define UDF_DEBUG_WRITE 0x0000400 #define UDF_DEBUG_CALL 0x0000800 #define UDF_DEBUG_ATTR 0x0001000 #define UDF_DEBUG_EXTATTR 0x0002000 #define UDF_DEBUG_ALLOC 0x0004000 #define UDF_DEBUG_ADWLK 0x0008000 #define UDF_DEBUG_DIRHASH 0x0010000 #define UDF_DEBUG_NOTIMPL 0x0020000 #define UDF_DEBUG_SHEDULE 0x0040000 #define UDF_DEBUG_ECCLINE 0x0080000 #define UDF_DEBUG_SYNC 0x0100000 #define UDF_DEBUG_PARANOIA 0x0200000 #define UDF_DEBUG_PARANOIDADWLK 0x0400000 #define UDF_DEBUG_NODEDUMP 0x0800000 #define UDF_DEBUG_RESERVE 0x1000000 /* initial value of udf_verbose */ #define UDF_DEBUGGING 0 #ifdef DEBUG #define DPRINTF(name, arg) { \ if (udf_verbose & UDF_DEBUG_##name) {\ printf arg;\ };\ } #define DPRINTFIF(name, cond, arg) { \ if (udf_verbose & UDF_DEBUG_##name) { \ if (cond) printf arg;\ };\ } #else #define DPRINTF(name, arg) {} #define DPRINTFIF(name, cond, arg) {} #endif /* constants to identify what kind of identifier we are dealing with */ #define UDF_REGID_DOMAIN 1 #define UDF_REGID_UDF 2 #define UDF_REGID_IMPLEMENTATION 3 #define UDF_REGID_APPLICATION 4 #define UDF_REGID_NAME 99 /* DON'T change these: they identify 13thmonkey's UDF implementation */ #define APP_NAME "*NetBSD UDF" #define APP_VERSION_MAIN 0 #define APP_VERSION_SUB 5 #define IMPL_NAME "*NetBSD kernel UDF" /* Configuration values */ #define UDF_INODE_HASHBITS 10 #define UDF_INODE_HASHSIZE (1< are normal */ #define UDF_VTOP_TYPE_RAW 0 #define UDF_VTOP_TYPE_UNKNOWN 0 #define UDF_VTOP_TYPE_PHYS 1 #define UDF_VTOP_TYPE_VIRT 2 #define UDF_VTOP_TYPE_SPARABLE 3 #define UDF_VTOP_TYPE_META 4 /* allocation strategies */ #define UDF_ALLOC_INVALID 0 #define UDF_ALLOC_SEQUENTIAL 1 /* linear on NWA */ #define UDF_ALLOC_VAT 2 /* VAT handling */ #define UDF_ALLOC_SPACEMAP 3 /* spacemaps */ #define UDF_ALLOC_METABITMAP 4 /* metadata bitmap */ #define UDF_ALLOC_METASEQUENTIAL 5 /* in chunks seq., nodes not seq */ #define UDF_ALLOC_RELAXEDSEQUENTIAL 6 /* only nodes not seq. */ /* logical volume open/close actions */ #define UDF_OPEN_SESSION 0x01 /* if needed writeout VRS + VDS */ #define UDF_CLOSE_SESSION 0x02 /* close session after writing VAT */ #define UDF_FINALISE_DISC 0x04 /* close session after writing VAT */ #define UDF_WRITE_VAT 0x08 /* sequential VAT filesystem */ #define UDF_WRITE_LVINT 0x10 /* write out open lvint */ #define UDF_WRITE_PART_BITMAPS 0x20 /* write out partition space bitmaps */ #define UDF_APPENDONLY_LVINT 0x40 /* no shifting, only appending */ #define UDF_WRITE_METAPART_NODES 0x80 /* write out metadata partition nodes*/ #define UDFLOGVOL_BITS "\20\1OPEN_SESSION\2CLOSE_SESSION\3FINALISE_DISC" \ "\4WRITE_VAT\5WRITE_LVINT\6WRITE_PART_BITMAPS" \ "\7APPENDONLY_LVINT\10WRITE_METAPART_NODES" /* logical volume error handling actions */ #define UDF_UPDATE_TRACKINFO 0x01 /* update trackinfo and re-shedule */ #define UDF_REMAP_BLOCK 0x02 /* remap the failing block length */ #define UDFONERROR_BITS "\20\1UPDATE_TRACKINFO\2REMAP_BLOCK" /* readdir cookies */ #define UDF_DIRCOOKIE_DOT 1 /* malloc pools */ MALLOC_DECLARE(M_UDFMNT); MALLOC_DECLARE(M_UDFVOLD); MALLOC_DECLARE(M_UDFTEMP); struct pool udf_node_pool; struct udf_node; struct udf_strategy; struct udf_lvintq { uint32_t start; uint32_t end; uint32_t pos; uint32_t wpos; }; struct udf_bitmap { uint8_t *blob; /* allocated */ uint8_t *bits; /* bits themselves */ uint8_t *pages; /* dirty pages */ uint32_t max_offset; /* in bits */ uint32_t data_pos; /* position in data */ uint32_t metadata_pos; /* .. in metadata */ }; struct udf_strat_args { struct udf_mount *ump; struct udf_node *udf_node; struct long_ad *icb; union dscrptr *dscr; struct buf *nestbuf; kauth_cred_t cred; int waitfor; }; struct udf_strategy { int (*create_logvol_dscr) (struct udf_strat_args *args); void (*free_logvol_dscr) (struct udf_strat_args *args); int (*read_logvol_dscr) (struct udf_strat_args *args); int (*write_logvol_dscr) (struct udf_strat_args *args); void (*queuebuf) (struct udf_strat_args *args); void (*discstrat_init) (struct udf_strat_args *args); void (*discstrat_finish) (struct udf_strat_args *args); }; extern struct udf_strategy udf_strat_bootstrap; extern struct udf_strategy udf_strat_sequential; extern struct udf_strategy udf_strat_direct; extern struct udf_strategy udf_strat_rmw; /* pre cleanup */ struct udf_mount { struct mount *vfs_mountp; struct vnode *devvp; struct mmc_discinfo discinfo; struct udf_args mount_args; /* format descriptors */ kmutex_t logvol_mutex; struct anchor_vdp *anchors[UDF_ANCHORS]; /* anchors to VDS */ struct pri_vol_desc *primary_vol; /* identification */ struct logvol_desc *logical_vol; /* main mapping v->p */ struct unalloc_sp_desc *unallocated; /* free UDF space */ struct impvol_desc *implementation; /* likely reduntant */ struct logvol_int_desc *logvol_integrity; /* current integrity */ struct part_desc *partitions[UDF_PARTITIONS]; /* partitions */ /* logvol_info is derived; points *into* other structures */ struct udf_logvol_info *logvol_info; /* integrity descr. */ /* fileset and root directories */ struct fileset_desc *fileset_desc; /* normally one */ /* tracing logvol integrity history */ struct udf_lvintq lvint_trace[UDF_LVDINT_SEGMENTS]; int lvopen; /* logvol actions */ int lvclose; /* logvol actions */ /* logical to physical translations */ int vtop[UDF_PMAPS+1]; /* vpartnr trans */ int vtop_tp[UDF_PMAPS+1]; /* type of trans */ /* disc allocation / writing method */ kmutex_t allocate_mutex; int lvreadwrite; /* error handling */ int vtop_alloc[UDF_PMAPS+1]; /* alloc scheme */ int data_part; int node_part; int fids_part; /* sequential track info */ struct mmc_trackinfo data_track; struct mmc_trackinfo metadata_track; /* VAT */ uint32_t first_possible_vat_location; uint32_t last_possible_vat_location; uint32_t vat_entries; uint32_t vat_offset; /* offset in table */ uint32_t vat_last_free_lb; /* last free lb_num */ uint32_t vat_table_len; uint32_t vat_table_alloc_len; uint8_t *vat_table; uint8_t *vat_pages; /* TODO */ struct udf_node *vat_node; /* system node */ /* space bitmaps for physical partitions */ struct space_bitmap_desc*part_unalloc_dscr[UDF_PARTITIONS]; struct space_bitmap_desc*part_freed_dscr [UDF_PARTITIONS]; struct udf_bitmap part_unalloc_bits[UDF_PARTITIONS]; struct udf_bitmap part_freed_bits [UDF_PARTITIONS]; /* sparable */ uint32_t sparable_packet_size; uint32_t packet_size; struct udf_sparing_table*sparing_table; /* meta */ struct udf_node *metadata_node; /* system node */ struct udf_node *metadatamirror_node; /* system node */ struct udf_node *metadatabitmap_node; /* system node */ struct space_bitmap_desc*metadata_unalloc_dscr; struct udf_bitmap metadata_unalloc_bits; uint32_t metadata_alloc_unit_size; uint16_t metadata_alignment_unit_size; uint8_t metadata_flags; /* rb tree for lookup icb to udf_node and sorted list for sync */ kmutex_t ihash_lock; kmutex_t get_node_lock; struct rb_tree udf_node_tree; /* syncing */ int syncing; /* are we syncing? */ kcondvar_t dirtynodes_cv; /* sleeping on sync */ /* late allocation */ int32_t uncommitted_lbs[UDF_PARTITIONS]; struct long_ad *la_node_ad_cpy; /* issue buf */ uint64_t *la_lmapping, *la_pmapping; /* issue buf */ /* lists */ STAILQ_HEAD(udfmntpts, udf_mount) all_udf_mntpnts; /* device strategy */ struct udf_strategy *strategy; void *strategy_private; }; /* * UDF node describing a file/directory. * * BUGALERT claim node_mutex before reading/writing to prevent inconsistencies ! */ struct udf_node { struct genfs_node i_gnode; /* has to be first */ struct vnode *vnode; /* vnode associated */ struct udf_mount *ump; kmutex_t node_mutex; kcondvar_t node_lock; /* sleeping lock */ char const *lock_fname; int lock_lineno; /* rb_node for fast lookup and fast sequential visiting */ struct rb_node rbnode; /* one of `fe' or `efe' can be set, not both (UDF file entry dscr.) */ struct file_entry *fe; struct extfile_entry *efe; struct alloc_ext_entry *ext[UDF_MAX_ALLOC_EXTENTS]; int num_extensions; /* location found, recording location & hints */ struct long_ad loc; /* FID/hash loc. */ struct long_ad write_loc; /* strat 4096 loc */ int needs_indirect; /* has missing indr. */ struct long_ad ext_loc[UDF_MAX_ALLOC_EXTENTS]; struct dirhash *dir_hash; /* misc */ uint32_t i_flags; /* associated flags */ struct lockf *lockf; /* lock list */ uint32_t outstanding_bufs; /* file data */ uint32_t outstanding_nodedscr; /* node dscr */ int32_t uncommitted_lbs; /* in UBC */ /* references to associated nodes */ struct udf_node *extattr; struct udf_node *streamdir; struct udf_node *my_parent; /* if extended attr. */ }; /* misc. flags stored in i_flags (XXX needs cleaning up) */ #define IN_ACCESS 0x0001 /* Inode access time update request */ #define IN_CHANGE 0x0002 /* Inode change time update request */ #define IN_UPDATE 0x0004 /* Inode was written to; update mtime*/ #define IN_MODIFY 0x0008 /* Modification time update request */ #define IN_MODIFIED 0x0010 /* node has been modified */ #define IN_ACCESSED 0x0020 /* node has been accessed */ #define IN_RENAME 0x0040 /* node is being renamed. XXX ?? */ #define IN_DELETED 0x0080 /* node is unlinked, no FID reference*/ #define IN_LOCKED 0x0100 /* node is locked by condvar */ #define IN_SYNCED 0x0200 /* node is being used by sync */ #define IN_CALLBACK_ULK 0x0400 /* node will be unlocked by callback */ #define IN_NODE_REBUILD 0x0800 /* node is rebuild */ #define IN_FLAGBITS \ "\10\1IN_ACCESS\2IN_CHANGE\3IN_UPDATE\4IN_MODIFY\5IN_MODIFIED" \ "\6IN_ACCESSED\7IN_RENAME\10IN_DELETED\11IN_LOCKED\12IN_SYNCED" \ "\13IN_CALLBACK_ULK\14IN_NODE_REBUILD" #endif /* !_FS_UDF_UDF_H_ */