Basic mkfs.lafs works.
Signed-off-by: NeilBrown <neilb@suse.de>
--- /dev/null
+*.o
+*.a
+mkfs.lafs
+core
--- /dev/null
+ GNU GENERAL PUBLIC LICENSE
+ Version 2, June 1991
+
+ Copyright (C) 1989, 1991 Free Software Foundation, Inc.
+ 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+ Preamble
+
+ The licenses for most software are designed to take away your
+freedom to share and change it. By contrast, the GNU General Public
+License is intended to guarantee your freedom to share and change free
+software--to make sure the software is free for all its users. This
+General Public License applies to most of the Free Software
+Foundation's software and to any other program whose authors commit to
+using it. (Some other Free Software Foundation software is covered by
+the GNU Library General Public License instead.) You can apply it to
+your programs, too.
+
+ When we speak of free software, we are referring to freedom, not
+price. Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+this service if you wish), that you receive source code or can get it
+if you want it, that you can change the software or use pieces of it
+in new free programs; and that you know you can do these things.
+
+ To protect your rights, we need to make restrictions that forbid
+anyone to deny you these rights or to ask you to surrender the rights.
+These restrictions translate to certain responsibilities for you if you
+distribute copies of the software, or if you modify it.
+
+ For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must give the recipients all the rights that
+you have. You must make sure that they, too, receive or can get the
+source code. And you must show them these terms so they know their
+rights.
+
+ We protect your rights with two steps: (1) copyright the software, and
+(2) offer you this license which gives you legal permission to copy,
+distribute and/or modify the software.
+
+ Also, for each author's protection and ours, we want to make certain
+that everyone understands that there is no warranty for this free
+software. If the software is modified by someone else and passed on, we
+want its recipients to know that what they have is not the original, so
+that any problems introduced by others will not reflect on the original
+authors' reputations.
+
+ Finally, any free program is threatened constantly by software
+patents. We wish to avoid the danger that redistributors of a free
+program will individually obtain patent licenses, in effect making the
+program proprietary. To prevent this, we have made it clear that any
+patent must be licensed for everyone's free use or not licensed at all.
+
+ The precise terms and conditions for copying, distribution and
+modification follow.
+\f
+ GNU GENERAL PUBLIC LICENSE
+ TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
+
+ 0. This License applies to any program or other work which contains
+a notice placed by the copyright holder saying it may be distributed
+under the terms of this General Public License. The "Program", below,
+refers to any such program or work, and a "work based on the Program"
+means either the Program or any derivative work under copyright law:
+that is to say, a work containing the Program or a portion of it,
+either verbatim or with modifications and/or translated into another
+language. (Hereinafter, translation is included without limitation in
+the term "modification".) Each licensee is addressed as "you".
+
+Activities other than copying, distribution and modification are not
+covered by this License; they are outside its scope. The act of
+running the Program is not restricted, and the output from the Program
+is covered only if its contents constitute a work based on the
+Program (independent of having been made by running the Program).
+Whether that is true depends on what the Program does.
+
+ 1. You may copy and distribute verbatim copies of the Program's
+source code as you receive it, in any medium, provided that you
+conspicuously and appropriately publish on each copy an appropriate
+copyright notice and disclaimer of warranty; keep intact all the
+notices that refer to this License and to the absence of any warranty;
+and give any other recipients of the Program a copy of this License
+along with the Program.
+
+You may charge a fee for the physical act of transferring a copy, and
+you may at your option offer warranty protection in exchange for a fee.
+
+ 2. You may modify your copy or copies of the Program or any portion
+of it, thus forming a work based on the Program, and copy and
+distribute such modifications or work under the terms of Section 1
+above, provided that you also meet all of these conditions:
+
+ a) You must cause the modified files to carry prominent notices
+ stating that you changed the files and the date of any change.
+
+ b) You must cause any work that you distribute or publish, that in
+ whole or in part contains or is derived from the Program or any
+ part thereof, to be licensed as a whole at no charge to all third
+ parties under the terms of this License.
+
+ c) If the modified program normally reads commands interactively
+ when run, you must cause it, when started running for such
+ interactive use in the most ordinary way, to print or display an
+ announcement including an appropriate copyright notice and a
+ notice that there is no warranty (or else, saying that you provide
+ a warranty) and that users may redistribute the program under
+ these conditions, and telling the user how to view a copy of this
+ License. (Exception: if the Program itself is interactive but
+ does not normally print such an announcement, your work based on
+ the Program is not required to print an announcement.)
+\f
+These requirements apply to the modified work as a whole. If
+identifiable sections of that work are not derived from the Program,
+and can be reasonably considered independent and separate works in
+themselves, then this License, and its terms, do not apply to those
+sections when you distribute them as separate works. But when you
+distribute the same sections as part of a whole which is a work based
+on the Program, the distribution of the whole must be on the terms of
+this License, whose permissions for other licensees extend to the
+entire whole, and thus to each and every part regardless of who wrote it.
+
+Thus, it is not the intent of this section to claim rights or contest
+your rights to work written entirely by you; rather, the intent is to
+exercise the right to control the distribution of derivative or
+collective works based on the Program.
+
+In addition, mere aggregation of another work not based on the Program
+with the Program (or with a work based on the Program) on a volume of
+a storage or distribution medium does not bring the other work under
+the scope of this License.
+
+ 3. You may copy and distribute the Program (or a work based on it,
+under Section 2) in object code or executable form under the terms of
+Sections 1 and 2 above provided that you also do one of the following:
+
+ a) Accompany it with the complete corresponding machine-readable
+ source code, which must be distributed under the terms of Sections
+ 1 and 2 above on a medium customarily used for software interchange; or,
+
+ b) Accompany it with a written offer, valid for at least three
+ years, to give any third party, for a charge no more than your
+ cost of physically performing source distribution, a complete
+ machine-readable copy of the corresponding source code, to be
+ distributed under the terms of Sections 1 and 2 above on a medium
+ customarily used for software interchange; or,
+
+ c) Accompany it with the information you received as to the offer
+ to distribute corresponding source code. (This alternative is
+ allowed only for noncommercial distribution and only if you
+ received the program in object code or executable form with such
+ an offer, in accord with Subsection b above.)
+
+The source code for a work means the preferred form of the work for
+making modifications to it. For an executable work, complete source
+code means all the source code for all modules it contains, plus any
+associated interface definition files, plus the scripts used to
+control compilation and installation of the executable. However, as a
+special exception, the source code distributed need not include
+anything that is normally distributed (in either source or binary
+form) with the major components (compiler, kernel, and so on) of the
+operating system on which the executable runs, unless that component
+itself accompanies the executable.
+
+If distribution of executable or object code is made by offering
+access to copy from a designated place, then offering equivalent
+access to copy the source code from the same place counts as
+distribution of the source code, even though third parties are not
+compelled to copy the source along with the object code.
+\f
+ 4. You may not copy, modify, sublicense, or distribute the Program
+except as expressly provided under this License. Any attempt
+otherwise to copy, modify, sublicense or distribute the Program is
+void, and will automatically terminate your rights under this License.
+However, parties who have received copies, or rights, from you under
+this License will not have their licenses terminated so long as such
+parties remain in full compliance.
+
+ 5. You are not required to accept this License, since you have not
+signed it. However, nothing else grants you permission to modify or
+distribute the Program or its derivative works. These actions are
+prohibited by law if you do not accept this License. Therefore, by
+modifying or distributing the Program (or any work based on the
+Program), you indicate your acceptance of this License to do so, and
+all its terms and conditions for copying, distributing or modifying
+the Program or works based on it.
+
+ 6. Each time you redistribute the Program (or any work based on the
+Program), the recipient automatically receives a license from the
+original licensor to copy, distribute or modify the Program subject to
+these terms and conditions. You may not impose any further
+restrictions on the recipients' exercise of the rights granted herein.
+You are not responsible for enforcing compliance by third parties to
+this License.
+
+ 7. If, as a consequence of a court judgment or allegation of patent
+infringement or for any other reason (not limited to patent issues),
+conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License. If you cannot
+distribute so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you
+may not distribute the Program at all. For example, if a patent
+license would not permit royalty-free redistribution of the Program by
+all those who receive copies directly or indirectly through you, then
+the only way you could satisfy both it and this License would be to
+refrain entirely from distribution of the Program.
+
+If any portion of this section is held invalid or unenforceable under
+any particular circumstance, the balance of the section is intended to
+apply and the section as a whole is intended to apply in other
+circumstances.
+
+It is not the purpose of this section to induce you to infringe any
+patents or other property right claims or to contest validity of any
+such claims; this section has the sole purpose of protecting the
+integrity of the free software distribution system, which is
+implemented by public license practices. Many people have made
+generous contributions to the wide range of software distributed
+through that system in reliance on consistent application of that
+system; it is up to the author/donor to decide if he or she is willing
+to distribute software through any other system and a licensee cannot
+impose that choice.
+
+This section is intended to make thoroughly clear what is believed to
+be a consequence of the rest of this License.
+\f
+ 8. If the distribution and/or use of the Program is restricted in
+certain countries either by patents or by copyrighted interfaces, the
+original copyright holder who places the Program under this License
+may add an explicit geographical distribution limitation excluding
+those countries, so that distribution is permitted only in or among
+countries not thus excluded. In such case, this License incorporates
+the limitation as if written in the body of this License.
+
+ 9. The Free Software Foundation may publish revised and/or new versions
+of the General Public License from time to time. Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+Each version is given a distinguishing version number. If the Program
+specifies a version number of this License which applies to it and "any
+later version", you have the option of following the terms and conditions
+either of that version or of any later version published by the Free
+Software Foundation. If the Program does not specify a version number of
+this License, you may choose any version ever published by the Free Software
+Foundation.
+
+ 10. If you wish to incorporate parts of the Program into other free
+programs whose distribution conditions are different, write to the author
+to ask for permission. For software which is copyrighted by the Free
+Software Foundation, write to the Free Software Foundation; we sometimes
+make exceptions for this. Our decision will be guided by the two goals
+of preserving the free status of all derivatives of our free software and
+of promoting the sharing and reuse of software generally.
+
+ NO WARRANTY
+
+ 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
+FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
+OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
+PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
+OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
+MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
+TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
+PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
+REPAIR OR CORRECTION.
+
+ 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
+REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
+INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
+OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
+TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
+YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
+PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGES.
+
+ END OF TERMS AND CONDITIONS
+\f
+ How to Apply These Terms to Your New Programs
+
+ If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+ To do so, attach the following notices to the program. It is safest
+to attach them to the start of each source file to most effectively
+convey the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+ <one line to give the program's name and a brief idea of what it does.>
+ Copyright (C) 19yy <name of author>
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+
+
+Also add information on how to contact you by electronic and paper mail.
+
+If the program is interactive, make it output a short notice like this
+when it starts in an interactive mode:
+
+ Gnomovision version 69, Copyright (C) 19yy name of author
+ Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+ This is free software, and you are welcome to redistribute it
+ under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License. Of course, the commands you use may
+be called something other than `show w' and `show c'; they could even be
+mouse-clicks or menu items--whatever suits your program.
+
+You should also get your employer (if you work as a programmer) or your
+school, if any, to sign a "copyright disclaimer" for the program, if
+necessary. Here is a sample; alter the names:
+
+ Yoyodyne, Inc., hereby disclaims all copyright interest in the program
+ `Gnomovision' (which makes passes at compilers) written by James Hacker.
+
+ <signature of Ty Coon>, 1 April 1989
+ Ty Coon, President of Vice
+
+This General Public License does not permit incorporating your program into
+proprietary programs. If your program is a subroutine library, you may
+consider it more useful to permit linking proprietary applications with the
+library. If this is what you want to do, use the GNU Library General
+Public License instead of this License.
--- /dev/null
+
+all:
+ $(MAKE) -C lib/
+ $(MAKE) -C tools/
\ No newline at end of file
--- /dev/null
+#include <endian.h>
+#define bswap_16(x) (((x) & 0x00ffU) << 8 | \
+ ((x) & 0xff00U) >> 8)
+#define bswap_32(x) (((x) & 0x000000ffU) << 24 | \
+ ((x) & 0xff000000U) >> 24 | \
+ ((x) & 0x0000ff00U) << 8 | \
+ ((x) & 0x00ff0000U) >> 8)
+#define bswap_64(x) (((x) & 0x00000000000000ffULL) << 56 | \
+ ((x) & 0xff00000000000000ULL) >> 56 | \
+ ((x) & 0x000000000000ff00ULL) << 40 | \
+ ((x) & 0x00ff000000000000ULL) >> 40 | \
+ ((x) & 0x0000000000ff0000ULL) << 24 | \
+ ((x) & 0x0000ff0000000000ULL) >> 24 | \
+ ((x) & 0x00000000ff000000ULL) << 8 | \
+ ((x) & 0x000000ff00000000ULL) >> 8)
+
+#if BYTE_ORDER == LITTLE_ENDIAN
+#define __cpu_to_le16(_x) (_x)
+#define __cpu_to_le32(_x) (_x)
+#define __cpu_to_le64(_x) (_x)
+#define __le16_to_cpu(_x) (_x)
+#define __le32_to_cpu(_x) (_x)
+#define __le64_to_cpu(_x) (_x)
+
+#define __cpu_to_be16(_x) bswap_16(_x)
+#define __cpu_to_be32(_x) bswap_32(_x)
+#define __cpu_to_be64(_x) bswap_64(_x)
+#define __be16_to_cpu(_x) bswap_16(_x)
+#define __be32_to_cpu(_x) bswap_32(_x)
+#define __be64_to_cpu(_x) bswap_64(_x)
+#elif BYTE_ORDER == BIG_ENDIAN
+#define __cpu_to_le16(_x) bswap_16(_x)
+#define __cpu_to_le32(_x) bswap_32(_x)
+#define __cpu_to_le64(_x) bswap_64(_x)
+#define __le16_to_cpu(_x) bswap_16(_x)
+#define __le32_to_cpu(_x) bswap_32(_x)
+#define __le64_to_cpu(_x) bswap_64(_x)
+
+#define __cpu_to_be16(_x) (_x)
+#define __cpu_to_be32(_x) (_x)
+#define __cpu_to_be64(_x) (_x)
+#define __be16_to_cpu(_x) (_x)
+#define __be32_to_cpu(_x) (_x)
+#define __be64_to_cpu(_x) (_x)
+#else
+# error "unknown endianness."
+#endif
--- /dev/null
+
+#include <stdint.h>
+#include <stdlib.h>
+
+#define u8 uint8_t
+#define u16 uint16_t
+#define u32 uint32_t
+#define u64 uint64_t
+#include <lafs/layout.h>
+#include <lafs/struct.h>
+#include <lafs/endian.h>
+
+
+struct lafs *lafs_alloc(void);
+int lafs_new(struct lafs *, int block_bytes);
+
+struct lafs_device *lafs_add_device(struct lafs *, char *devname, int fd,
+ loff_t segblocks, loff_t strideblock,
+ int width, int usage_inum);
+
+struct lafs_ino *lafs_get_itable(struct lafs *);
+struct lafs_ino *lafs_add_inode(struct lafs_ino*, int inum, int type);
+int lafs_imap_set(struct lafs_ino *, int inum);
+
+int lafs_add_free_seg(struct lafs*, int dev, loff_t seg);
+
+void lafs_inode_init(struct lafs *, char *, int type);
+struct lafs_ino *lafs_import_inode_buf(struct lafs *fs,
+ char *buf, int inum,
+ struct lafs_ino *parent);
+void lafs_dirty_inode(struct lafs_ino *);
+void lafs_inode_fillblock(struct lafs_ino *ino, char *buf);
+void lafs_make_iblock(struct lafs_ino *ino);
+
+struct lafs_dblk *lafs_dblk(struct lafs_ino *ino,
+ loff_t bnum);
+int lafs_load_dblk(struct lafs_dblk *);
+int lafs_find_dblk(struct lafs_dblk *);
+struct lafs_ino *lafs_import_inode(struct lafs_dblk *db);
+
+int lafs_read_virtual(struct lafs *, char *, loff_t);
+int lafs_sched_blk(struct lafs_blk *);
+
+int lafs_write_dev(struct lafs_device *dev);
+int lafs_write_state(struct lafs *fs);
+int lafs_checkpoint(struct lafs *fs);
+void lafs_incorporate(struct lafs_iblk *ib);
+void lafs_cluster_allocate(struct lafs_blk *b, int cnum);
+void lafs_cluster_flush(struct lafs *fs, int cnum);
+int lafs_new_segment(struct lafs *, int cnum);
+struct lafs_device *lafs_dev_find(struct lafs *fs, loff_t virt);
+void lafs_allocated_block(struct lafs_blk *b, loff_t addr);
+void lafs_cluster_init(struct lafs *fs, int cnum,
+ loff_t addr, loff_t prev, loff_t seq);
+void lafs_summary_update(struct lafs_ino *ino,
+ loff_t oldaddr, loff_t newaddr,
+ int is_index);
+void lafs_segment_count(struct lafs *fs, loff_t addr, int diff);
+
+unsigned long crc32(
+ unsigned long crc,
+ const uint32_t *buf,
+ unsigned len);
+
+static inline struct lafs_dblk *dblk(struct lafs_blk *b)
+{
+ return container_of(b, struct lafs_dblk, b);
+}
+
+static inline struct lafs_iblk *iblk(struct lafs_blk *b)
+{
+ return container_of(b, struct lafs_iblk, b);
+}
+
+static inline int lafs_dirty_blk(struct lafs_blk *blk)
+{
+ blk->flags |= B_Dirty;
+ return lafs_sched_blk(blk);
+}
+
+static inline uint64_t lafs_encode_timeval(struct timeval *tm)
+{
+ uint64_t nano = tm->tv_usec * 1000;
+ uint64_t sec = tm->tv_sec & (0x7FFFFFFFFULL);
+ nano &= ~1ULL;
+ return sec | (nano << 34);
+}
+
+static inline struct lafs_device *dev_by_num(struct lafs *fs, int num)
+{
+ struct lafs_device *dv;
+ for (dv = fs->devs ; dv ; dv = dv->next)
+ if (dv->devnum == num)
+ return dv;
+ return NULL;
+}
+
+
+static inline void
+virttoseg(struct lafs *fs, loff_t virt, int *devp, loff_t *segp, loff_t *offsetp)
+{
+ struct lafs_device *dv = lafs_dev_find(fs, virt);
+
+ virt -= dv->start;
+ if (dv->segment_size >= dv->width * dv->stride) {
+ *offsetp = virt % dv->segment_size;
+ *segp = virt / dv->segment_size;
+ } else {
+ int of = virt % dv->stride;
+ int strp =virt / (dv->width * dv->stride);
+
+ *segp = (strp * dv->stride + of) /
+ (dv->segment_size / dv->width);
+ *offsetp = virt - dv->segment_stride * *segp;
+ }
+ *devp = dv->devnum;
+}
+
+static inline void
+virttophys(struct lafs *fs, loff_t virt, int *devp, loff_t *sectp)
+{
+ struct lafs_device *dv = lafs_dev_find(fs, virt);
+
+ if (dv == NULL)
+ return;
+ *devp = dv->devnum;
+
+ virt -= dv->start;
+ virt *= fs->blocksize;
+ virt += dv->segment_offset;
+ *sectp = virt;
+}
+
+#include <stdio.h>
+static inline void
+de_sched(struct lafs_blk *b)
+{
+ if (!(b->flags & B_Sched))
+ return;
+ b->flags &= ~B_Sched;
+
+ if (b->parent) {
+ struct lafs_iblk *p = b->parent;
+ printf("ds %d/%ld %d\n", p->b.ino->inum, p->b.fileaddr, p->sched_cnt);
+ p->sched_cnt--;
+ if (p->sched_cnt == 0)
+ lafs_sched_blk(&p->b);
+ }
+}
--- /dev/null
+/*
+ * fs/lafs/layout.h
+ * Copyright (C) 2005-2009
+ * Neil Brown <neilb@suse.de>
+ * Released under the GPL, version 2
+ *
+ * layout of device superblock
+ * and array state block
+ */
+
+/* All multibyte numerical values are in little-endian order
+ */
+
+/* The "superblock" describes a particular device in the filesystem.
+ * different devices have different superblocks.
+ */
+struct lafs_dev {
+ char idtag[16]; /* LaFS-DeviceBlock */
+ char version[16]; /* number space options */
+ u8 uuid[16];
+ u32 checksum;
+ u32 seq;
+
+ u64 ctime;
+ u64 start, size; /* in array address space (array block)*/
+ u64 devaddr[2]; /* (device byte) one at each "end" */
+ u64 stateaddr[4]; /* (device byte) 4 state blocks, two at each end */
+
+ u8 statebits; /* log of size of stateblock - normally 10-14 */
+ u8 blockbits; /* bits in fs block (byte)- 9 - 16 */
+ u16 width; /* devices in array - 1 to a few hundred */
+ u32 stride; /* Blocks in a stride - 1 to a very large number */
+ u32 segment_size; /* blocks in a segment (block) */
+ u32 segment_offset; /* offset of first segment (device byte) */
+ u32 segment_count;
+ u32 usage_inum; /* inum of segment usage file */
+ u32 level;
+} __attribute__((packed));
+#define LAFS_DEVBLK_SIZE 1024
+
+struct lafs_state {
+ char idtag[16]; /* LaFS-State-Block */
+ char version[16]; /* number space options */
+ u32 checksum;
+ u32 seq;
+ u8 uuid[16];
+ u32 levels;
+ u32 devices;
+ u32 nonlog_segment; /* segment number and */
+ u16 nonlog_dev; /* device number of active non-logged segment */
+ u16 nonlog_offset; /* offset into above segment of next non-logged
+ * block to allocate
+ */
+ u32 maxsnapshot;
+ u16 nextyouth;
+ u16 pad0;
+
+ u64 checkpointcluster; /* (array block) */
+ u64 root_inodes[0]; /* (array block) */
+} __attribute__((packed));
+
+struct descriptor {
+ u32 block_num; /* (file block) */
+ u16 block_cnt; /* int */
+ u16 block_bytes; /* 0..blocksize - 0 means 'punch a hole',
+ * 1..blocksize means advance EOF to there
+ */
+} __attribute__((packed));
+#define DescHole 0
+#define DescIndex 0xffff
+#define DescMiniOffset 0x8000
+
+#define ROUND_UP(x) (((x)+3)&~3)
+
+struct miniblock {
+ u32 block_num; /* (file block) */
+ u16 block_offset; /* (byte) */
+ u16 length; /* (bytes) + 0x8000 */
+ u8 data[0];
+} __attribute__((packed));
+
+struct group_head {
+ u32 inum;
+ u32 fsnum;
+ u16 truncatenum_and_flag;
+ u16 group_size_words; /* 4byte words */
+ union {
+ struct descriptor desc[0];
+ struct miniblock mb[0];
+ } u;
+} __attribute__((packed));
+
+struct cluster_head {
+ char idtag[8]; /* LaFSHead */
+ u8 uuid[16];
+ u64 seq;
+ u32 flags;
+ u16 Hlength; /* header length - (bytes) */
+ u16 Clength; /* cluster length including header - (blocks) */
+ u32 checksum; /* over Hlength bytes */
+ u16 verify_type;
+ u16 pad0;
+ u8 verify_data[16];
+ u64 next_addr; /* (Array block) */
+ u64 this_addr; /* (array block) */
+ u64 prev_addr; /* (array block) */
+ struct group_head groups[0];
+} __attribute__((packed));
+
+#define CH_Checkpoint 1
+#define CH_CheckpointStart 2
+#define CH_CheckpointEnd 4
+
+/* values for verify_type */
+#define VerifyNull 0 /* if you found me, I'm valid */
+#define VerifyNext 1 /* if next head is valid, this cluster is */
+#define VerifyNext2 2 /* if next 2 heads are valid, this cluster is */
+#define VerifySum 3 /* maybe some sort of MIC is in _data */
+
+struct la_inode {
+ /* 16 bytes is constant */
+ u32 data_blocks; /* (blocks) */
+ u32 index_blocks; /* (blocks) */
+ u16 generation;
+ u16 metadata_size; /* (bytes) */
+ u8 depth;
+ u8 trunc_gen;
+ u8 filetype;
+ u8 flags;
+#define File_nonlogged 1
+ union {
+ struct fs_metadata {
+ /* 116 bytes */
+ u64 update_time;
+ u64 blocks_used; /* data+index */
+ u64 blocks_allowed;
+ u64 creation_age;
+ u32 inodes_used;
+ u32 quota_inodes[3];
+ u16 snapshot_usage_table;
+ u16 pad;
+ char name[64];
+ } fs;
+ struct inodemap_metadata {
+ u32 size;
+ } inodemap;
+ struct su_metadata {
+ u32 table_size; /* (blocks) */
+ } segmentusage;
+ struct file_metadata {
+ u16 flags;
+ u16 mode;
+ u32 userid;
+ u32 groupid;
+ u32 treeid;
+ u64 creationtime;
+ u64 modifytime;
+ u64 ctime;
+ u64 accesstime;
+ u64 size;
+ u32 parent;
+ u32 linkcount;
+ u32 attrinode;
+ u32 attributes[0];
+ } __attribute__((packed)) file;
+ struct dir_metadata {
+ struct file_metadata h;
+ u32 hash_seed;
+ u32 attributes[0];
+ } dir;
+ struct special_metadata {
+ struct file_metadata h;
+ u32 major;
+ u32 minor;
+ u32 attributes[0];
+ } special;
+ struct quota_metadata {
+ u32 gracetime; /* typically '7' */
+ u32 graceunits; /* typically 24*60*60 */
+ } quota;
+ } metadata[0];
+} __attribute__((packed));
+
+#define LAFS_INODE_LOG_START offsetof(struct la_inode, metadata[0].file.flags)
+#define LAFS_INODE_LOG_END offsetof(struct la_inode, metadata[0].file.size)
+#define LAFS_INODE_LOG_SIZE (LAFS_INODE_LOG_END - LAFS_INODE_LOG_START)
+
+#define TypeInodeFile 1
+#define TypeInodeMap 2
+#define TypeSegmentMap 3
+#define TypeQuota 4
+#define TypeOrphanList 5
+#define TypeAccessTime 6
+
+#define TypeBase 16
+
+#define TypeFile 16
+#define TypeDir 17
+#define TypeSymlink 18
+#define TypeSpecial 19 /* char or block, or pipe or socket */
+#define TypeAttr 20
+
+#define LAFS_MAX_LINKS ((1UL<<31)-1)
+
+struct index {
+ u32 logical;
+ u32 phys_lo;
+ u16 phys_hi;
+} __attribute__((packed));
+
+struct extent {
+ u32 phys_lo;
+ u16 phys_hi;
+ u16 size;
+ u32 logical;
+} __attribute__((packed));
+#define IBLK_INDEX (0)
+#define IBLK_INDIRECT (1)
+#define IBLK_EXTENT (2)
+
+#define MaxDirHash 0x7fffffffUL
+struct dirpiece {
+ u32 target; /* inode number */
+ u8 next[2]; /* back, fore */
+ u8 type:4, chain_info:2, longer:2;
+/* 'longer' is 3 if fore and back the same length
+ * 0 if back (next[0]) is longer
+ * 1 if fore (next[1]) is longer
+ * 'chain_info' is
+ * 0,1: add that number to the hash of filename
+ * 2 : add one trailing byte to hash
+ * 3 : add 4 trailing bytes (little-endian) to hash
+ */
+#define Neither 3
+ u8 length;
+ char name[0];
+} __attribute__((packed));
+
+#define NoBlock (0xFFFFFFFF)
+struct dirheader {
+ u8 root;
+ u8 lastpiece;
+ u8 freepieces;
+ u8 pad;
+} __attribute__((packed));
+
+/*
+ * Miniblock for directory updates record the operation type
+ * in the block_offset
+ */
+#define DIROP_LINK 0
+#define DIROP_UNLINK 1
+#define DIROP_REN_SOURCE 2
+#define DIROP_REN_NEW_TARGET 3
+#define DIROP_REN_OLD_TARGET 4
+
+/*
+ * The orphan file has a very simple structure with
+ * 16 byte records identifying blocks in files that
+ * might be orphans
+ */
+struct orphan {
+ u32 type; /* 0 if free */
+ u32 filesys;
+ u32 inum;
+ u32 addr;
+} __attribute__((packed));
+
+/* Youth values are decayed when nextyouth gets too big */
+static int inline decay_youth(int y)
+{
+ if (y < 8)
+ return y;
+ if (y < 32768+8)
+ y = (y-8)/2 + 8;
+ else
+ y -= 16384;
+ return y;
+}
+/* This is only called on large youth values */
+static int inline decay_undo(int y)
+{
+ return y + 16384;
+}
--- /dev/null
+
+/* fromlinux-2.5.43 but with prefetch removed
+ and container_of added
+*/
+#ifndef _LINUX_LIST_H
+#define _LINUX_LIST_H
+/**
+ * container_of - cast a member of a structure out to the containing structure
+ *
+ * @ptr: the pointer to the member.
+ * @type: the type of the container struct this is embedded in.
+ * @member: the name of the member within the struct.
+ *
+ */
+#ifndef offsetof
+# define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
+#endif
+#define container_of(ptr, type, member) ({ \
+ const typeof( ((type *)0)->member ) *__mptr = (ptr); \
+ (type *)( (char *)__mptr - offsetof(type,member) );})
+
+
+/*
+ * Simple doubly linked list implementation.
+ *
+ * Some of the internal functions ("__xxx") are useful when
+ * manipulating whole lists rather than single entries, as
+ * sometimes we already know the next/prev entries and we can
+ * generate better code by using them directly rather than
+ * using the generic single-entry routines.
+ */
+
+struct list_head {
+ struct list_head *next, *prev;
+};
+
+#define LIST_HEAD_INIT(name) { &(name), &(name) }
+
+#define LIST_HEAD(name) \
+ struct list_head name = LIST_HEAD_INIT(name)
+
+#define INIT_LIST_HEAD(ptr) do { \
+ (ptr)->next = (ptr); (ptr)->prev = (ptr); \
+} while (0)
+
+/*
+ * Insert a new entry between two known consecutive entries.
+ *
+ * This is only for internal list manipulation where we know
+ * the prev/next entries already!
+ */
+static inline void __list_add(struct list_head *new,
+ struct list_head *prev,
+ struct list_head *next)
+{
+ next->prev = new;
+ new->next = next;
+ new->prev = prev;
+ prev->next = new;
+}
+
+/**
+ * list_add - add a new entry
+ * @new: new entry to be added
+ * @head: list head to add it after
+ *
+ * Insert a new entry after the specified head.
+ * This is good for implementing stacks.
+ */
+static inline void list_add(struct list_head *new, struct list_head *head)
+{
+ __list_add(new, head, head->next);
+}
+
+/**
+ * list_add_tail - add a new entry
+ * @new: new entry to be added
+ * @head: list head to add it before
+ *
+ * Insert a new entry before the specified head.
+ * This is useful for implementing queues.
+ */
+static inline void list_add_tail(struct list_head *new, struct list_head *head)
+{
+ __list_add(new, head->prev, head);
+}
+
+/*
+ * Delete a list entry by making the prev/next entries
+ * point to each other.
+ *
+ * This is only for internal list manipulation where we know
+ * the prev/next entries already!
+ */
+static inline void __list_del(struct list_head * prev, struct list_head * next)
+{
+ next->prev = prev;
+ prev->next = next;
+}
+
+/**
+ * list_del - deletes entry from list.
+ * @entry: the element to delete from the list.
+ * Note: list_empty on entry does not return true after this, the entry is
+ * in an undefined state.
+ */
+static inline void list_del(struct list_head *entry)
+{
+ __list_del(entry->prev, entry->next);
+}
+
+/**
+ * list_del_init - deletes entry from list and reinitialize it.
+ * @entry: the element to delete from the list.
+ */
+static inline void list_del_init(struct list_head *entry)
+{
+ __list_del(entry->prev, entry->next);
+ INIT_LIST_HEAD(entry);
+}
+
+/**
+ * list_move - delete from one list and add as another's head
+ * @list: the entry to move
+ * @head: the head that will precede our entry
+ */
+static inline void list_move(struct list_head *list, struct list_head *head)
+{
+ __list_del(list->prev, list->next);
+ list_add(list, head);
+}
+
+/**
+ * list_move_tail - delete from one list and add as another's tail
+ * @list: the entry to move
+ * @head: the head that will follow our entry
+ */
+static inline void list_move_tail(struct list_head *list,
+ struct list_head *head)
+{
+ __list_del(list->prev, list->next);
+ list_add_tail(list, head);
+}
+
+/**
+ * list_empty - tests whether a list is empty
+ * @head: the list to test.
+ */
+static inline int list_empty(struct list_head *head)
+{
+ return head->next == head;
+}
+
+static inline void __list_splice(struct list_head *list,
+ struct list_head *head)
+{
+ struct list_head *first = list->next;
+ struct list_head *last = list->prev;
+ struct list_head *at = head->next;
+
+ first->prev = head;
+ head->next = first;
+
+ last->next = at;
+ at->prev = last;
+}
+
+/**
+ * list_splice - join two lists
+ * @list: the new list to add.
+ * @head: the place to add it in the first list.
+ */
+static inline void list_splice(struct list_head *list, struct list_head *head)
+{
+ if (!list_empty(list))
+ __list_splice(list, head);
+}
+
+/**
+ * list_splice_init - join two lists and reinitialise the emptied list.
+ * @list: the new list to add.
+ * @head: the place to add it in the first list.
+ *
+ * The list at @list is reinitialised
+ */
+static inline void list_splice_init(struct list_head *list,
+ struct list_head *head)
+{
+ if (!list_empty(list)) {
+ __list_splice(list, head);
+ INIT_LIST_HEAD(list);
+ }
+}
+
+/**
+ * list_entry - get the struct for this entry
+ * @ptr: the &struct list_head pointer.
+ * @type: the type of the struct this is embedded in.
+ * @member: the name of the list_struct within the struct.
+ */
+#define list_entry(ptr, type, member) \
+ container_of(ptr, type, member)
+
+/**
+ * list_first_entry - get the first element from a list
+ * @ptr: the list head to take the element from.
+ * @type: the type of the struct this is embedded in.
+ * @member: the name of the list_struct within the struct.
+ *
+ * Note, that list is expected to be not empty.
+ */
+#define list_first_entry(ptr, type, member) \
+ list_entry((ptr)->next, type, member)
+
+/**
+ * list_for_each - iterate over a list
+ * @pos: the &struct list_head to use as a loop counter.
+ * @head: the head for your list.
+ */
+#define list_for_each(pos, head) \
+ for (pos = (head)->next; pos != (head); \
+ pos = pos->next)
+
+/**
+ * __list_for_each - iterate over a list
+ * @pos: the &struct list_head to use as a loop counter.
+ * @head: the head for your list.
+ *
+ * This variant differs from list_for_each() in that it's the
+ * simplest possible list iteration code, no prefetching is done.
+ * Use this for code that knows the list to be very short (empty
+ * or 1 entry) most of the time.
+ */
+#define __list_for_each(pos, head) \
+ for (pos = (head)->next; pos != (head); pos = pos->next)
+
+/**
+ * list_for_each_prev - iterate over a list backwards
+ * @pos: the &struct list_head to use as a loop counter.
+ * @head: the head for your list.
+ */
+#define list_for_each_prev(pos, head) \
+ for (pos = (head)->prev; pos != (head); \
+ pos = pos->prev)
+
+/**
+ * list_for_each_safe - iterate over a list safe against removal of list entry
+ * @pos: the &struct list_head to use as a loop counter.
+ * @n: another &struct list_head to use as temporary storage
+ * @head: the head for your list.
+ */
+#define list_for_each_safe(pos, n, head) \
+ for (pos = (head)->next, n = pos->next; pos != (head); \
+ pos = n, n = pos->next)
+
+/**
+ * list_for_each_entry - iterate over list of given type
+ * @pos: the type * to use as a loop counter.
+ * @head: the head for your list.
+ * @member: the name of the list_struct within the struct.
+ */
+#define list_for_each_entry(pos, head, member) \
+ for (pos = list_entry((head)->next, typeof(*pos), member); \
+ &pos->member != (head); \
+ pos = list_entry(pos->member.next, typeof(*pos), member) \
+ )
+
+#define list_for_each_entry_safe(pos, n, head, member) \
+ for (pos = list_entry((head)->next, typeof(*pos), member), \
+ n = pos->member.next; \
+ &pos->member != (head); \
+ pos = list_entry(n, typeof(*pos), member), \
+ n= pos->member.next \
+ )
+#endif
--- /dev/null
+#include <sys/types.h>
+#include <lafs/list.h>
+#include <stdint.h>
+
+#define WC_NUM 2
+
+#define HASH_BITS 8
+
+struct lafs {
+ char version[16];
+ uint8_t uuid[16];
+ uint32_t seq;
+ int devices;
+ int blocksize;
+ int statesize;
+ int max_segment;
+
+ int flags;
+#define LAFS_DELAY_UPDATES 1
+
+ int checkpointing;
+#define CHECKPOINTING 1
+#define CHECKPOINT_START 2
+#define CHECKPOINT_END 4
+ loff_t checkpoint_cluster;
+
+ struct lafs_snapshot {
+ struct lafs_snapshot *next;
+ loff_t root_addr;
+ struct lafs_ino *root;
+ struct lafs_ino *rootdir;
+ } ss;
+
+ struct lafs_device *devs;
+ int loaded_devs;
+
+ struct lafs_cluster {
+ struct list_head blocks;
+ loff_t prev_addr;
+ uint64_t seq;
+ int remaining;
+
+ char *chead;
+ int chead_size;
+ int chead_blocks;
+
+ struct lafs_segpos {
+ int dev;
+ loff_t num;
+
+ uint32_t st_table, st_row;
+ uint32_t nxt_table, nxt_row;
+ uint32_t table, row, col;
+
+ } seg;
+ } wc[WC_NUM];
+
+ struct list_head leafs, account_leafs;
+
+ int youth_next;
+
+ struct list_head htable[1 << HASH_BITS];
+
+ /* free segment list
+ * If free_head == free_tail, the list is empty
+ * Else free_head is the first to use and free_tail
+ * is the next to fill.
+ */
+ int free_head, free_tail;
+ struct {
+ int dev;
+ loff_t seg;
+ } freelist[128];
+
+ struct lafs_delayed {
+ struct lafs_delayed *next;
+ int dev;
+ loff_t seg;
+ int diff;
+ } *delayed;
+};
+
+
+struct lafs_ino {
+ struct lafs *fs;
+ struct lafs_dblk *dblock;
+ struct lafs_iblk *iblock;
+ int inum;
+ struct lafs_ino *filesys;
+
+ u32 cblocks, /* data blocks which are commited to this file */
+ ablocks, /* data blocks that are allocated */
+
+ ciblocks; /* index blocks committed */
+
+ int iflags;
+#define I_Phase1 1 /* set to 'this' phase when iblocks correct */
+#define I_Dirty 2
+/* next three indicate if we hold a reference on the relevant qent */
+#define I_QUid 8
+#define I_QGid 16
+#define I_QTid 32
+
+ int generation;
+ int trunc_gen;
+ int flags;
+
+ int type;
+ int metadata_size;
+ int depth;
+
+ union {
+ struct fs_md {
+ int usagetable;
+ uint64_t update_time;
+ uint64_t cblocks_used; /* blocks commited */
+ uint64_t ablocks_used; /* extra blocks allocated */
+ uint64_t blocks_allowed;
+ uint64_t blocks_unalloc;
+ uint64_t creation_age;
+ uint32_t inodes_used;
+ uint32_t quota_inums[3];
+ struct inode *quota_inodes[3];
+ char name[65];
+ } fs;
+ struct inodemap_md {
+ uint32_t size;
+ } inodemap;
+ struct su_md {
+ uint32_t table_size; /* (blocks) */
+ } segmentusage;
+ struct file_md {
+ uint16_t flags;
+ uint16_t mode;
+ uint32_t uid;
+ uint32_t gid;
+ uint32_t treeid;
+ uint64_t creationtime;
+ uint64_t modifytime;
+ uint64_t ctime;
+ uint64_t accesstime;
+ uint64_t i_accesstime; /* as stored in inode */
+ uint64_t size;
+ uint32_t parent;
+ uint32_t linkcount;
+ /* for directories */
+ uint32_t seed;
+ /* for special */
+ uint32_t major, minor;
+ /* lists of blocks which need orphan-treatment */
+ struct list_head dirorphan;
+ } file;
+ struct orphan_md {
+ uint32_t nextfree;
+ uint32_t reserved;
+ } orphan;
+ struct quota_md {
+ uint32_t gracetime; /* multiplier for graceunits */
+ uint32_t graceunits; /* seconds per unit */
+ } quota;
+ } md;
+};
+
+struct lafs_device {
+ struct lafs_device *next;
+ struct lafs *fs;
+ int fd;
+ int seq;
+ loff_t start, size;
+ loff_t devaddr[2];
+ loff_t stateaddr[4];
+ int devnum;
+ char *name;
+
+ struct timeval ctime;
+
+ int width;
+ loff_t stride, segment_size;
+ loff_t segment_offset, segment_stride, segment_count;
+ int usage_inum;
+
+ int rows_per_table, tables_per_seg;
+ int tablesize;
+
+ int recent_super, recent_state;
+
+ struct lafs_ino *segsum;
+};
+
+struct lafs_blk {
+ char *data;
+ int flags;
+ loff_t fileaddr;
+ loff_t physaddr;
+
+ struct lafs_ino *ino;
+ struct list_head hash;
+
+ struct lafs_iblk *parent;
+ struct list_head siblings;
+ struct list_head leafs;
+
+ struct lafs_blk *chain;
+};
+
+struct lafs_dblk {
+ struct lafs_blk b;
+ struct lafs_ino *my_inode;
+};
+
+struct lafs_iblk {
+ struct lafs_blk b;
+ int depth;
+ struct list_head children;
+ struct lafs_blk *uninc;
+ int sched_cnt;
+};
+
+#define B_Valid 1
+#define B_Dirty 2
+#define B_Index 4
+#define B_Sched 8
+#define B_InoIdx 16
+
--- /dev/null
+
+
+SRC = $(wildcard *.c)
+OBJ = $(patsubst %.c,%.o,$(SRC))
+INCL = $(wildcard ../include/lafs/*.h)
+
+CPPFLAGS = -I../include
+CFLAGS = -Wall -Werror -g
+
+liblafs.a : $(OBJ)
+ ar cr liblafs.a $(OBJ)
+ ranlib liblafs.a
+
+$(OBJ): %.o : %.c $(INCL)
--- /dev/null
+/* crc32.c -- compute the CRC-32 of a data stream
+ * Copyright (C) 1995-2003 Mark Adler
+ * For conditions of distribution and use, see copyright notice in zlib.h
+ *
+ * Thanks to Rodney Brown <rbrown64@csc.com.au> for his contribution of faster
+ * CRC methods: exclusive-oring 32 bits of data at a time, and pre-computing
+ * tables for updating the shift register in one step with three exclusive-ors
+ * instead of four steps with four exclusive-ors. This results about a factor
+ * of two increase in speed on a Power PC G4 (PPC7455) using gcc -O3.
+ */
+
+/* @(#) $Id$ */
+
+/*
+ Note on the use of DYNAMIC_CRC_TABLE: there is no mutex or semaphore
+ protection on the static variables used to control the first-use generation
+ of the crc tables. Therefore, if you #define DYNAMIC_CRC_TABLE, you should
+ first call get_crc_table() to initialize the tables before allowing more than
+ one thread to use crc32().
+ */
+
+#ifdef MAKECRCH
+# include <stdio.h>
+# ifndef DYNAMIC_CRC_TABLE
+# define DYNAMIC_CRC_TABLE
+# endif /* !DYNAMIC_CRC_TABLE */
+#endif /* MAKECRCH */
+
+/* #include "zutil.h" / * for STDC and FAR definitions */
+#define STDC
+#define FAR
+#define Z_NULL ((void*)0)
+#define OF(X) X
+#define ZEXPORT
+typedef long ptrdiff_t;
+#define NOBYFOUR
+
+#define local static
+
+/* Find a four-byte integer type for crc32_little() and crc32_big(). */
+#ifndef NOBYFOUR
+# ifdef STDC /* need ANSI C limits.h to determine sizes */
+# include <limits.h>
+# define BYFOUR
+# if (UINT_MAX == 0xffffffffUL)
+ typedef unsigned int u4;
+# else
+# if (ULONG_MAX == 0xffffffffUL)
+ typedef unsigned long u4;
+# else
+# if (USHRT_MAX == 0xffffffffUL)
+ typedef unsigned short u4;
+# else
+# undef BYFOUR /* can't find a four-byte integer type! */
+# endif
+# endif
+# endif
+# endif /* STDC */
+#endif /* !NOBYFOUR */
+
+/* Definitions for doing the crc four data bytes at a time. */
+#ifdef BYFOUR
+# define REV(w) (((w)>>24)+(((w)>>8)&0xff00)+ \
+ (((w)&0xff00)<<8)+(((w)&0xff)<<24))
+ local unsigned long crc32_little OF((unsigned long,
+ const unsigned char FAR *, unsigned));
+ local unsigned long crc32_big OF((unsigned long,
+ const unsigned char FAR *, unsigned));
+# define TBLS 8
+#else
+# define TBLS 1
+#endif /* BYFOUR */
+
+#ifdef DYNAMIC_CRC_TABLE
+
+local volatile int crc_table_empty = 1;
+local unsigned long FAR crc_table[TBLS][256];
+local void make_crc_table OF((void));
+#ifdef MAKECRCH
+ local void write_table OF((FILE *, const unsigned long FAR *));
+#endif /* MAKECRCH */
+
+/*
+ Generate tables for a byte-wise 32-bit CRC calculation on the polynomial:
+ x^32+x^26+x^23+x^22+x^16+x^12+x^11+x^10+x^8+x^7+x^5+x^4+x^2+x+1.
+
+ Polynomials over GF(2) are represented in binary, one bit per coefficient,
+ with the lowest powers in the most significant bit. Then adding polynomials
+ is just exclusive-or, and multiplying a polynomial by x is a right shift by
+ one. If we call the above polynomial p, and represent a byte as the
+ polynomial q, also with the lowest power in the most significant bit (so the
+ byte 0xb1 is the polynomial x^7+x^3+x+1), then the CRC is (q*x^32) mod p,
+ where a mod b means the remainder after dividing a by b.
+
+ This calculation is done using the shift-register method of multiplying and
+ taking the remainder. The register is initialized to zero, and for each
+ incoming bit, x^32 is added mod p to the register if the bit is a one (where
+ x^32 mod p is p+x^32 = x^26+...+1), and the register is multiplied mod p by
+ x (which is shifting right by one and adding x^32 mod p if the bit shifted
+ out is a one). We start with the highest power (least significant bit) of
+ q and repeat for all eight bits of q.
+
+ The first table is simply the CRC of all possible eight bit values. This is
+ all the information needed to generate CRCs on data a byte at a time for all
+ combinations of CRC register values and incoming bytes. The remaining tables
+ allow for word-at-a-time CRC calculation for both big-endian and little-
+ endian machines, where a word is four bytes.
+*/
+local void make_crc_table()
+{
+ unsigned long c;
+ int n, k;
+ unsigned long poly; /* polynomial exclusive-or pattern */
+ /* terms of polynomial defining this crc (except x^32): */
+ static volatile int first = 1; /* flag to limit concurrent making */
+ static const unsigned char p[] = {0,1,2,4,5,7,8,10,11,12,16,22,23,26};
+
+ /* See if another task is already doing this (not thread-safe, but better
+ than nothing -- significantly reduces duration of vulnerability in
+ case the advice about DYNAMIC_CRC_TABLE is ignored) */
+ if (first) {
+ first = 0;
+
+ /* make exclusive-or pattern from polynomial (0xedb88320UL) */
+ poly = 0UL;
+ for (n = 0; n < sizeof(p)/sizeof(unsigned char); n++)
+ poly |= 1UL << (31 - p[n]);
+
+ /* generate a crc for every 8-bit value */
+ for (n = 0; n < 256; n++) {
+ c = (unsigned long)n;
+ for (k = 0; k < 8; k++)
+ c = c & 1 ? poly ^ (c >> 1) : c >> 1;
+ crc_table[0][n] = c;
+ }
+
+#ifdef BYFOUR
+ /* generate crc for each value followed by one, two, and three zeros,
+ and then the byte reversal of those as well as the first table */
+ for (n = 0; n < 256; n++) {
+ c = crc_table[0][n];
+ crc_table[4][n] = REV(c);
+ for (k = 1; k < 4; k++) {
+ c = crc_table[0][c & 0xff] ^ (c >> 8);
+ crc_table[k][n] = c;
+ crc_table[k + 4][n] = REV(c);
+ }
+ }
+#endif /* BYFOUR */
+
+ crc_table_empty = 0;
+ }
+ else { /* not first */
+ /* wait for the other guy to finish (not efficient, but rare) */
+ while (crc_table_empty)
+ ;
+ }
+
+#ifdef MAKECRCH
+ /* write out CRC tables to crc32.h */
+ {
+ FILE *out;
+
+ out = fopen("crc32.h", "w");
+ if (out == NULL) return;
+ fprintf(out, "/* crc32.h -- tables for rapid CRC calculation\n");
+ fprintf(out, " * Generated automatically by crc32.c\n */\n\n");
+ fprintf(out, "local const unsigned long FAR ");
+ fprintf(out, "crc_table[TBLS][256] =\n{\n {\n");
+ write_table(out, crc_table[0]);
+# ifdef BYFOUR
+ fprintf(out, "#ifdef BYFOUR\n");
+ for (k = 1; k < 8; k++) {
+ fprintf(out, " },\n {\n");
+ write_table(out, crc_table[k]);
+ }
+ fprintf(out, "#endif\n");
+# endif /* BYFOUR */
+ fprintf(out, " }\n};\n");
+ fclose(out);
+ }
+#endif /* MAKECRCH */
+}
+
+#ifdef MAKECRCH
+local void write_table(out, table)
+ FILE *out;
+ const unsigned long FAR *table;
+{
+ int n;
+
+ for (n = 0; n < 256; n++)
+ fprintf(out, "%s0x%08lxUL%s", n % 5 ? "" : " ", table[n],
+ n == 255 ? "\n" : (n % 5 == 4 ? ",\n" : ", "));
+}
+#endif /* MAKECRCH */
+
+#else /* !DYNAMIC_CRC_TABLE */
+/* ========================================================================
+ * Tables of CRC-32s of all single-byte values, made by make_crc_table().
+ */
+#include "crc32.h"
+#endif /* DYNAMIC_CRC_TABLE */
+
+/* =========================================================================
+ * This function can be used by asm versions of crc32()
+ */
+const unsigned long FAR * ZEXPORT get_crc_table(void)
+{
+#ifdef DYNAMIC_CRC_TABLE
+ if (crc_table_empty)
+ make_crc_table();
+#endif /* DYNAMIC_CRC_TABLE */
+ return (const unsigned long FAR *)crc_table;
+}
+
+/* ========================================================================= */
+#define DO1 crc = crc_table[0][((int)crc ^ (*buf++)) & 0xff] ^ (crc >> 8)
+#define DO8 DO1; DO1; DO1; DO1; DO1; DO1; DO1; DO1
+
+/* ========================================================================= */
+unsigned long ZEXPORT crc32(
+ unsigned long crc,
+ const unsigned char FAR *buf,
+ unsigned len)
+{
+ if (buf == Z_NULL) return 0UL;
+
+#ifdef DYNAMIC_CRC_TABLE
+ if (crc_table_empty)
+ make_crc_table();
+#endif /* DYNAMIC_CRC_TABLE */
+
+#ifdef BYFOUR
+ if (sizeof(void *) == sizeof(ptrdiff_t)) {
+ u4 endian;
+
+ endian = 1;
+ if (*((unsigned char *)(&endian)))
+ return crc32_little(crc, buf, len);
+ else
+ return crc32_big(crc, buf, len);
+ }
+#endif /* BYFOUR */
+/* crc = crc ^ 0xffffffffUL;*/
+ while (len >= 8) {
+ DO8;
+ len -= 8;
+ }
+ if (len) do {
+ DO1;
+ } while (--len);
+ return crc /* ^ 0xffffffffUL*/;
+}
+
+#ifdef BYFOUR
+
+/* ========================================================================= */
+#define DOLIT4 c ^= *buf4++; \
+ c = crc_table[3][c & 0xff] ^ crc_table[2][(c >> 8) & 0xff] ^ \
+ crc_table[1][(c >> 16) & 0xff] ^ crc_table[0][c >> 24]
+#define DOLIT32 DOLIT4; DOLIT4; DOLIT4; DOLIT4; DOLIT4; DOLIT4; DOLIT4; DOLIT4
+
+/* ========================================================================= */
+local unsigned long crc32_little(crc, buf, len)
+ unsigned long crc;
+ const unsigned char FAR *buf;
+ unsigned len;
+{
+ register u4 c;
+ register const u4 FAR *buf4;
+
+ c = (u4)crc;
+ c = ~c;
+ while (len && ((ptrdiff_t)buf & 3)) {
+ c = crc_table[0][(c ^ *buf++) & 0xff] ^ (c >> 8);
+ len--;
+ }
+
+ buf4 = (const u4 FAR *)buf;
+ while (len >= 32) {
+ DOLIT32;
+ len -= 32;
+ }
+ while (len >= 4) {
+ DOLIT4;
+ len -= 4;
+ }
+ buf = (const unsigned char FAR *)buf4;
+
+ if (len) do {
+ c = crc_table[0][(c ^ *buf++) & 0xff] ^ (c >> 8);
+ } while (--len);
+ c = ~c;
+ return (unsigned long)c;
+}
+
+/* ========================================================================= */
+#define DOBIG4 c ^= *++buf4; \
+ c = crc_table[4][c & 0xff] ^ crc_table[5][(c >> 8) & 0xff] ^ \
+ crc_table[6][(c >> 16) & 0xff] ^ crc_table[7][c >> 24]
+#define DOBIG32 DOBIG4; DOBIG4; DOBIG4; DOBIG4; DOBIG4; DOBIG4; DOBIG4; DOBIG4
+
+/* ========================================================================= */
+local unsigned long crc32_big(crc, buf, len)
+ unsigned long crc;
+ const unsigned char FAR *buf;
+ unsigned len;
+{
+ register u4 c;
+ register const u4 FAR *buf4;
+
+ c = REV((u4)crc);
+ c = ~c;
+ while (len && ((ptrdiff_t)buf & 3)) {
+ c = crc_table[4][(c >> 24) ^ *buf++] ^ (c << 8);
+ len--;
+ }
+
+ buf4 = (const u4 FAR *)buf;
+ buf4--;
+ while (len >= 32) {
+ DOBIG32;
+ len -= 32;
+ }
+ while (len >= 4) {
+ DOBIG4;
+ len -= 4;
+ }
+ buf4++;
+ buf = (const unsigned char FAR *)buf4;
+
+ if (len) do {
+ c = crc_table[4][(c >> 24) ^ *buf++] ^ (c << 8);
+ } while (--len);
+ c = ~c;
+ return (unsigned long)(REV(c));
+}
+
+#endif /* BYFOUR */
--- /dev/null
+/* crc32.h -- tables for rapid CRC calculation
+ * Generated automatically by crc32.c
+ */
+
+local const unsigned long FAR crc_table[TBLS][256] =
+{
+ {
+ 0x00000000UL, 0x77073096UL, 0xee0e612cUL, 0x990951baUL, 0x076dc419UL,
+ 0x706af48fUL, 0xe963a535UL, 0x9e6495a3UL, 0x0edb8832UL, 0x79dcb8a4UL,
+ 0xe0d5e91eUL, 0x97d2d988UL, 0x09b64c2bUL, 0x7eb17cbdUL, 0xe7b82d07UL,
+ 0x90bf1d91UL, 0x1db71064UL, 0x6ab020f2UL, 0xf3b97148UL, 0x84be41deUL,
+ 0x1adad47dUL, 0x6ddde4ebUL, 0xf4d4b551UL, 0x83d385c7UL, 0x136c9856UL,
+ 0x646ba8c0UL, 0xfd62f97aUL, 0x8a65c9ecUL, 0x14015c4fUL, 0x63066cd9UL,
+ 0xfa0f3d63UL, 0x8d080df5UL, 0x3b6e20c8UL, 0x4c69105eUL, 0xd56041e4UL,
+ 0xa2677172UL, 0x3c03e4d1UL, 0x4b04d447UL, 0xd20d85fdUL, 0xa50ab56bUL,
+ 0x35b5a8faUL, 0x42b2986cUL, 0xdbbbc9d6UL, 0xacbcf940UL, 0x32d86ce3UL,
+ 0x45df5c75UL, 0xdcd60dcfUL, 0xabd13d59UL, 0x26d930acUL, 0x51de003aUL,
+ 0xc8d75180UL, 0xbfd06116UL, 0x21b4f4b5UL, 0x56b3c423UL, 0xcfba9599UL,
+ 0xb8bda50fUL, 0x2802b89eUL, 0x5f058808UL, 0xc60cd9b2UL, 0xb10be924UL,
+ 0x2f6f7c87UL, 0x58684c11UL, 0xc1611dabUL, 0xb6662d3dUL, 0x76dc4190UL,
+ 0x01db7106UL, 0x98d220bcUL, 0xefd5102aUL, 0x71b18589UL, 0x06b6b51fUL,
+ 0x9fbfe4a5UL, 0xe8b8d433UL, 0x7807c9a2UL, 0x0f00f934UL, 0x9609a88eUL,
+ 0xe10e9818UL, 0x7f6a0dbbUL, 0x086d3d2dUL, 0x91646c97UL, 0xe6635c01UL,
+ 0x6b6b51f4UL, 0x1c6c6162UL, 0x856530d8UL, 0xf262004eUL, 0x6c0695edUL,
+ 0x1b01a57bUL, 0x8208f4c1UL, 0xf50fc457UL, 0x65b0d9c6UL, 0x12b7e950UL,
+ 0x8bbeb8eaUL, 0xfcb9887cUL, 0x62dd1ddfUL, 0x15da2d49UL, 0x8cd37cf3UL,
+ 0xfbd44c65UL, 0x4db26158UL, 0x3ab551ceUL, 0xa3bc0074UL, 0xd4bb30e2UL,
+ 0x4adfa541UL, 0x3dd895d7UL, 0xa4d1c46dUL, 0xd3d6f4fbUL, 0x4369e96aUL,
+ 0x346ed9fcUL, 0xad678846UL, 0xda60b8d0UL, 0x44042d73UL, 0x33031de5UL,
+ 0xaa0a4c5fUL, 0xdd0d7cc9UL, 0x5005713cUL, 0x270241aaUL, 0xbe0b1010UL,
+ 0xc90c2086UL, 0x5768b525UL, 0x206f85b3UL, 0xb966d409UL, 0xce61e49fUL,
+ 0x5edef90eUL, 0x29d9c998UL, 0xb0d09822UL, 0xc7d7a8b4UL, 0x59b33d17UL,
+ 0x2eb40d81UL, 0xb7bd5c3bUL, 0xc0ba6cadUL, 0xedb88320UL, 0x9abfb3b6UL,
+ 0x03b6e20cUL, 0x74b1d29aUL, 0xead54739UL, 0x9dd277afUL, 0x04db2615UL,
+ 0x73dc1683UL, 0xe3630b12UL, 0x94643b84UL, 0x0d6d6a3eUL, 0x7a6a5aa8UL,
+ 0xe40ecf0bUL, 0x9309ff9dUL, 0x0a00ae27UL, 0x7d079eb1UL, 0xf00f9344UL,
+ 0x8708a3d2UL, 0x1e01f268UL, 0x6906c2feUL, 0xf762575dUL, 0x806567cbUL,
+ 0x196c3671UL, 0x6e6b06e7UL, 0xfed41b76UL, 0x89d32be0UL, 0x10da7a5aUL,
+ 0x67dd4accUL, 0xf9b9df6fUL, 0x8ebeeff9UL, 0x17b7be43UL, 0x60b08ed5UL,
+ 0xd6d6a3e8UL, 0xa1d1937eUL, 0x38d8c2c4UL, 0x4fdff252UL, 0xd1bb67f1UL,
+ 0xa6bc5767UL, 0x3fb506ddUL, 0x48b2364bUL, 0xd80d2bdaUL, 0xaf0a1b4cUL,
+ 0x36034af6UL, 0x41047a60UL, 0xdf60efc3UL, 0xa867df55UL, 0x316e8eefUL,
+ 0x4669be79UL, 0xcb61b38cUL, 0xbc66831aUL, 0x256fd2a0UL, 0x5268e236UL,
+ 0xcc0c7795UL, 0xbb0b4703UL, 0x220216b9UL, 0x5505262fUL, 0xc5ba3bbeUL,
+ 0xb2bd0b28UL, 0x2bb45a92UL, 0x5cb36a04UL, 0xc2d7ffa7UL, 0xb5d0cf31UL,
+ 0x2cd99e8bUL, 0x5bdeae1dUL, 0x9b64c2b0UL, 0xec63f226UL, 0x756aa39cUL,
+ 0x026d930aUL, 0x9c0906a9UL, 0xeb0e363fUL, 0x72076785UL, 0x05005713UL,
+ 0x95bf4a82UL, 0xe2b87a14UL, 0x7bb12baeUL, 0x0cb61b38UL, 0x92d28e9bUL,
+ 0xe5d5be0dUL, 0x7cdcefb7UL, 0x0bdbdf21UL, 0x86d3d2d4UL, 0xf1d4e242UL,
+ 0x68ddb3f8UL, 0x1fda836eUL, 0x81be16cdUL, 0xf6b9265bUL, 0x6fb077e1UL,
+ 0x18b74777UL, 0x88085ae6UL, 0xff0f6a70UL, 0x66063bcaUL, 0x11010b5cUL,
+ 0x8f659effUL, 0xf862ae69UL, 0x616bffd3UL, 0x166ccf45UL, 0xa00ae278UL,
+ 0xd70dd2eeUL, 0x4e048354UL, 0x3903b3c2UL, 0xa7672661UL, 0xd06016f7UL,
+ 0x4969474dUL, 0x3e6e77dbUL, 0xaed16a4aUL, 0xd9d65adcUL, 0x40df0b66UL,
+ 0x37d83bf0UL, 0xa9bcae53UL, 0xdebb9ec5UL, 0x47b2cf7fUL, 0x30b5ffe9UL,
+ 0xbdbdf21cUL, 0xcabac28aUL, 0x53b39330UL, 0x24b4a3a6UL, 0xbad03605UL,
+ 0xcdd70693UL, 0x54de5729UL, 0x23d967bfUL, 0xb3667a2eUL, 0xc4614ab8UL,
+ 0x5d681b02UL, 0x2a6f2b94UL, 0xb40bbe37UL, 0xc30c8ea1UL, 0x5a05df1bUL,
+ 0x2d02ef8dUL
+#ifdef BYFOUR
+ },
+ {
+ 0x00000000UL, 0x191b3141UL, 0x32366282UL, 0x2b2d53c3UL, 0x646cc504UL,
+ 0x7d77f445UL, 0x565aa786UL, 0x4f4196c7UL, 0xc8d98a08UL, 0xd1c2bb49UL,
+ 0xfaefe88aUL, 0xe3f4d9cbUL, 0xacb54f0cUL, 0xb5ae7e4dUL, 0x9e832d8eUL,
+ 0x87981ccfUL, 0x4ac21251UL, 0x53d92310UL, 0x78f470d3UL, 0x61ef4192UL,
+ 0x2eaed755UL, 0x37b5e614UL, 0x1c98b5d7UL, 0x05838496UL, 0x821b9859UL,
+ 0x9b00a918UL, 0xb02dfadbUL, 0xa936cb9aUL, 0xe6775d5dUL, 0xff6c6c1cUL,
+ 0xd4413fdfUL, 0xcd5a0e9eUL, 0x958424a2UL, 0x8c9f15e3UL, 0xa7b24620UL,
+ 0xbea97761UL, 0xf1e8e1a6UL, 0xe8f3d0e7UL, 0xc3de8324UL, 0xdac5b265UL,
+ 0x5d5daeaaUL, 0x44469febUL, 0x6f6bcc28UL, 0x7670fd69UL, 0x39316baeUL,
+ 0x202a5aefUL, 0x0b07092cUL, 0x121c386dUL, 0xdf4636f3UL, 0xc65d07b2UL,
+ 0xed705471UL, 0xf46b6530UL, 0xbb2af3f7UL, 0xa231c2b6UL, 0x891c9175UL,
+ 0x9007a034UL, 0x179fbcfbUL, 0x0e848dbaUL, 0x25a9de79UL, 0x3cb2ef38UL,
+ 0x73f379ffUL, 0x6ae848beUL, 0x41c51b7dUL, 0x58de2a3cUL, 0xf0794f05UL,
+ 0xe9627e44UL, 0xc24f2d87UL, 0xdb541cc6UL, 0x94158a01UL, 0x8d0ebb40UL,
+ 0xa623e883UL, 0xbf38d9c2UL, 0x38a0c50dUL, 0x21bbf44cUL, 0x0a96a78fUL,
+ 0x138d96ceUL, 0x5ccc0009UL, 0x45d73148UL, 0x6efa628bUL, 0x77e153caUL,
+ 0xbabb5d54UL, 0xa3a06c15UL, 0x888d3fd6UL, 0x91960e97UL, 0xded79850UL,
+ 0xc7cca911UL, 0xece1fad2UL, 0xf5facb93UL, 0x7262d75cUL, 0x6b79e61dUL,
+ 0x4054b5deUL, 0x594f849fUL, 0x160e1258UL, 0x0f152319UL, 0x243870daUL,
+ 0x3d23419bUL, 0x65fd6ba7UL, 0x7ce65ae6UL, 0x57cb0925UL, 0x4ed03864UL,
+ 0x0191aea3UL, 0x188a9fe2UL, 0x33a7cc21UL, 0x2abcfd60UL, 0xad24e1afUL,
+ 0xb43fd0eeUL, 0x9f12832dUL, 0x8609b26cUL, 0xc94824abUL, 0xd05315eaUL,
+ 0xfb7e4629UL, 0xe2657768UL, 0x2f3f79f6UL, 0x362448b7UL, 0x1d091b74UL,
+ 0x04122a35UL, 0x4b53bcf2UL, 0x52488db3UL, 0x7965de70UL, 0x607eef31UL,
+ 0xe7e6f3feUL, 0xfefdc2bfUL, 0xd5d0917cUL, 0xcccba03dUL, 0x838a36faUL,
+ 0x9a9107bbUL, 0xb1bc5478UL, 0xa8a76539UL, 0x3b83984bUL, 0x2298a90aUL,
+ 0x09b5fac9UL, 0x10aecb88UL, 0x5fef5d4fUL, 0x46f46c0eUL, 0x6dd93fcdUL,
+ 0x74c20e8cUL, 0xf35a1243UL, 0xea412302UL, 0xc16c70c1UL, 0xd8774180UL,
+ 0x9736d747UL, 0x8e2de606UL, 0xa500b5c5UL, 0xbc1b8484UL, 0x71418a1aUL,
+ 0x685abb5bUL, 0x4377e898UL, 0x5a6cd9d9UL, 0x152d4f1eUL, 0x0c367e5fUL,
+ 0x271b2d9cUL, 0x3e001cddUL, 0xb9980012UL, 0xa0833153UL, 0x8bae6290UL,
+ 0x92b553d1UL, 0xddf4c516UL, 0xc4eff457UL, 0xefc2a794UL, 0xf6d996d5UL,
+ 0xae07bce9UL, 0xb71c8da8UL, 0x9c31de6bUL, 0x852aef2aUL, 0xca6b79edUL,
+ 0xd37048acUL, 0xf85d1b6fUL, 0xe1462a2eUL, 0x66de36e1UL, 0x7fc507a0UL,
+ 0x54e85463UL, 0x4df36522UL, 0x02b2f3e5UL, 0x1ba9c2a4UL, 0x30849167UL,
+ 0x299fa026UL, 0xe4c5aeb8UL, 0xfdde9ff9UL, 0xd6f3cc3aUL, 0xcfe8fd7bUL,
+ 0x80a96bbcUL, 0x99b25afdUL, 0xb29f093eUL, 0xab84387fUL, 0x2c1c24b0UL,
+ 0x350715f1UL, 0x1e2a4632UL, 0x07317773UL, 0x4870e1b4UL, 0x516bd0f5UL,
+ 0x7a468336UL, 0x635db277UL, 0xcbfad74eUL, 0xd2e1e60fUL, 0xf9ccb5ccUL,
+ 0xe0d7848dUL, 0xaf96124aUL, 0xb68d230bUL, 0x9da070c8UL, 0x84bb4189UL,
+ 0x03235d46UL, 0x1a386c07UL, 0x31153fc4UL, 0x280e0e85UL, 0x674f9842UL,
+ 0x7e54a903UL, 0x5579fac0UL, 0x4c62cb81UL, 0x8138c51fUL, 0x9823f45eUL,
+ 0xb30ea79dUL, 0xaa1596dcUL, 0xe554001bUL, 0xfc4f315aUL, 0xd7626299UL,
+ 0xce7953d8UL, 0x49e14f17UL, 0x50fa7e56UL, 0x7bd72d95UL, 0x62cc1cd4UL,
+ 0x2d8d8a13UL, 0x3496bb52UL, 0x1fbbe891UL, 0x06a0d9d0UL, 0x5e7ef3ecUL,
+ 0x4765c2adUL, 0x6c48916eUL, 0x7553a02fUL, 0x3a1236e8UL, 0x230907a9UL,
+ 0x0824546aUL, 0x113f652bUL, 0x96a779e4UL, 0x8fbc48a5UL, 0xa4911b66UL,
+ 0xbd8a2a27UL, 0xf2cbbce0UL, 0xebd08da1UL, 0xc0fdde62UL, 0xd9e6ef23UL,
+ 0x14bce1bdUL, 0x0da7d0fcUL, 0x268a833fUL, 0x3f91b27eUL, 0x70d024b9UL,
+ 0x69cb15f8UL, 0x42e6463bUL, 0x5bfd777aUL, 0xdc656bb5UL, 0xc57e5af4UL,
+ 0xee530937UL, 0xf7483876UL, 0xb809aeb1UL, 0xa1129ff0UL, 0x8a3fcc33UL,
+ 0x9324fd72UL
+ },
+ {
+ 0x00000000UL, 0x01c26a37UL, 0x0384d46eUL, 0x0246be59UL, 0x0709a8dcUL,
+ 0x06cbc2ebUL, 0x048d7cb2UL, 0x054f1685UL, 0x0e1351b8UL, 0x0fd13b8fUL,
+ 0x0d9785d6UL, 0x0c55efe1UL, 0x091af964UL, 0x08d89353UL, 0x0a9e2d0aUL,
+ 0x0b5c473dUL, 0x1c26a370UL, 0x1de4c947UL, 0x1fa2771eUL, 0x1e601d29UL,
+ 0x1b2f0bacUL, 0x1aed619bUL, 0x18abdfc2UL, 0x1969b5f5UL, 0x1235f2c8UL,
+ 0x13f798ffUL, 0x11b126a6UL, 0x10734c91UL, 0x153c5a14UL, 0x14fe3023UL,
+ 0x16b88e7aUL, 0x177ae44dUL, 0x384d46e0UL, 0x398f2cd7UL, 0x3bc9928eUL,
+ 0x3a0bf8b9UL, 0x3f44ee3cUL, 0x3e86840bUL, 0x3cc03a52UL, 0x3d025065UL,
+ 0x365e1758UL, 0x379c7d6fUL, 0x35dac336UL, 0x3418a901UL, 0x3157bf84UL,
+ 0x3095d5b3UL, 0x32d36beaUL, 0x331101ddUL, 0x246be590UL, 0x25a98fa7UL,
+ 0x27ef31feUL, 0x262d5bc9UL, 0x23624d4cUL, 0x22a0277bUL, 0x20e69922UL,
+ 0x2124f315UL, 0x2a78b428UL, 0x2bbade1fUL, 0x29fc6046UL, 0x283e0a71UL,
+ 0x2d711cf4UL, 0x2cb376c3UL, 0x2ef5c89aUL, 0x2f37a2adUL, 0x709a8dc0UL,
+ 0x7158e7f7UL, 0x731e59aeUL, 0x72dc3399UL, 0x7793251cUL, 0x76514f2bUL,
+ 0x7417f172UL, 0x75d59b45UL, 0x7e89dc78UL, 0x7f4bb64fUL, 0x7d0d0816UL,
+ 0x7ccf6221UL, 0x798074a4UL, 0x78421e93UL, 0x7a04a0caUL, 0x7bc6cafdUL,
+ 0x6cbc2eb0UL, 0x6d7e4487UL, 0x6f38fadeUL, 0x6efa90e9UL, 0x6bb5866cUL,
+ 0x6a77ec5bUL, 0x68315202UL, 0x69f33835UL, 0x62af7f08UL, 0x636d153fUL,
+ 0x612bab66UL, 0x60e9c151UL, 0x65a6d7d4UL, 0x6464bde3UL, 0x662203baUL,
+ 0x67e0698dUL, 0x48d7cb20UL, 0x4915a117UL, 0x4b531f4eUL, 0x4a917579UL,
+ 0x4fde63fcUL, 0x4e1c09cbUL, 0x4c5ab792UL, 0x4d98dda5UL, 0x46c49a98UL,
+ 0x4706f0afUL, 0x45404ef6UL, 0x448224c1UL, 0x41cd3244UL, 0x400f5873UL,
+ 0x4249e62aUL, 0x438b8c1dUL, 0x54f16850UL, 0x55330267UL, 0x5775bc3eUL,
+ 0x56b7d609UL, 0x53f8c08cUL, 0x523aaabbUL, 0x507c14e2UL, 0x51be7ed5UL,
+ 0x5ae239e8UL, 0x5b2053dfUL, 0x5966ed86UL, 0x58a487b1UL, 0x5deb9134UL,
+ 0x5c29fb03UL, 0x5e6f455aUL, 0x5fad2f6dUL, 0xe1351b80UL, 0xe0f771b7UL,
+ 0xe2b1cfeeUL, 0xe373a5d9UL, 0xe63cb35cUL, 0xe7fed96bUL, 0xe5b86732UL,
+ 0xe47a0d05UL, 0xef264a38UL, 0xeee4200fUL, 0xeca29e56UL, 0xed60f461UL,
+ 0xe82fe2e4UL, 0xe9ed88d3UL, 0xebab368aUL, 0xea695cbdUL, 0xfd13b8f0UL,
+ 0xfcd1d2c7UL, 0xfe976c9eUL, 0xff5506a9UL, 0xfa1a102cUL, 0xfbd87a1bUL,
+ 0xf99ec442UL, 0xf85cae75UL, 0xf300e948UL, 0xf2c2837fUL, 0xf0843d26UL,
+ 0xf1465711UL, 0xf4094194UL, 0xf5cb2ba3UL, 0xf78d95faUL, 0xf64fffcdUL,
+ 0xd9785d60UL, 0xd8ba3757UL, 0xdafc890eUL, 0xdb3ee339UL, 0xde71f5bcUL,
+ 0xdfb39f8bUL, 0xddf521d2UL, 0xdc374be5UL, 0xd76b0cd8UL, 0xd6a966efUL,
+ 0xd4efd8b6UL, 0xd52db281UL, 0xd062a404UL, 0xd1a0ce33UL, 0xd3e6706aUL,
+ 0xd2241a5dUL, 0xc55efe10UL, 0xc49c9427UL, 0xc6da2a7eUL, 0xc7184049UL,
+ 0xc25756ccUL, 0xc3953cfbUL, 0xc1d382a2UL, 0xc011e895UL, 0xcb4dafa8UL,
+ 0xca8fc59fUL, 0xc8c97bc6UL, 0xc90b11f1UL, 0xcc440774UL, 0xcd866d43UL,
+ 0xcfc0d31aUL, 0xce02b92dUL, 0x91af9640UL, 0x906dfc77UL, 0x922b422eUL,
+ 0x93e92819UL, 0x96a63e9cUL, 0x976454abUL, 0x9522eaf2UL, 0x94e080c5UL,
+ 0x9fbcc7f8UL, 0x9e7eadcfUL, 0x9c381396UL, 0x9dfa79a1UL, 0x98b56f24UL,
+ 0x99770513UL, 0x9b31bb4aUL, 0x9af3d17dUL, 0x8d893530UL, 0x8c4b5f07UL,
+ 0x8e0de15eUL, 0x8fcf8b69UL, 0x8a809decUL, 0x8b42f7dbUL, 0x89044982UL,
+ 0x88c623b5UL, 0x839a6488UL, 0x82580ebfUL, 0x801eb0e6UL, 0x81dcdad1UL,
+ 0x8493cc54UL, 0x8551a663UL, 0x8717183aUL, 0x86d5720dUL, 0xa9e2d0a0UL,
+ 0xa820ba97UL, 0xaa6604ceUL, 0xaba46ef9UL, 0xaeeb787cUL, 0xaf29124bUL,
+ 0xad6fac12UL, 0xacadc625UL, 0xa7f18118UL, 0xa633eb2fUL, 0xa4755576UL,
+ 0xa5b73f41UL, 0xa0f829c4UL, 0xa13a43f3UL, 0xa37cfdaaUL, 0xa2be979dUL,
+ 0xb5c473d0UL, 0xb40619e7UL, 0xb640a7beUL, 0xb782cd89UL, 0xb2cddb0cUL,
+ 0xb30fb13bUL, 0xb1490f62UL, 0xb08b6555UL, 0xbbd72268UL, 0xba15485fUL,
+ 0xb853f606UL, 0xb9919c31UL, 0xbcde8ab4UL, 0xbd1ce083UL, 0xbf5a5edaUL,
+ 0xbe9834edUL
+ },
+ {
+ 0x00000000UL, 0xb8bc6765UL, 0xaa09c88bUL, 0x12b5afeeUL, 0x8f629757UL,
+ 0x37def032UL, 0x256b5fdcUL, 0x9dd738b9UL, 0xc5b428efUL, 0x7d084f8aUL,
+ 0x6fbde064UL, 0xd7018701UL, 0x4ad6bfb8UL, 0xf26ad8ddUL, 0xe0df7733UL,
+ 0x58631056UL, 0x5019579fUL, 0xe8a530faUL, 0xfa109f14UL, 0x42acf871UL,
+ 0xdf7bc0c8UL, 0x67c7a7adUL, 0x75720843UL, 0xcdce6f26UL, 0x95ad7f70UL,
+ 0x2d111815UL, 0x3fa4b7fbUL, 0x8718d09eUL, 0x1acfe827UL, 0xa2738f42UL,
+ 0xb0c620acUL, 0x087a47c9UL, 0xa032af3eUL, 0x188ec85bUL, 0x0a3b67b5UL,
+ 0xb28700d0UL, 0x2f503869UL, 0x97ec5f0cUL, 0x8559f0e2UL, 0x3de59787UL,
+ 0x658687d1UL, 0xdd3ae0b4UL, 0xcf8f4f5aUL, 0x7733283fUL, 0xeae41086UL,
+ 0x525877e3UL, 0x40edd80dUL, 0xf851bf68UL, 0xf02bf8a1UL, 0x48979fc4UL,
+ 0x5a22302aUL, 0xe29e574fUL, 0x7f496ff6UL, 0xc7f50893UL, 0xd540a77dUL,
+ 0x6dfcc018UL, 0x359fd04eUL, 0x8d23b72bUL, 0x9f9618c5UL, 0x272a7fa0UL,
+ 0xbafd4719UL, 0x0241207cUL, 0x10f48f92UL, 0xa848e8f7UL, 0x9b14583dUL,
+ 0x23a83f58UL, 0x311d90b6UL, 0x89a1f7d3UL, 0x1476cf6aUL, 0xaccaa80fUL,
+ 0xbe7f07e1UL, 0x06c36084UL, 0x5ea070d2UL, 0xe61c17b7UL, 0xf4a9b859UL,
+ 0x4c15df3cUL, 0xd1c2e785UL, 0x697e80e0UL, 0x7bcb2f0eUL, 0xc377486bUL,
+ 0xcb0d0fa2UL, 0x73b168c7UL, 0x6104c729UL, 0xd9b8a04cUL, 0x446f98f5UL,
+ 0xfcd3ff90UL, 0xee66507eUL, 0x56da371bUL, 0x0eb9274dUL, 0xb6054028UL,
+ 0xa4b0efc6UL, 0x1c0c88a3UL, 0x81dbb01aUL, 0x3967d77fUL, 0x2bd27891UL,
+ 0x936e1ff4UL, 0x3b26f703UL, 0x839a9066UL, 0x912f3f88UL, 0x299358edUL,
+ 0xb4446054UL, 0x0cf80731UL, 0x1e4da8dfUL, 0xa6f1cfbaUL, 0xfe92dfecUL,
+ 0x462eb889UL, 0x549b1767UL, 0xec277002UL, 0x71f048bbUL, 0xc94c2fdeUL,
+ 0xdbf98030UL, 0x6345e755UL, 0x6b3fa09cUL, 0xd383c7f9UL, 0xc1366817UL,
+ 0x798a0f72UL, 0xe45d37cbUL, 0x5ce150aeUL, 0x4e54ff40UL, 0xf6e89825UL,
+ 0xae8b8873UL, 0x1637ef16UL, 0x048240f8UL, 0xbc3e279dUL, 0x21e91f24UL,
+ 0x99557841UL, 0x8be0d7afUL, 0x335cb0caUL, 0xed59b63bUL, 0x55e5d15eUL,
+ 0x47507eb0UL, 0xffec19d5UL, 0x623b216cUL, 0xda874609UL, 0xc832e9e7UL,
+ 0x708e8e82UL, 0x28ed9ed4UL, 0x9051f9b1UL, 0x82e4565fUL, 0x3a58313aUL,
+ 0xa78f0983UL, 0x1f336ee6UL, 0x0d86c108UL, 0xb53aa66dUL, 0xbd40e1a4UL,
+ 0x05fc86c1UL, 0x1749292fUL, 0xaff54e4aUL, 0x322276f3UL, 0x8a9e1196UL,
+ 0x982bbe78UL, 0x2097d91dUL, 0x78f4c94bUL, 0xc048ae2eUL, 0xd2fd01c0UL,
+ 0x6a4166a5UL, 0xf7965e1cUL, 0x4f2a3979UL, 0x5d9f9697UL, 0xe523f1f2UL,
+ 0x4d6b1905UL, 0xf5d77e60UL, 0xe762d18eUL, 0x5fdeb6ebUL, 0xc2098e52UL,
+ 0x7ab5e937UL, 0x680046d9UL, 0xd0bc21bcUL, 0x88df31eaUL, 0x3063568fUL,
+ 0x22d6f961UL, 0x9a6a9e04UL, 0x07bda6bdUL, 0xbf01c1d8UL, 0xadb46e36UL,
+ 0x15080953UL, 0x1d724e9aUL, 0xa5ce29ffUL, 0xb77b8611UL, 0x0fc7e174UL,
+ 0x9210d9cdUL, 0x2aacbea8UL, 0x38191146UL, 0x80a57623UL, 0xd8c66675UL,
+ 0x607a0110UL, 0x72cfaefeUL, 0xca73c99bUL, 0x57a4f122UL, 0xef189647UL,
+ 0xfdad39a9UL, 0x45115eccUL, 0x764dee06UL, 0xcef18963UL, 0xdc44268dUL,
+ 0x64f841e8UL, 0xf92f7951UL, 0x41931e34UL, 0x5326b1daUL, 0xeb9ad6bfUL,
+ 0xb3f9c6e9UL, 0x0b45a18cUL, 0x19f00e62UL, 0xa14c6907UL, 0x3c9b51beUL,
+ 0x842736dbUL, 0x96929935UL, 0x2e2efe50UL, 0x2654b999UL, 0x9ee8defcUL,
+ 0x8c5d7112UL, 0x34e11677UL, 0xa9362eceUL, 0x118a49abUL, 0x033fe645UL,
+ 0xbb838120UL, 0xe3e09176UL, 0x5b5cf613UL, 0x49e959fdUL, 0xf1553e98UL,
+ 0x6c820621UL, 0xd43e6144UL, 0xc68bceaaUL, 0x7e37a9cfUL, 0xd67f4138UL,
+ 0x6ec3265dUL, 0x7c7689b3UL, 0xc4caeed6UL, 0x591dd66fUL, 0xe1a1b10aUL,
+ 0xf3141ee4UL, 0x4ba87981UL, 0x13cb69d7UL, 0xab770eb2UL, 0xb9c2a15cUL,
+ 0x017ec639UL, 0x9ca9fe80UL, 0x241599e5UL, 0x36a0360bUL, 0x8e1c516eUL,
+ 0x866616a7UL, 0x3eda71c2UL, 0x2c6fde2cUL, 0x94d3b949UL, 0x090481f0UL,
+ 0xb1b8e695UL, 0xa30d497bUL, 0x1bb12e1eUL, 0x43d23e48UL, 0xfb6e592dUL,
+ 0xe9dbf6c3UL, 0x516791a6UL, 0xccb0a91fUL, 0x740cce7aUL, 0x66b96194UL,
+ 0xde0506f1UL
+ },
+ {
+ 0x00000000UL, 0x96300777UL, 0x2c610eeeUL, 0xba510999UL, 0x19c46d07UL,
+ 0x8ff46a70UL, 0x35a563e9UL, 0xa395649eUL, 0x3288db0eUL, 0xa4b8dc79UL,
+ 0x1ee9d5e0UL, 0x88d9d297UL, 0x2b4cb609UL, 0xbd7cb17eUL, 0x072db8e7UL,
+ 0x911dbf90UL, 0x6410b71dUL, 0xf220b06aUL, 0x4871b9f3UL, 0xde41be84UL,
+ 0x7dd4da1aUL, 0xebe4dd6dUL, 0x51b5d4f4UL, 0xc785d383UL, 0x56986c13UL,
+ 0xc0a86b64UL, 0x7af962fdUL, 0xecc9658aUL, 0x4f5c0114UL, 0xd96c0663UL,
+ 0x633d0ffaUL, 0xf50d088dUL, 0xc8206e3bUL, 0x5e10694cUL, 0xe44160d5UL,
+ 0x727167a2UL, 0xd1e4033cUL, 0x47d4044bUL, 0xfd850dd2UL, 0x6bb50aa5UL,
+ 0xfaa8b535UL, 0x6c98b242UL, 0xd6c9bbdbUL, 0x40f9bcacUL, 0xe36cd832UL,
+ 0x755cdf45UL, 0xcf0dd6dcUL, 0x593dd1abUL, 0xac30d926UL, 0x3a00de51UL,
+ 0x8051d7c8UL, 0x1661d0bfUL, 0xb5f4b421UL, 0x23c4b356UL, 0x9995bacfUL,
+ 0x0fa5bdb8UL, 0x9eb80228UL, 0x0888055fUL, 0xb2d90cc6UL, 0x24e90bb1UL,
+ 0x877c6f2fUL, 0x114c6858UL, 0xab1d61c1UL, 0x3d2d66b6UL, 0x9041dc76UL,
+ 0x0671db01UL, 0xbc20d298UL, 0x2a10d5efUL, 0x8985b171UL, 0x1fb5b606UL,
+ 0xa5e4bf9fUL, 0x33d4b8e8UL, 0xa2c90778UL, 0x34f9000fUL, 0x8ea80996UL,
+ 0x18980ee1UL, 0xbb0d6a7fUL, 0x2d3d6d08UL, 0x976c6491UL, 0x015c63e6UL,
+ 0xf4516b6bUL, 0x62616c1cUL, 0xd8306585UL, 0x4e0062f2UL, 0xed95066cUL,
+ 0x7ba5011bUL, 0xc1f40882UL, 0x57c40ff5UL, 0xc6d9b065UL, 0x50e9b712UL,
+ 0xeab8be8bUL, 0x7c88b9fcUL, 0xdf1ddd62UL, 0x492dda15UL, 0xf37cd38cUL,
+ 0x654cd4fbUL, 0x5861b24dUL, 0xce51b53aUL, 0x7400bca3UL, 0xe230bbd4UL,
+ 0x41a5df4aUL, 0xd795d83dUL, 0x6dc4d1a4UL, 0xfbf4d6d3UL, 0x6ae96943UL,
+ 0xfcd96e34UL, 0x468867adUL, 0xd0b860daUL, 0x732d0444UL, 0xe51d0333UL,
+ 0x5f4c0aaaUL, 0xc97c0dddUL, 0x3c710550UL, 0xaa410227UL, 0x10100bbeUL,
+ 0x86200cc9UL, 0x25b56857UL, 0xb3856f20UL, 0x09d466b9UL, 0x9fe461ceUL,
+ 0x0ef9de5eUL, 0x98c9d929UL, 0x2298d0b0UL, 0xb4a8d7c7UL, 0x173db359UL,
+ 0x810db42eUL, 0x3b5cbdb7UL, 0xad6cbac0UL, 0x2083b8edUL, 0xb6b3bf9aUL,
+ 0x0ce2b603UL, 0x9ad2b174UL, 0x3947d5eaUL, 0xaf77d29dUL, 0x1526db04UL,
+ 0x8316dc73UL, 0x120b63e3UL, 0x843b6494UL, 0x3e6a6d0dUL, 0xa85a6a7aUL,
+ 0x0bcf0ee4UL, 0x9dff0993UL, 0x27ae000aUL, 0xb19e077dUL, 0x44930ff0UL,
+ 0xd2a30887UL, 0x68f2011eUL, 0xfec20669UL, 0x5d5762f7UL, 0xcb676580UL,
+ 0x71366c19UL, 0xe7066b6eUL, 0x761bd4feUL, 0xe02bd389UL, 0x5a7ada10UL,
+ 0xcc4add67UL, 0x6fdfb9f9UL, 0xf9efbe8eUL, 0x43beb717UL, 0xd58eb060UL,
+ 0xe8a3d6d6UL, 0x7e93d1a1UL, 0xc4c2d838UL, 0x52f2df4fUL, 0xf167bbd1UL,
+ 0x6757bca6UL, 0xdd06b53fUL, 0x4b36b248UL, 0xda2b0dd8UL, 0x4c1b0aafUL,
+ 0xf64a0336UL, 0x607a0441UL, 0xc3ef60dfUL, 0x55df67a8UL, 0xef8e6e31UL,
+ 0x79be6946UL, 0x8cb361cbUL, 0x1a8366bcUL, 0xa0d26f25UL, 0x36e26852UL,
+ 0x95770cccUL, 0x03470bbbUL, 0xb9160222UL, 0x2f260555UL, 0xbe3bbac5UL,
+ 0x280bbdb2UL, 0x925ab42bUL, 0x046ab35cUL, 0xa7ffd7c2UL, 0x31cfd0b5UL,
+ 0x8b9ed92cUL, 0x1daede5bUL, 0xb0c2649bUL, 0x26f263ecUL, 0x9ca36a75UL,
+ 0x0a936d02UL, 0xa906099cUL, 0x3f360eebUL, 0x85670772UL, 0x13570005UL,
+ 0x824abf95UL, 0x147ab8e2UL, 0xae2bb17bUL, 0x381bb60cUL, 0x9b8ed292UL,
+ 0x0dbed5e5UL, 0xb7efdc7cUL, 0x21dfdb0bUL, 0xd4d2d386UL, 0x42e2d4f1UL,
+ 0xf8b3dd68UL, 0x6e83da1fUL, 0xcd16be81UL, 0x5b26b9f6UL, 0xe177b06fUL,
+ 0x7747b718UL, 0xe65a0888UL, 0x706a0fffUL, 0xca3b0666UL, 0x5c0b0111UL,
+ 0xff9e658fUL, 0x69ae62f8UL, 0xd3ff6b61UL, 0x45cf6c16UL, 0x78e20aa0UL,
+ 0xeed20dd7UL, 0x5483044eUL, 0xc2b30339UL, 0x612667a7UL, 0xf71660d0UL,
+ 0x4d476949UL, 0xdb776e3eUL, 0x4a6ad1aeUL, 0xdc5ad6d9UL, 0x660bdf40UL,
+ 0xf03bd837UL, 0x53aebca9UL, 0xc59ebbdeUL, 0x7fcfb247UL, 0xe9ffb530UL,
+ 0x1cf2bdbdUL, 0x8ac2bacaUL, 0x3093b353UL, 0xa6a3b424UL, 0x0536d0baUL,
+ 0x9306d7cdUL, 0x2957de54UL, 0xbf67d923UL, 0x2e7a66b3UL, 0xb84a61c4UL,
+ 0x021b685dUL, 0x942b6f2aUL, 0x37be0bb4UL, 0xa18e0cc3UL, 0x1bdf055aUL,
+ 0x8def022dUL
+ },
+ {
+ 0x00000000UL, 0x41311b19UL, 0x82623632UL, 0xc3532d2bUL, 0x04c56c64UL,
+ 0x45f4777dUL, 0x86a75a56UL, 0xc796414fUL, 0x088ad9c8UL, 0x49bbc2d1UL,
+ 0x8ae8effaUL, 0xcbd9f4e3UL, 0x0c4fb5acUL, 0x4d7eaeb5UL, 0x8e2d839eUL,
+ 0xcf1c9887UL, 0x5112c24aUL, 0x1023d953UL, 0xd370f478UL, 0x9241ef61UL,
+ 0x55d7ae2eUL, 0x14e6b537UL, 0xd7b5981cUL, 0x96848305UL, 0x59981b82UL,
+ 0x18a9009bUL, 0xdbfa2db0UL, 0x9acb36a9UL, 0x5d5d77e6UL, 0x1c6c6cffUL,
+ 0xdf3f41d4UL, 0x9e0e5acdUL, 0xa2248495UL, 0xe3159f8cUL, 0x2046b2a7UL,
+ 0x6177a9beUL, 0xa6e1e8f1UL, 0xe7d0f3e8UL, 0x2483dec3UL, 0x65b2c5daUL,
+ 0xaaae5d5dUL, 0xeb9f4644UL, 0x28cc6b6fUL, 0x69fd7076UL, 0xae6b3139UL,
+ 0xef5a2a20UL, 0x2c09070bUL, 0x6d381c12UL, 0xf33646dfUL, 0xb2075dc6UL,
+ 0x715470edUL, 0x30656bf4UL, 0xf7f32abbUL, 0xb6c231a2UL, 0x75911c89UL,
+ 0x34a00790UL, 0xfbbc9f17UL, 0xba8d840eUL, 0x79dea925UL, 0x38efb23cUL,
+ 0xff79f373UL, 0xbe48e86aUL, 0x7d1bc541UL, 0x3c2ade58UL, 0x054f79f0UL,
+ 0x447e62e9UL, 0x872d4fc2UL, 0xc61c54dbUL, 0x018a1594UL, 0x40bb0e8dUL,
+ 0x83e823a6UL, 0xc2d938bfUL, 0x0dc5a038UL, 0x4cf4bb21UL, 0x8fa7960aUL,
+ 0xce968d13UL, 0x0900cc5cUL, 0x4831d745UL, 0x8b62fa6eUL, 0xca53e177UL,
+ 0x545dbbbaUL, 0x156ca0a3UL, 0xd63f8d88UL, 0x970e9691UL, 0x5098d7deUL,
+ 0x11a9ccc7UL, 0xd2fae1ecUL, 0x93cbfaf5UL, 0x5cd76272UL, 0x1de6796bUL,
+ 0xdeb55440UL, 0x9f844f59UL, 0x58120e16UL, 0x1923150fUL, 0xda703824UL,
+ 0x9b41233dUL, 0xa76bfd65UL, 0xe65ae67cUL, 0x2509cb57UL, 0x6438d04eUL,
+ 0xa3ae9101UL, 0xe29f8a18UL, 0x21cca733UL, 0x60fdbc2aUL, 0xafe124adUL,
+ 0xeed03fb4UL, 0x2d83129fUL, 0x6cb20986UL, 0xab2448c9UL, 0xea1553d0UL,
+ 0x29467efbUL, 0x687765e2UL, 0xf6793f2fUL, 0xb7482436UL, 0x741b091dUL,
+ 0x352a1204UL, 0xf2bc534bUL, 0xb38d4852UL, 0x70de6579UL, 0x31ef7e60UL,
+ 0xfef3e6e7UL, 0xbfc2fdfeUL, 0x7c91d0d5UL, 0x3da0cbccUL, 0xfa368a83UL,
+ 0xbb07919aUL, 0x7854bcb1UL, 0x3965a7a8UL, 0x4b98833bUL, 0x0aa99822UL,
+ 0xc9fab509UL, 0x88cbae10UL, 0x4f5def5fUL, 0x0e6cf446UL, 0xcd3fd96dUL,
+ 0x8c0ec274UL, 0x43125af3UL, 0x022341eaUL, 0xc1706cc1UL, 0x804177d8UL,
+ 0x47d73697UL, 0x06e62d8eUL, 0xc5b500a5UL, 0x84841bbcUL, 0x1a8a4171UL,
+ 0x5bbb5a68UL, 0x98e87743UL, 0xd9d96c5aUL, 0x1e4f2d15UL, 0x5f7e360cUL,
+ 0x9c2d1b27UL, 0xdd1c003eUL, 0x120098b9UL, 0x533183a0UL, 0x9062ae8bUL,
+ 0xd153b592UL, 0x16c5f4ddUL, 0x57f4efc4UL, 0x94a7c2efUL, 0xd596d9f6UL,
+ 0xe9bc07aeUL, 0xa88d1cb7UL, 0x6bde319cUL, 0x2aef2a85UL, 0xed796bcaUL,
+ 0xac4870d3UL, 0x6f1b5df8UL, 0x2e2a46e1UL, 0xe136de66UL, 0xa007c57fUL,
+ 0x6354e854UL, 0x2265f34dUL, 0xe5f3b202UL, 0xa4c2a91bUL, 0x67918430UL,
+ 0x26a09f29UL, 0xb8aec5e4UL, 0xf99fdefdUL, 0x3accf3d6UL, 0x7bfde8cfUL,
+ 0xbc6ba980UL, 0xfd5ab299UL, 0x3e099fb2UL, 0x7f3884abUL, 0xb0241c2cUL,
+ 0xf1150735UL, 0x32462a1eUL, 0x73773107UL, 0xb4e17048UL, 0xf5d06b51UL,
+ 0x3683467aUL, 0x77b25d63UL, 0x4ed7facbUL, 0x0fe6e1d2UL, 0xccb5ccf9UL,
+ 0x8d84d7e0UL, 0x4a1296afUL, 0x0b238db6UL, 0xc870a09dUL, 0x8941bb84UL,
+ 0x465d2303UL, 0x076c381aUL, 0xc43f1531UL, 0x850e0e28UL, 0x42984f67UL,
+ 0x03a9547eUL, 0xc0fa7955UL, 0x81cb624cUL, 0x1fc53881UL, 0x5ef42398UL,
+ 0x9da70eb3UL, 0xdc9615aaUL, 0x1b0054e5UL, 0x5a314ffcUL, 0x996262d7UL,
+ 0xd85379ceUL, 0x174fe149UL, 0x567efa50UL, 0x952dd77bUL, 0xd41ccc62UL,
+ 0x138a8d2dUL, 0x52bb9634UL, 0x91e8bb1fUL, 0xd0d9a006UL, 0xecf37e5eUL,
+ 0xadc26547UL, 0x6e91486cUL, 0x2fa05375UL, 0xe836123aUL, 0xa9070923UL,
+ 0x6a542408UL, 0x2b653f11UL, 0xe479a796UL, 0xa548bc8fUL, 0x661b91a4UL,
+ 0x272a8abdUL, 0xe0bccbf2UL, 0xa18dd0ebUL, 0x62defdc0UL, 0x23efe6d9UL,
+ 0xbde1bc14UL, 0xfcd0a70dUL, 0x3f838a26UL, 0x7eb2913fUL, 0xb924d070UL,
+ 0xf815cb69UL, 0x3b46e642UL, 0x7a77fd5bUL, 0xb56b65dcUL, 0xf45a7ec5UL,
+ 0x370953eeUL, 0x763848f7UL, 0xb1ae09b8UL, 0xf09f12a1UL, 0x33cc3f8aUL,
+ 0x72fd2493UL
+ },
+ {
+ 0x00000000UL, 0x376ac201UL, 0x6ed48403UL, 0x59be4602UL, 0xdca80907UL,
+ 0xebc2cb06UL, 0xb27c8d04UL, 0x85164f05UL, 0xb851130eUL, 0x8f3bd10fUL,
+ 0xd685970dUL, 0xe1ef550cUL, 0x64f91a09UL, 0x5393d808UL, 0x0a2d9e0aUL,
+ 0x3d475c0bUL, 0x70a3261cUL, 0x47c9e41dUL, 0x1e77a21fUL, 0x291d601eUL,
+ 0xac0b2f1bUL, 0x9b61ed1aUL, 0xc2dfab18UL, 0xf5b56919UL, 0xc8f23512UL,
+ 0xff98f713UL, 0xa626b111UL, 0x914c7310UL, 0x145a3c15UL, 0x2330fe14UL,
+ 0x7a8eb816UL, 0x4de47a17UL, 0xe0464d38UL, 0xd72c8f39UL, 0x8e92c93bUL,
+ 0xb9f80b3aUL, 0x3cee443fUL, 0x0b84863eUL, 0x523ac03cUL, 0x6550023dUL,
+ 0x58175e36UL, 0x6f7d9c37UL, 0x36c3da35UL, 0x01a91834UL, 0x84bf5731UL,
+ 0xb3d59530UL, 0xea6bd332UL, 0xdd011133UL, 0x90e56b24UL, 0xa78fa925UL,
+ 0xfe31ef27UL, 0xc95b2d26UL, 0x4c4d6223UL, 0x7b27a022UL, 0x2299e620UL,
+ 0x15f32421UL, 0x28b4782aUL, 0x1fdeba2bUL, 0x4660fc29UL, 0x710a3e28UL,
+ 0xf41c712dUL, 0xc376b32cUL, 0x9ac8f52eUL, 0xada2372fUL, 0xc08d9a70UL,
+ 0xf7e75871UL, 0xae591e73UL, 0x9933dc72UL, 0x1c259377UL, 0x2b4f5176UL,
+ 0x72f11774UL, 0x459bd575UL, 0x78dc897eUL, 0x4fb64b7fUL, 0x16080d7dUL,
+ 0x2162cf7cUL, 0xa4748079UL, 0x931e4278UL, 0xcaa0047aUL, 0xfdcac67bUL,
+ 0xb02ebc6cUL, 0x87447e6dUL, 0xdefa386fUL, 0xe990fa6eUL, 0x6c86b56bUL,
+ 0x5bec776aUL, 0x02523168UL, 0x3538f369UL, 0x087faf62UL, 0x3f156d63UL,
+ 0x66ab2b61UL, 0x51c1e960UL, 0xd4d7a665UL, 0xe3bd6464UL, 0xba032266UL,
+ 0x8d69e067UL, 0x20cbd748UL, 0x17a11549UL, 0x4e1f534bUL, 0x7975914aUL,
+ 0xfc63de4fUL, 0xcb091c4eUL, 0x92b75a4cUL, 0xa5dd984dUL, 0x989ac446UL,
+ 0xaff00647UL, 0xf64e4045UL, 0xc1248244UL, 0x4432cd41UL, 0x73580f40UL,
+ 0x2ae64942UL, 0x1d8c8b43UL, 0x5068f154UL, 0x67023355UL, 0x3ebc7557UL,
+ 0x09d6b756UL, 0x8cc0f853UL, 0xbbaa3a52UL, 0xe2147c50UL, 0xd57ebe51UL,
+ 0xe839e25aUL, 0xdf53205bUL, 0x86ed6659UL, 0xb187a458UL, 0x3491eb5dUL,
+ 0x03fb295cUL, 0x5a456f5eUL, 0x6d2fad5fUL, 0x801b35e1UL, 0xb771f7e0UL,
+ 0xeecfb1e2UL, 0xd9a573e3UL, 0x5cb33ce6UL, 0x6bd9fee7UL, 0x3267b8e5UL,
+ 0x050d7ae4UL, 0x384a26efUL, 0x0f20e4eeUL, 0x569ea2ecUL, 0x61f460edUL,
+ 0xe4e22fe8UL, 0xd388ede9UL, 0x8a36abebUL, 0xbd5c69eaUL, 0xf0b813fdUL,
+ 0xc7d2d1fcUL, 0x9e6c97feUL, 0xa90655ffUL, 0x2c101afaUL, 0x1b7ad8fbUL,
+ 0x42c49ef9UL, 0x75ae5cf8UL, 0x48e900f3UL, 0x7f83c2f2UL, 0x263d84f0UL,
+ 0x115746f1UL, 0x944109f4UL, 0xa32bcbf5UL, 0xfa958df7UL, 0xcdff4ff6UL,
+ 0x605d78d9UL, 0x5737bad8UL, 0x0e89fcdaUL, 0x39e33edbUL, 0xbcf571deUL,
+ 0x8b9fb3dfUL, 0xd221f5ddUL, 0xe54b37dcUL, 0xd80c6bd7UL, 0xef66a9d6UL,
+ 0xb6d8efd4UL, 0x81b22dd5UL, 0x04a462d0UL, 0x33cea0d1UL, 0x6a70e6d3UL,
+ 0x5d1a24d2UL, 0x10fe5ec5UL, 0x27949cc4UL, 0x7e2adac6UL, 0x494018c7UL,
+ 0xcc5657c2UL, 0xfb3c95c3UL, 0xa282d3c1UL, 0x95e811c0UL, 0xa8af4dcbUL,
+ 0x9fc58fcaUL, 0xc67bc9c8UL, 0xf1110bc9UL, 0x740744ccUL, 0x436d86cdUL,
+ 0x1ad3c0cfUL, 0x2db902ceUL, 0x4096af91UL, 0x77fc6d90UL, 0x2e422b92UL,
+ 0x1928e993UL, 0x9c3ea696UL, 0xab546497UL, 0xf2ea2295UL, 0xc580e094UL,
+ 0xf8c7bc9fUL, 0xcfad7e9eUL, 0x9613389cUL, 0xa179fa9dUL, 0x246fb598UL,
+ 0x13057799UL, 0x4abb319bUL, 0x7dd1f39aUL, 0x3035898dUL, 0x075f4b8cUL,
+ 0x5ee10d8eUL, 0x698bcf8fUL, 0xec9d808aUL, 0xdbf7428bUL, 0x82490489UL,
+ 0xb523c688UL, 0x88649a83UL, 0xbf0e5882UL, 0xe6b01e80UL, 0xd1dadc81UL,
+ 0x54cc9384UL, 0x63a65185UL, 0x3a181787UL, 0x0d72d586UL, 0xa0d0e2a9UL,
+ 0x97ba20a8UL, 0xce0466aaUL, 0xf96ea4abUL, 0x7c78ebaeUL, 0x4b1229afUL,
+ 0x12ac6fadUL, 0x25c6adacUL, 0x1881f1a7UL, 0x2feb33a6UL, 0x765575a4UL,
+ 0x413fb7a5UL, 0xc429f8a0UL, 0xf3433aa1UL, 0xaafd7ca3UL, 0x9d97bea2UL,
+ 0xd073c4b5UL, 0xe71906b4UL, 0xbea740b6UL, 0x89cd82b7UL, 0x0cdbcdb2UL,
+ 0x3bb10fb3UL, 0x620f49b1UL, 0x55658bb0UL, 0x6822d7bbUL, 0x5f4815baUL,
+ 0x06f653b8UL, 0x319c91b9UL, 0xb48adebcUL, 0x83e01cbdUL, 0xda5e5abfUL,
+ 0xed3498beUL
+ },
+ {
+ 0x00000000UL, 0x6567bcb8UL, 0x8bc809aaUL, 0xeeafb512UL, 0x5797628fUL,
+ 0x32f0de37UL, 0xdc5f6b25UL, 0xb938d79dUL, 0xef28b4c5UL, 0x8a4f087dUL,
+ 0x64e0bd6fUL, 0x018701d7UL, 0xb8bfd64aUL, 0xddd86af2UL, 0x3377dfe0UL,
+ 0x56106358UL, 0x9f571950UL, 0xfa30a5e8UL, 0x149f10faUL, 0x71f8ac42UL,
+ 0xc8c07bdfUL, 0xada7c767UL, 0x43087275UL, 0x266fcecdUL, 0x707fad95UL,
+ 0x1518112dUL, 0xfbb7a43fUL, 0x9ed01887UL, 0x27e8cf1aUL, 0x428f73a2UL,
+ 0xac20c6b0UL, 0xc9477a08UL, 0x3eaf32a0UL, 0x5bc88e18UL, 0xb5673b0aUL,
+ 0xd00087b2UL, 0x6938502fUL, 0x0c5fec97UL, 0xe2f05985UL, 0x8797e53dUL,
+ 0xd1878665UL, 0xb4e03addUL, 0x5a4f8fcfUL, 0x3f283377UL, 0x8610e4eaUL,
+ 0xe3775852UL, 0x0dd8ed40UL, 0x68bf51f8UL, 0xa1f82bf0UL, 0xc49f9748UL,
+ 0x2a30225aUL, 0x4f579ee2UL, 0xf66f497fUL, 0x9308f5c7UL, 0x7da740d5UL,
+ 0x18c0fc6dUL, 0x4ed09f35UL, 0x2bb7238dUL, 0xc518969fUL, 0xa07f2a27UL,
+ 0x1947fdbaUL, 0x7c204102UL, 0x928ff410UL, 0xf7e848a8UL, 0x3d58149bUL,
+ 0x583fa823UL, 0xb6901d31UL, 0xd3f7a189UL, 0x6acf7614UL, 0x0fa8caacUL,
+ 0xe1077fbeUL, 0x8460c306UL, 0xd270a05eUL, 0xb7171ce6UL, 0x59b8a9f4UL,
+ 0x3cdf154cUL, 0x85e7c2d1UL, 0xe0807e69UL, 0x0e2fcb7bUL, 0x6b4877c3UL,
+ 0xa20f0dcbUL, 0xc768b173UL, 0x29c70461UL, 0x4ca0b8d9UL, 0xf5986f44UL,
+ 0x90ffd3fcUL, 0x7e5066eeUL, 0x1b37da56UL, 0x4d27b90eUL, 0x284005b6UL,
+ 0xc6efb0a4UL, 0xa3880c1cUL, 0x1ab0db81UL, 0x7fd76739UL, 0x9178d22bUL,
+ 0xf41f6e93UL, 0x03f7263bUL, 0x66909a83UL, 0x883f2f91UL, 0xed589329UL,
+ 0x546044b4UL, 0x3107f80cUL, 0xdfa84d1eUL, 0xbacff1a6UL, 0xecdf92feUL,
+ 0x89b82e46UL, 0x67179b54UL, 0x027027ecUL, 0xbb48f071UL, 0xde2f4cc9UL,
+ 0x3080f9dbUL, 0x55e74563UL, 0x9ca03f6bUL, 0xf9c783d3UL, 0x176836c1UL,
+ 0x720f8a79UL, 0xcb375de4UL, 0xae50e15cUL, 0x40ff544eUL, 0x2598e8f6UL,
+ 0x73888baeUL, 0x16ef3716UL, 0xf8408204UL, 0x9d273ebcUL, 0x241fe921UL,
+ 0x41785599UL, 0xafd7e08bUL, 0xcab05c33UL, 0x3bb659edUL, 0x5ed1e555UL,
+ 0xb07e5047UL, 0xd519ecffUL, 0x6c213b62UL, 0x094687daUL, 0xe7e932c8UL,
+ 0x828e8e70UL, 0xd49eed28UL, 0xb1f95190UL, 0x5f56e482UL, 0x3a31583aUL,
+ 0x83098fa7UL, 0xe66e331fUL, 0x08c1860dUL, 0x6da63ab5UL, 0xa4e140bdUL,
+ 0xc186fc05UL, 0x2f294917UL, 0x4a4ef5afUL, 0xf3762232UL, 0x96119e8aUL,
+ 0x78be2b98UL, 0x1dd99720UL, 0x4bc9f478UL, 0x2eae48c0UL, 0xc001fdd2UL,
+ 0xa566416aUL, 0x1c5e96f7UL, 0x79392a4fUL, 0x97969f5dUL, 0xf2f123e5UL,
+ 0x05196b4dUL, 0x607ed7f5UL, 0x8ed162e7UL, 0xebb6de5fUL, 0x528e09c2UL,
+ 0x37e9b57aUL, 0xd9460068UL, 0xbc21bcd0UL, 0xea31df88UL, 0x8f566330UL,
+ 0x61f9d622UL, 0x049e6a9aUL, 0xbda6bd07UL, 0xd8c101bfUL, 0x366eb4adUL,
+ 0x53090815UL, 0x9a4e721dUL, 0xff29cea5UL, 0x11867bb7UL, 0x74e1c70fUL,
+ 0xcdd91092UL, 0xa8beac2aUL, 0x46111938UL, 0x2376a580UL, 0x7566c6d8UL,
+ 0x10017a60UL, 0xfeaecf72UL, 0x9bc973caUL, 0x22f1a457UL, 0x479618efUL,
+ 0xa939adfdUL, 0xcc5e1145UL, 0x06ee4d76UL, 0x6389f1ceUL, 0x8d2644dcUL,
+ 0xe841f864UL, 0x51792ff9UL, 0x341e9341UL, 0xdab12653UL, 0xbfd69aebUL,
+ 0xe9c6f9b3UL, 0x8ca1450bUL, 0x620ef019UL, 0x07694ca1UL, 0xbe519b3cUL,
+ 0xdb362784UL, 0x35999296UL, 0x50fe2e2eUL, 0x99b95426UL, 0xfcdee89eUL,
+ 0x12715d8cUL, 0x7716e134UL, 0xce2e36a9UL, 0xab498a11UL, 0x45e63f03UL,
+ 0x208183bbUL, 0x7691e0e3UL, 0x13f65c5bUL, 0xfd59e949UL, 0x983e55f1UL,
+ 0x2106826cUL, 0x44613ed4UL, 0xaace8bc6UL, 0xcfa9377eUL, 0x38417fd6UL,
+ 0x5d26c36eUL, 0xb389767cUL, 0xd6eecac4UL, 0x6fd61d59UL, 0x0ab1a1e1UL,
+ 0xe41e14f3UL, 0x8179a84bUL, 0xd769cb13UL, 0xb20e77abUL, 0x5ca1c2b9UL,
+ 0x39c67e01UL, 0x80fea99cUL, 0xe5991524UL, 0x0b36a036UL, 0x6e511c8eUL,
+ 0xa7166686UL, 0xc271da3eUL, 0x2cde6f2cUL, 0x49b9d394UL, 0xf0810409UL,
+ 0x95e6b8b1UL, 0x7b490da3UL, 0x1e2eb11bUL, 0x483ed243UL, 0x2d596efbUL,
+ 0xc3f6dbe9UL, 0xa6916751UL, 0x1fa9b0ccUL, 0x7ace0c74UL, 0x9461b966UL,
+ 0xf10605deUL
+#endif
+ }
+};
--- /dev/null
+
+The following is brief documentation for the interface provided by the lafs library.
+More details can be found in the various source files.
+
+The lafs library allow a lafs filesystems to be created, loaded, examined, and modified.
+
+The key objects managed are:
+
+ - struct lafs - a handle for the filesystem as a whole
+ - struct lafs_device - an individual device within a struct lafs
+ - struct lafs_ino - an 'inode' or file object within the filesystem
+ - struct lafs_blk - a block in a file/inode
+
+
+Each subordinate structure has a reference to the whole filesystem so
+mutliple pointers do not need to be passed around.
+
+talloc is used for memory management, so talloc_free(lafs) will free the whole filesystem.
+
+
+A filesystem handle must be created and separately initialised, either from a device
+or as a new array.
+
+Function all:
+
+struct lafs *lafs_alloc(void)
+ Create a new handle. Most things won't work on this. Normally either
+ lafs_new() or lafs_load() must be used.
+
+
+int lafs_new(struct lafs*, int block_bytes)
+ Initialise the filesystem with a new random uuid etc. It will have no devices
+ attached and so no size
+
+struct lafs_device * lafs_add_device(struct lafs*, char *dev, int fd, long segment_blocks, long stride_blocks, int width,
+ int usage_inum)
+ Add a device to a lafs. A number of segments will be calculated leaving room for metadata
+ at start and end. The device will be inserted into the address-space and some suitable point.
+
+struct lafs_ino *lafs_get_itable(struct lafs *)
+ Return a handle on the inode file which holds all inodes
+
+struct lafs_ino *lafs_add_inode(struct lafs_ino *itable, long inum, int type)
+ Create a new file with given inode number and type.
+
+int lafs_write_dev(struct lafs_device*)
+ write the device_blocks for this device
+
+int lafs_checkpoint(struct lafs *)
+ write out a checkpoint and then the state blocks to all devices.
+ This involves writing all blocks that have been marked as 'sched'.
+ Doing that might cause other blocks (parents) to be scheduled, so
+ multiple write-clusters might be needed.
+
+
+int lafs_im_set(struct lafs_ino*, int inum)
+ The lafs_ino must be an InodeMap file, and inum is marked as being in-use
+int lafs_im_clr(struct lafs_ino*, int inum)
+ The lafs_ino must be an InodeMap file, and inum is marked as being available
+
+int lafs_add_free_seg(struct lafs*, int dev, long segment)
+ Add the given segment to the list of segments available for write-out.
+
+struct lafs_dblk(struct lafs_ino*, loff_t bnum)
+ Get the datablock in the inode. Data blocks always referred to their parent in
+ the indexing tree.
+
+int lafs_sched_dblk(struct lafs_dblk*)
+ Mark the block as dirty and to be written in the next checkpoint. This places it on a list
+ an references the parent to ensure the parent isn't written before this block.
\ No newline at end of file
--- /dev/null
+
+#include <unistd.h>
+#include <lafs/lafs.h>
+#include <talloc.h>
+#include <memory.h>
+#include <sys/mount.h>
+#include <sys/stat.h>
+#include <sys/time.h>
+#include <fcntl.h>
+
+/* Add a new device to lafs.
+ * This just sets up the internal data structures.
+ * We at least need to lafs_write_dev() to write metadata out, and may want some
+ * mount call to connect this to an active lafs.
+ */
+
+static int destroy(struct lafs_device *dev)
+{
+ close(dev->fd);
+ return 0;
+}
+
+static int get_logical_block_size(int fd)
+{
+ struct stat stb;
+ int sfd;
+ int rv = -1;
+ char path[60];
+ char buf[20];
+ fstat(fd, &stb);
+ sprintf(path, "/sys/dev/block/%d:%d/queue/logical_block_size",
+ major(stb.st_rdev), minor(stb.st_rdev));
+ sfd = open(path, O_RDONLY);
+ if (sfd >= 0) {
+ int n = read(sfd, buf, sizeof(buf));
+ if (n >= 0)
+ buf[n] = 0;
+ close(sfd);
+ rv = atoi(buf);
+ }
+ if (rv > 0)
+ return rv;
+ return 512;
+}
+
+struct lafs_device *lafs_add_device(struct lafs *fs, char *devname, int fd,
+ loff_t segblocks, loff_t strideblocks,
+ int width, int usage_inum)
+{
+ struct lafs_device *dev = talloc(fs, struct lafs_device);
+ struct lafs_device *d2;
+ unsigned long long size;
+ int devblk;
+
+ memset(dev, 0, sizeof(*dev));
+ dev->fd = fd;
+ dev->seq = 1;
+ dev->devnum = fs->loaded_devs++;
+ dev->name = devname;
+
+ dev->width = width;
+ dev->stride = strideblocks;
+ dev->segment_size = segblocks;
+ dev->usage_inum = usage_inum;
+ dev->next = fs->devs;
+ fs->devs = dev;
+ dev->fs = fs;
+ fs->devices++;
+
+ if (dev->segment_size > fs->max_segment)
+ fs->max_segment = dev->segment_size;
+
+ if (dev->width * dev->stride <= dev->segment_size) {
+ dev->tables_per_seg = dev->segment_size /
+ dev->width / dev->stride;
+ dev->rows_per_table = dev->stride;
+ dev->segment_stride = dev->segment_size;
+ } else {
+ dev->tables_per_seg = 1;
+ dev->rows_per_table = dev->segment_size / dev->width;
+ dev->segment_stride = dev->rows_per_table;
+ }
+ gettimeofday(&dev->ctime, NULL);
+
+ talloc_set_destructor(dev, destroy);
+
+ /* now need to work out where the metadata goes and how much room
+ * is left for segments.
+ * Device block needs 1K and goes once at start and once at end.
+ * Start location is 1K in unless basic block size is larger
+ */
+ ioctl(fd, BLKGETSIZE64, &size);
+ devblk = get_logical_block_size(fd);
+ if (devblk < LAFS_DEVBLK_SIZE)
+ devblk = LAFS_DEVBLK_SIZE;
+
+ size &= ~(unsigned long long)(devblk-1);
+
+
+#if 0
+ dev->devaddr[0] = devblk;
+#else
+ dev->devaddr[0] = 0;
+#endif
+ dev->devaddr[1] = size - devblk;
+ /* State block has size set by 'fs->statesize'.
+ * We have two at the start of the device and two at the end.
+ * If stride*width < segment size we put them at multiples of width*stride
+ * If stride*width > segment size we put them at multiples of statesize
+ */
+ /* FIXME just do the simple version for now */
+ if (dev->width != 1 || dev->stride != 1)
+ abort();
+ if (devblk < fs->statesize)
+ devblk = fs->statesize;
+ dev->stateaddr[0] = 2 * devblk;
+ dev->stateaddr[1] = 3 * devblk;
+ dev->stateaddr[2] = size - 2*devblk;
+ dev->stateaddr[3] = size - 3*devblk;
+
+ /* segments need to align with width*stride too - later */
+ dev->segment_offset = 4 * devblk;
+ dev->segment_stride = dev->segment_size;
+ dev->segment_count = (dev->stateaddr[3] - dev->segment_offset) / fs->blocksize / dev->segment_size;
+
+
+ dev->tablesize = ((dev->segment_count + fs->blocksize/2 + 1)
+ / (fs->blocksize/2));
+
+ dev->size = dev->segment_count * dev->segment_size;
+
+ /* Need to find a suitable offset */
+ dev->start = 0;
+ for (d2 = dev->next; d2 ; d2 = d2->next) {
+ if (dev->start <= d2->start + d2->size &&
+ dev->start + dev->size > d2->start) {
+ dev->start = d2->start + d2->size;
+ /* start again from top */
+ d2 = dev;
+ }
+ }
+
+ return dev;
+}
--- /dev/null
+/*
+ * Add a free segment to the freelist if there is room
+ */
+
+#include <lafs/lafs.h>
+
+int lafs_add_free_seg(struct lafs *fs, int dev, loff_t seg)
+{
+ int next_tail = (fs->free_tail+1) & (128-1);
+
+ if (next_tail == fs->free_head)
+ /* no room */
+ return 0;
+ fs->freelist[fs->free_tail].dev = dev;
+ fs->freelist[fs->free_tail].seg = seg;
+ fs->free_tail = next_tail;
+ return 1;
+}
--- /dev/null
+
+/*
+ * Create an inode in the file with the given inum and type.
+ * We get the data block, file it in, and import it.
+ */
+
+#include <lafs/lafs.h>
+
+
+struct lafs_ino *lafs_add_inode(struct lafs_ino *fsys, int inum, int type)
+{
+ struct lafs_dblk *db;
+ struct lafs_ino *ino;
+
+ db = lafs_dblk(fsys, inum);
+
+ lafs_inode_init(fsys->fs, db->b.data, type);
+
+ ino = lafs_import_inode(db);
+ lafs_dirty_blk(&db->b);
+ return ino;
+}
--- /dev/null
+
+#include <lafs/lafs.h>
+#include <talloc.h>
+#include <memory.h>
+#include <lafs/list.h>
+
+struct lafs *lafs_alloc(void)
+{
+ struct lafs *fs = talloc(NULL, struct lafs);
+ int i;
+
+ memset(fs, 0, sizeof(*fs));
+
+ INIT_LIST_HEAD(&fs->wc[0].blocks);
+ INIT_LIST_HEAD(&fs->wc[1].blocks);
+ INIT_LIST_HEAD(&fs->leafs);
+ INIT_LIST_HEAD(&fs->account_leafs);
+
+ for (i=0; i < (1<<HASH_BITS); i++)
+ INIT_LIST_HEAD(&fs->htable[i]);
+
+ return fs;
+}
--- /dev/null
+
+/* lafs_allocated_block.
+ * This block has been allocated to this address (and will soon
+ * be written there).
+ * We need to update the accounting info.
+ */
+
+#include <lafs/lafs.h>
+#include <stdlib.h>
+#include <stdio.h>
+
+void lafs_allocated_block(struct lafs_blk *b, loff_t addr)
+{
+ struct lafs *fs = b->ino->fs;
+ struct lafs_iblk *p;
+
+ printf("allocate %d/%ld %s to %ld\n", b->ino->inum, b->fileaddr,
+ (b->flags & B_Index) ? "index":"data", addr);
+ if (b->parent == NULL) {
+ /* This is the root inode. Its address goes
+ * directly in the 'state' block.
+ */
+ if (b->flags & B_Index) abort();
+
+ /* FIXME this might be a different snapshot */
+ printf("Root inode at %ld\n", addr);
+ b->physaddr = addr;
+ fs->ss.root_addr = addr;
+ return;
+ }
+
+ p = b->parent;
+ lafs_summary_update(b->ino, b->physaddr, addr,
+ !!(b->flags & B_Index));
+ b->physaddr = addr;
+ b->chain = p->uninc;
+ p->uninc = b;
+}
+
--- /dev/null
+
+/* Force out a checkpoint
+ * All scheduled blocks get incorporated and/or written
+ * then we write out the new state blocks
+ */
+
+#include <stdlib.h>
+#include <lafs/lafs.h>
+
+#include <stdio.h>
+
+int lafs_checkpoint(struct lafs *fs)
+{
+ struct lafs_blk *b;
+
+ fs->checkpointing = CHECKPOINTING | CHECKPOINT_START;
+
+ while (!list_empty(&fs->leafs)) {
+ while (!list_empty(&fs->leafs)) {
+ b = list_first_entry(&fs->leafs,
+ struct lafs_blk,
+ leafs);
+ printf("checkpoint %p %d/%ld %s %p\n", b, b->ino->inum, b->fileaddr,
+ (b->flags & B_Index) ? "index":"data", b->parent);
+ list_del_init(&b->leafs);
+ if (!(b->flags & B_Index)) {
+ struct lafs_dblk *db = dblk(b);
+ if (b->ino->type == TypeSegmentMap) {
+ /* Allow parents to be processed, but freeze this */
+ de_sched(b); b->flags |= B_Sched;
+ list_add(&b->leafs, &fs->account_leafs);
+ continue;
+ }
+ if (b->ino->type == TypeInodeFile &&
+ db->my_inode &&
+ db->my_inode->iblock &&
+ db->my_inode->iblock->sched_cnt) {
+ de_sched(b);
+ continue;
+ }
+ }
+
+ if (b->flags & B_Index) {
+ struct lafs_iblk *ib = iblk(b);
+ if (ib->sched_cnt) {
+ de_sched(b);
+ continue;
+ }
+ if (ib->uninc) {
+ de_sched(b);
+ lafs_incorporate(ib);
+ if (!(ib->b.flags & B_Sched))
+ abort();
+ /* We'll pick it up next time 'round */
+ continue;
+ }
+ }
+ if (b->flags & B_Dirty) {
+ printf("...alloc\n");
+ lafs_cluster_allocate(b, 0);
+ } else
+ de_sched(b);
+ }
+ lafs_cluster_flush(fs, 0);
+ }
+ fs->flags |= LAFS_DELAY_UPDATES;
+ while (!list_empty(&fs->account_leafs)) {
+ b = list_first_entry(&fs->account_leafs,
+ struct lafs_blk, leafs);
+ b->flags &= ~B_Sched;
+ printf("Account %d/%ld\n", b->ino->inum, b->fileaddr);
+ list_del_init(&b->leafs);
+
+ lafs_sched_blk(b);
+ list_del_init(&b->leafs);
+
+ lafs_cluster_allocate(b, 0);
+ }
+ fs->checkpointing |= CHECKPOINT_END;
+ lafs_cluster_flush(fs, 0);
+ fs->flags &= ~LAFS_DELAY_UPDATES;
+
+ lafs_write_state(fs);
+ return 0;
+}
--- /dev/null
+
+/*
+ * allocate a dirty block to a cluster so we can easily write it
+ * for a flush.
+ *
+ * FIXME
+ * For now, don't sort the blocks and assume that the cluster head
+ * and segment will have adequate room for all blocks.
+ * That is sufficient for mkfs
+ */
+#include <lafs/lafs.h>
+#include <stdlib.h>
+
+void lafs_cluster_allocate(struct lafs_blk *b, int cnum)
+{
+ struct lafs *fs = b->ino->fs;
+ struct lafs_cluster *wc;
+ if (!(b->flags & B_Dirty))
+ abort();
+ if (!(b->flags & B_Sched))
+ abort();
+
+ if (!(b->flags & B_Index) &&
+ b->ino->type == TypeInodeFile &&
+ dblk(b)->my_inode &&
+ (dblk(b)->my_inode->iflags & I_Dirty))
+ lafs_inode_fillblock(dblk(b)->my_inode, b->data);
+
+ if ((b->flags & B_Index) &&
+ b->ino->iblock == iblk(b)) {
+ /* InoIdx block - cannot write that, must write the
+ * data block instead */
+ lafs_dirty_blk(&b->ino->dblock->b);
+ de_sched(b);
+ return;
+ }
+
+
+ wc = &fs->wc[cnum];
+
+ if (wc->remaining == 0) {
+ /* FIXME lafs_cluster_flush(fs, cnum) ?? */
+ lafs_new_segment(fs, cnum);
+ if (!wc->remaining)
+ abort();
+ }
+ wc->remaining--;
+ list_add_tail(&b->leafs, &wc->blocks);
+}
--- /dev/null
+
+/*
+ * Flush out the given cluster with a cluster head.
+ * We "know" there is room in the cluster head and
+ * in the segment
+ */
+#define _GNU_SOURCE
+#define _FILE_OFFSET_BITS 64
+
+#include <lafs/lafs.h>
+#include <unistd.h>
+#include <stdlib.h>
+#include <memory.h>
+#include <stdio.h>
+
+/*-----------------------------------------------------------------------
+ * A segment is divided up in a slightly complicated way into
+ * tables, rows and columns. This allows us to align writes with
+ * stripes in a raid4 array or similar.
+ * So we have some support routines to help track our way around
+ * the segment.
+ *
+ * A write cluster always comprises a whole number of rows - it is padded
+ * with zeros if necessary.
+ * However block addressing follow columns, so the blocks written to
+ * a cluster may well not be contiguously addressed.
+ *
+ * There are 'width' blocks (or columns) in a row, and either:
+ * - segment_stride rows in a table, if segment_stride < segment_size or
+ * - one table per segment, when segment_stride > segment_size.
+ * These are calculated and summarised in rows_per_table and tables_per_seg
+ * in the 'struct lafs_device' structure.
+ *
+ * A 'seg_pos' records the current segment and where we are in it.
+ * It has:
+ * dev,num to identify the segment
+ * st_table,st_row to identify where the cluster starts.
+ * nxt_table,nxt_row to identify where the cluster ends.
+ * table,row,col to identify where the next block will be found
+ */
+
+static int seg_remainder(struct lafs *fs, struct lafs_segpos *seg)
+{
+ /* return the number of blocks from the start of segpos to the
+ * end of the segment.
+ * i.e. remaining rows in this table, plus remaining tables in
+ * this segment
+ */
+ struct lafs_device *dv = dev_by_num(fs, seg->dev);
+ int rows = dv->rows_per_table - seg->st_row;
+
+ if (seg->dev < 0) abort();
+ rows += dv->rows_per_table * (dv->tables_per_seg - seg->st_table - 1);
+ return rows * dv->width;
+}
+
+#if 0
+static void seg_step(struct lafs *fs, struct lafs_segpos *seg)
+{
+ /* reposition this segpos to be immediately after it's current end
+ * and make the 'current' point be the start.
+ * Size will be empty
+ */
+ seg->st_table = seg->nxt_table;
+ seg->st_row = seg->nxt_row;
+ seg->table = seg->st_table;
+ seg->row = seg->st_row;
+ seg->col = 0;
+}
+#endif
+
+static u32 seg_setsize(struct lafs *fs, struct lafs_segpos *seg, u32 size)
+{
+ /* move the 'nxt' table/row to be 'size' blocks beyond
+ * current start. size will be rounded up to a multiple
+ * of width.
+ */
+ struct lafs_device *dv = dev_by_num(fs, seg->dev);
+ u32 rv;
+ int rows;
+
+ if (seg->dev < 0) abort();
+ rows = (size + dv->width - 1) / dv->width;
+ rv = rows * dv->width;
+ rows += seg->st_row;
+ seg->nxt_table = seg->st_table + rows / dv->rows_per_table;
+ seg->nxt_row = rows % dv->rows_per_table;
+ return rv;
+}
+
+static u64 seg_addr(struct lafs *fs, struct lafs_segpos *seg)
+{
+ /* Return the virtual address of the blocks pointed
+ * to by 'seg'.
+ */
+ struct lafs_device *dv = dev_by_num(fs, seg->dev);
+ u64 addr;
+
+ if (seg->dev < 0)
+ /* Setting 'next' address for last cluster in
+ * a cleaner segment */
+ return 0;
+ addr = seg->col * dv->stride;
+ addr += seg->row;
+ addr += seg->table * dv->rows_per_table;
+ addr += seg->num * dv->segment_stride;
+ addr += dv->start;
+ return addr;
+}
+
+static u64 seg_next(struct lafs *fs, struct lafs_segpos *seg)
+{
+ /* step forward one block, returning the address of
+ * the block stepped over
+ */
+ struct lafs_device *dv = dev_by_num(fs, seg->dev);
+ u64 addr = seg_addr(fs, seg);
+
+ /* now step forward in column or table or seg */
+ seg->col++;
+ if (seg->col >= dv->width) {
+ seg->col = 0;
+ seg->row++;
+ if (seg->row >= dv->rows_per_table) {
+ seg->row = 0;
+ seg->table++;
+ }
+ }
+ return addr;
+}
+
+/*-------------------------------------------------------------------------
+ * Cluster head building.
+ * We build the cluster head bit by bit as we find blocks
+ * in the list. These routines help.
+ */
+
+static void cluster_addhead(struct lafs_cluster *wc, struct lafs_ino *ino,
+ int cnum,
+ struct group_head **headstart)
+{
+ struct group_head *gh = (void*)((char *)wc->chead +
+ wc->chead_size);
+ u16 tnf;
+
+ *headstart = gh;
+ gh->inum = __cpu_to_le32(ino->inum);
+ gh->fsnum = __cpu_to_le32(ino->filesys->inum);
+ tnf = ((ino->generation<<8) | (ino->trunc_gen & 0xff))
+ & 0x7fff;
+ if (cnum)
+ tnf |= 0x8000;
+ gh->truncatenum_and_flag = __cpu_to_le16(tnf);
+ wc->chead_size += sizeof(struct group_head);
+}
+
+static void cluster_closehead(struct lafs_cluster *wc,
+ struct group_head *headstart)
+{
+ int size = wc->chead_size - (((char *)headstart) - (char *)wc->chead);
+
+ headstart->group_size_words = size / 4;
+}
+
+#if 0
+static void cluster_addmini(struct lafs_cluster *wc, u32 addr, int offset,
+ int size, const char *data,
+ int size2, const char *data2)
+{
+ /* if size2 !=0, then only
+ * (size-size2) is at 'data' and the rest is at 'data2'
+ */
+ struct miniblock *mb = ((struct miniblock *)
+ ((char *)wc->chead + wc->chead_size));
+
+ mb->block_num = __cpu_to_le32(addr);
+ mb->block_offset = __cpu_to_le16(offset);
+ mb->length = __cpu_to_le16(size + DescMiniOffset);
+ wc->chead_size += sizeof(struct miniblock);
+ memcpy(mb->data, data, size-size2);
+ if (size2)
+ memcpy(mb->data + size-size2, data2, size2);
+ memset(mb->data+size, 0, (-size)&3);
+ wc->chead_size += ROUND_UP(size);
+}
+#endif
+
+static void cluster_adddesc(struct lafs_cluster *wc, struct lafs_blk *blk,
+ struct descriptor **desc_start)
+{
+ struct descriptor *dh = (struct descriptor *)((char *)wc->chead +
+ wc->chead_size);
+ *desc_start = dh;
+ dh->block_num = __cpu_to_le32(blk->fileaddr);
+ dh->block_cnt = __cpu_to_le32(0);
+ dh->block_bytes = 0;
+ if (blk->flags && B_Index)
+ dh->block_bytes = DescIndex;
+ wc->chead_size += sizeof(struct descriptor);
+}
+
+static void cluster_incdesc(struct lafs_cluster *wc, struct descriptor *desc_start,
+ struct lafs_blk *b, int blksz)
+{
+ desc_start->block_cnt =
+ __cpu_to_le32(__le32_to_cpu(desc_start->block_cnt)+1);
+ if (!(b->flags & B_Index)) {
+ if (b->ino->type >= TypeBase) {
+ u64 size = b->ino->md.file.size;
+ if (size > ((loff_t)b->fileaddr * blksz) &&
+ size <= ((loff_t)(b->fileaddr + 1) * blksz))
+ desc_start->block_bytes =
+ __cpu_to_le32(size & (blksz-1));
+ else
+ desc_start->block_bytes =
+ __cpu_to_le32(blksz);
+ } else
+ desc_start->block_bytes = __cpu_to_le32(blksz);
+ }
+}
+
+
+static int calc_cluster_csum(struct cluster_head *head)
+{
+ unsigned int oldcsum = head->checksum;
+ unsigned long long newcsum = 0;
+ unsigned long csum;
+ int i;
+ unsigned int *superc = (unsigned int *) head;
+ head->checksum = 0;
+
+ for (i = 0; i < __le16_to_cpu(head->Hlength)/4; i++)
+ newcsum += __le32_to_cpu(superc[i]);
+ csum = (newcsum & 0xffffffff) + (newcsum>>32);
+ head->checksum = oldcsum;
+ return __cpu_to_le32(csum);
+}
+
+void lafs_cluster_flush(struct lafs *fs, int cnum)
+{
+ char chead_buf[4096];
+ struct cluster_head *ch;
+ struct lafs_cluster *wc = &fs->wc[cnum];
+ struct lafs_blk *b;
+ int cluster_size;
+ int i;
+ loff_t head_addr[4];
+ struct lafs_ino *current_inode = NULL;
+ off_t current_block = NoBlock;
+ struct descriptor *desc_start;
+ struct group_head *head_start = NULL;
+ struct lafs_device *dv;
+
+ printf("flush\n");
+ if (list_empty(&wc->blocks) &&
+ !(fs->checkpointing & CHECKPOINT_END)) {
+ printf("...skipped\n");
+ return;
+ }
+
+ ch = (void*)chead_buf;
+
+ wc->chead = chead_buf;
+ wc->chead_size = sizeof(struct cluster_head);
+ wc->chead_blocks = 1;
+ memcpy(ch->idtag, "LaFSHead", 8);
+ memcpy(ch->uuid, fs->uuid, 16);
+ ch->seq = __cpu_to_le64(fs->wc[cnum].seq);
+
+ cluster_size = seg_setsize(fs, &wc->seg,
+ seg_remainder(fs, &wc->seg) - wc->remaining);
+
+ /* find, and step over, address header block(s) */
+ for (i = 0; i < wc->chead_blocks ; i++)
+ head_addr[i] = seg_next(fs, &wc->seg);
+
+ ch->flags = 0;
+ if (cnum == 0 && fs->checkpointing) {
+ ch->flags = __cpu_to_le32(fs->checkpointing);
+ if (fs->checkpointing & CHECKPOINT_END)
+ fs->checkpointing = 0;
+ else if (fs->checkpointing & CHECKPOINT_START) {
+ fs->checkpoint_cluster = head_addr[0];
+ fs->checkpointing &= ~CHECKPOINT_START;
+ }
+ }
+
+ list_for_each_entry(b, &wc->blocks, leafs) {
+ if (b->ino != current_inode) {
+ /* need to create a new group_head */
+ desc_start = NULL;
+ if (head_start)
+ cluster_closehead(wc, head_start);
+ cluster_addhead(wc, b->ino, cnum, &head_start);
+ current_inode = b->ino;
+ current_block = NoBlock;
+ }
+ if (desc_start == NULL || b->fileaddr != current_block+1 ||
+ (b->flags & B_Index)) {
+ cluster_adddesc(wc, b, &desc_start);
+ current_block = b->fileaddr;
+ } else
+ current_block++;
+ cluster_incdesc(wc, desc_start, b, fs->blocksize);
+
+ lafs_allocated_block(b, seg_next(fs, &wc->seg));
+ }
+ if (head_start)
+ cluster_closehead(wc, head_start);
+
+ ch->Hlength = __cpu_to_le16(wc->chead_size);
+ ch->Clength = __cpu_to_le16(cluster_size);
+ ch->verify_type = VerifyNull;
+
+ ch->next_addr = __cpu_to_le64(seg_addr(fs, &wc->seg));
+ ch->prev_addr = __cpu_to_le64(wc->prev_addr);
+ wc->prev_addr = head_addr[i];
+ ch->this_addr = __cpu_to_le64(wc->prev_addr);
+ ch->checksum = calc_cluster_csum(ch);
+
+ dv = dev_by_num(fs, wc->seg.dev);
+
+ for (i = 0; i < wc->chead_blocks; i++) {
+ int dev;
+ loff_t sect;
+ virttophys(fs, head_addr[i], &dev, §);
+ lseek64(dv->fd, sect, SEEK_SET);
+ write(dv->fd, chead_buf+ i * fs->blocksize, fs->blocksize);
+ }
+ while (!list_empty(&wc->blocks)) {
+ int dev;
+ loff_t sect;
+
+ b = list_first_entry(&wc->blocks, struct lafs_blk, leafs);
+ list_del_init(&b->leafs);
+
+ virttophys(fs, b->physaddr, &dev, §);
+ lseek64(dv->fd, sect, SEEK_SET);
+ b->flags &= ~B_Dirty;
+ write(dv->fd, b->data, fs->blocksize);
+ de_sched(b);
+ }
+}
--- /dev/null
+
+/* lafs_cluster_init.
+ * initialise the write-cluster structure. This can identify
+ * a location in the middle of a segment where write can continue
+ * or no valid segment, so a new one will be allocated.
+ */
+#include <lafs/lafs.h>
+
+static int seg_remainder(struct lafs *fs, struct lafs_segpos *seg)
+{
+ /* return the number of blocks from the start of segpos to the
+ * end of the segment.
+ * i.e. remaining rows in this table, plus remaining tables in
+ * this segment
+ */
+ struct lafs_device *dv = dev_by_num(fs, seg->dev);
+ int rows = dv->rows_per_table - seg->st_row;
+
+ if (seg->dev < 0) abort();
+ rows += dv->rows_per_table * (dv->tables_per_seg - seg->st_table - 1);
+ return rows * dv->width;
+}
+
+static void cluster_reset(struct lafs *fs, struct lafs_cluster *wc)
+{
+ if (wc->seg.dev < 0) {
+ wc->remaining = 0;
+ wc->chead_size = 0;
+ return;
+ }
+ wc->remaining = seg_remainder(fs, &wc->seg);
+ wc->chead_blocks = 1;
+ wc->remaining--;
+ wc->chead_size = sizeof(struct cluster_head);
+}
+
+static void seg_setpos(struct lafs *fs, struct lafs_segpos *seg, loff_t addr)
+{
+ /* set seg to start at the given virtual address, and be
+ * of size 0
+ * 'addr' should always be at the start of a row, though
+ * as it might be read off storage, we need to be
+ * a little bit careful.
+ */
+ loff_t offset;
+ u32 row, col, table;
+ struct lafs_device *dv;
+ virttoseg(fs, addr, &seg->dev, &seg->num, &offset);
+ dv = &fs->devs[seg->dev];
+ col = offset / dv->stride;
+ row = offset % dv->stride;
+ table = col / dv->width;
+ col = col % dv->width;
+ if (col) abort(); /* FIXME return an error instead */
+
+ seg->st_table = table;
+ seg->st_row = row;
+
+ seg->table = seg->nxt_table = seg->st_table;
+ seg->row = seg->nxt_row = seg->st_row;
+ seg->col = 0;
+}
+
+void lafs_cluster_init(struct lafs *fs, int cnum,
+ loff_t addr, loff_t prev, loff_t seq)
+{
+ struct lafs_cluster *wc = &fs->wc[cnum];
+
+ INIT_LIST_HEAD(&wc->blocks);
+ wc->prev_addr = prev;
+ wc->seq = seq;
+ if (addr)
+ seg_setpos(fs, &wc->seg, addr);
+ else
+ wc->seg.dev = -1;
+ cluster_reset(fs, wc);
+ //fs->free_blocks = wc->remaining;
+}
--- /dev/null
+/*
+ * find or create a dblk for a given offset in a file,
+ * and return it.
+ */
+
+#include <lafs/lafs.h>
+#include <talloc.h>
+#include <memory.h>
+
+
+struct lafs_dblk *lafs_dblk(struct lafs_ino *ino, loff_t bnum)
+{
+ struct lafs_dblk *db;
+ struct lafs *fs = ino->fs;
+ int hash = (long)ino;
+
+ hash ^= bnum;
+ hash ^= hash >> 10;
+ hash &= (1<<HASH_BITS)-1;
+
+ list_for_each_entry(db, &fs->htable[hash], b.hash) {
+ if (db->b.flags & B_Index)
+ continue;
+ if (db->b.ino != ino ||
+ db->b.fileaddr != bnum)
+ continue;
+ return db;
+ }
+
+ /* Not in the table, so need to create it */
+ db = talloc(ino, struct lafs_dblk);
+ memset(db, 0, sizeof(*db));
+
+ db->b.data = talloc_size(db, fs->blocksize);
+ db->b.fileaddr = bnum;
+ db->b.ino = ino;
+ list_add(&db->b.hash, &fs->htable[hash]);
+ INIT_LIST_HEAD(&db->b.siblings);
+ INIT_LIST_HEAD(&db->b.leafs);
+ return db;
+}
--- /dev/null
+
+#include <lafs/lafs.h>
+
+struct lafs_device *
+lafs_dev_find(struct lafs *fs, loff_t virt)
+{
+ struct lafs_device *dev;
+ for (dev = fs->devs ; dev; dev = dev->next)
+ if (virt >= dev->start &&
+ virt < dev->start + dev->size)
+ return dev;
+ abort();
+ return NULL;
+}
--- /dev/null
+/*
+ * lafs_dirty_inode
+ * This inode has been changed so we need to ensure it gets
+ * written out. In particular lafs_inode_fillblock needs to be
+ * called at some stage.
+ *
+ */
+
+#include <lafs/lafs.h>
+
+void lafs_dirty_inode(struct lafs_ino *ino)
+{
+ ino->iflags |= I_Dirty;
+ lafs_sched_blk(&ino->dblock->b);
+}
--- /dev/null
+
+/*
+ * Connect this dblk into indexing tree and find
+ * its phys addr.
+ * After this call, ->parent must be set, and ->physaddr must
+ * be correct
+ *
+ * FIXME
+ * For now we only work with indexing directly from the inode
+ * Separate index blocks are not yet supported
+ */
+
+#include <stdlib.h>
+#include <lafs/lafs.h>
+#include <talloc.h>
+
+static u64 leaf_lookup(unsigned char *buf, int len, u32 target, u32 *nextp);
+
+int lafs_find_dblk(struct lafs_dblk *db)
+{
+ struct lafs_ino *ino = db->b.ino;
+ struct lafs *fs = ino->fs;
+
+ if (db->b.parent)
+ return 0;
+
+ if (!ino->iblock)
+ lafs_make_iblock(ino);
+
+ if (ino->depth > 1)
+ abort();
+
+ if (ino->filesys != ino ||
+ db->b.fileaddr != 0) {
+ db->b.parent = ino->iblock;
+ (void)talloc_reference(db, ino->iblock);
+ list_add(&db->b.siblings, &ino->iblock->children);
+ } else
+ db->b.parent = NULL;
+
+ if (ino->depth == 0) {
+ db->b.physaddr = 0;
+ return 0;
+ }
+
+ db->b.physaddr = leaf_lookup((unsigned char*)ino->dblock->b.data + ino->metadata_size,
+ fs->blocksize - ino->metadata_size,
+ db->b.fileaddr,
+ NULL);
+ return 0;
+}
+
+
+/*
+ * extract little-endian values out of memory.
+ * Each function is given a char*, and moves it forwards
+ */
+
+#define decode16(p) ({ unsigned int _a; _a= (unsigned char)*(p++); _a + (((unsigned char)*p++)<<8); })
+#define decode32(p) ({ long _b; _b = decode16(p); _b + ((u32)decode16(p)<<16); })
+#define decode48(p) ({ u64 _c; _c = decode32(p); _c + ((u64)decode16(p)<<32); })
+
+
+uint64_t leaf_lookup(unsigned char *buf, int len, u32 target, u32 *nextp)
+{
+ /* buf starts with a 2byte header
+ * if 1, then 4 byte offset followed by 6byte littleending indirect entries.
+ * if 2, then 12byte extent entries
+ */
+ uint64_t p;
+ unsigned char *cp;
+ int hi,lo;
+ int elen;
+ u32 addr, next;
+ u32 startaddr;
+ if (nextp) *nextp = NoBlock;
+ if (buf[1]) return 0;
+ switch (buf[0]) {
+ default: p = 0;
+ break;
+
+ case 1: /* indirect */
+ len -= 2+4;
+ buf += 2;
+ startaddr = decode32(buf);
+
+ if ( target < startaddr)
+ return 0;
+
+ next = target;
+ target -= startaddr;
+
+ if (target*6 + 6 > len)
+ return 0;
+ buf += target*6;
+ p = decode48(buf);
+ if (nextp) {
+ /* find the next allocated block */
+ uint64_t p2;
+
+ len -= target*6;
+ len -= 6;
+ next++;
+ *nextp = NoBlock;
+ while (len > 6) {
+ p2 = decode48(buf);
+ if (p2) {
+ *nextp = next;
+ break;
+ }
+ len -= 6;
+ next++;
+ }
+ }
+ break;
+
+ case 2: /* extent */
+ /* 12 byte records: 6byte device, 2 byte length, 4 byte fileaddr */
+
+ len -= 2;
+ buf += 2;
+ lo = 0;
+ hi = len/12;
+ while (hi > lo+1) {
+ int mid = (lo+hi)/2;
+ int len;
+ cp = buf;
+ cp += mid*12 + 6;
+
+ len = decode16(cp);
+ addr = decode32(cp);
+
+ if (len && addr <= target)
+ lo = mid;
+ else
+ hi = mid;
+ }
+ cp = buf;
+ cp += lo*12 + 8;
+
+ addr = decode32(cp);
+
+ cp = buf;
+ cp += lo*12+6;
+ elen = decode16(cp);
+ if (target < addr || target >= addr + elen)
+ p = 0;
+ else {
+ cp = buf;
+ cp += lo*12;
+ p = decode48(cp);
+ p += target - addr;
+ }
+
+ if (nextp) {
+ if (lo == len/12)
+ *nextp = NoBlock; /* next extent */
+ else {
+ uint64_t p2;
+ cp = buf;
+ cp += (lo+1)*12;
+ p2 = decode48(cp);
+ len = decode16(cp);
+ if (len == 0)
+ *nextp = NoBlock; /* no more meaningful extents*/
+ else
+ *nextp = decode32(cp);
+ }
+ }
+ }
+ return p;
+}
+
--- /dev/null
+#include <unistd.h>
+#include <stdlib.h>
+#include <lafs/lafs.h>
+#include <memory.h>
+
+/*
+ * Get the primary inode table for a lafs filesystem.
+ * If it doesn't exist we need to create it.
+ */
+
+struct lafs_ino *lafs_get_itable(struct lafs *fs)
+{
+ char *buf;
+ struct lafs_dblk *blk;
+ struct lafs_ino *ino;
+
+ if (fs->ss.root)
+ return fs->ss.root;
+
+ buf = malloc(fs->blocksize);
+
+ if (fs->ss.root_addr)
+ lafs_read_virtual(fs, buf, fs->ss.root_addr);
+ else
+ lafs_inode_init(fs, buf, TypeInodeFile);
+
+ ino = lafs_import_inode_buf(fs, buf, 0, NULL);
+
+ blk = lafs_dblk(ino, 0);
+ memcpy(blk->b.data, buf, fs->blocksize);
+ free(buf);
+
+ ino->dblock = blk;
+ ino->filesys = ino;
+
+ blk->b.flags |= B_Valid;
+ blk->b.physaddr = fs->ss.root_addr;
+ blk->my_inode = ino;
+ if (blk->b.physaddr == 0)
+ lafs_dirty_blk(&blk->b);
+ fs->ss.root = ino;
+ return ino;
+}
--- /dev/null
+
+/*
+ * Mark an inode number as being not-used in the inode map.
+ */
+
+#include <lafs/lafs.h>
+
+int lafs_imap_clr(struct lafs_ino *ino, int inum)
+{
+ struct lafs *fs = ino->fs;
+ struct lafs_dblk *db;
+ int blknum = (inum / fs->blocksize) / 8;
+
+ if (ino->type != TypeInodeMap)
+ return -1;
+ if (blknum >= ino->md.inodemap.size)
+ return 0;
+
+ db = lafs_dblk(ino, blknum);
+ lafs_load_dblk(db);
+ inum -= blknum * fs->blocksize * 8;
+ db->b.data[inum/8] |= (1<<(inum&7));
+ /* FIXME if block is now empty, possibly contract file */
+ lafs_dirty_blk(&db->b);
+ return 0;
+}
--- /dev/null
+
+/*
+ * Mark an inode number as being in-use in the inode map.
+ * If the number is more than one block beyond end-of-file,
+ * we fail.
+ */
+
+#include <lafs/lafs.h>
+#include <memory.h>
+#include <stdio.h>
+
+int lafs_imap_set(struct lafs_ino *ino, int inum)
+{
+ struct lafs *fs = ino->fs;
+ struct lafs_dblk *db;
+ int blknum = (inum / fs->blocksize) / 8;
+
+ if (ino->type != TypeInodeMap)
+ return -1;
+ if (blknum > ino->md.inodemap.size)
+ return -1;
+
+ printf("imap set %d %d %d\n", inum, blknum, ino->md.inodemap.size);
+ db = lafs_dblk(ino, blknum);
+ lafs_load_dblk(db);
+ if (blknum == ino->md.inodemap.size) {
+ memset(db->b.data, 0xff, fs->blocksize);
+ ino->md.inodemap.size++;
+ lafs_dirty_inode(ino);
+ }
+ inum -= blknum * fs->blocksize * 8;
+ db->b.data[inum/8] &= ~ (1<<(inum&7));
+ /* FIXME if block is now empty, punch a hole */
+ lafs_dirty_blk(&db->b);
+ return 0;
+}
--- /dev/null
+#include <lafs/lafs.h>
+
+struct lafs_ino *lafs_import_inode(struct lafs_dblk *db)
+{
+ struct lafs_ino *parent = db->b.ino;
+ struct lafs_ino *ino = lafs_import_inode_buf(parent->fs, db->b.data, db->b.fileaddr, parent);
+
+ ino->dblock = db;
+ db->my_inode = ino;
+ return ino;
+}
--- /dev/null
+
+/*
+ * create a new inode using the disk-image info in the given
+ * buf. Inode is in the inodefile 'parent'.
+ * This should not be used directly. It is only separate from
+ * lafs_import_inode so that lafs_get_itable can break a loop.
+ */
+
+#include <lafs/lafs.h>
+#include <talloc.h>
+#include <memory.h>
+
+static int inode_load(struct lafs_ino *ino, struct la_inode *lai);
+
+
+struct lafs_ino *lafs_import_inode_buf(struct lafs *fs,
+ char *buf, int inum,
+ struct lafs_ino *parent)
+{
+ struct lafs_ino *ino;
+
+ ino = talloc((void*)parent ?: (void*)fs, struct lafs_ino);
+
+ memset(ino, 0, sizeof(*ino));
+
+ ino->fs = fs;
+ ino->inum = inum;
+ ino->filesys = parent;
+
+ inode_load(ino, (struct la_inode *)buf);
+ /* FIXME callers of this should reparent it to themselves. */
+
+ return ino;
+}
+
+
+static int inode_load(struct lafs_ino *ino, struct la_inode *lai)
+{
+ if (lai->filetype == 0) {
+ ino->type = 0;
+ return 0; /* no inode here */
+ }
+
+ ino->cblocks = __le32_to_cpu(lai->data_blocks);
+ ino->ciblocks = __le32_to_cpu(lai->index_blocks);
+ ino->generation = __le16_to_cpu(lai->generation);
+ ino->trunc_gen = lai->trunc_gen;
+ ino->flags = lai->flags;
+ ino->type = lai->filetype;
+ ino->metadata_size = __le16_to_cpu(lai->metadata_size);
+ ino->depth = lai->depth;
+
+ switch(ino->type) {
+ case TypeInodeFile:
+ {
+ struct fs_md *i = &ino->md.fs;
+ struct fs_metadata *l = &lai->metadata[0].fs;
+ i->usagetable = __le16_to_cpu(l->snapshot_usage_table);
+ i->update_time = __le64_to_cpu(l->update_time);
+ i->cblocks_used = __le64_to_cpu(l->blocks_used);
+ i->blocks_allowed = __le64_to_cpu(l->blocks_allowed);
+ i->creation_age = __le64_to_cpu(l->creation_age);
+ i->inodes_used = __le32_to_cpu(l->inodes_used);
+ i->quota_inums[0] = __le32_to_cpu(l->quota_inodes[0]);
+ i->quota_inums[1] = __le32_to_cpu(l->quota_inodes[1]);
+ i->quota_inums[2] = __le32_to_cpu(l->quota_inodes[2]);
+ i->quota_inodes[0] = i->quota_inodes[1] = i->quota_inodes[2] = NULL;
+ memcpy(i->name, l->name, 64);
+ i->name[64] = 0;
+ break;
+ }
+ case TypeInodeMap:
+ {
+ struct inodemap_md *m = &ino->md.inodemap;
+ struct inodemap_metadata *s = &lai->metadata[0].inodemap;
+ m->size = __le32_to_cpu(s->size);
+ break;
+ }
+ case TypeSegmentMap:
+ {
+ struct su_md *m = &ino->md.segmentusage;
+ struct su_metadata *s = &lai->metadata[0].segmentusage;
+ m->table_size = __le32_to_cpu(s->table_size);
+ break;
+ }
+ case TypeQuota:
+ {
+ struct quota_md *m = &ino->md.quota;
+ struct quota_metadata *s = &lai->metadata[0].quota;
+ m->gracetime = __le32_to_cpu(s->gracetime);
+ m->graceunits = __le32_to_cpu(s->graceunits);
+ break;
+ }
+ case TypeOrphanList:
+ case TypeAccessTime:
+ break;
+ default: /* TypeBase or larger */
+ {
+ struct file_md *i = &ino->md.file;
+ struct file_metadata *l = &lai->metadata[0].file;
+ struct dir_metadata *d = &lai->metadata[0].dir;
+ struct special_metadata *s = &lai->metadata[0].special;
+
+ if (ino->type < TypeBase) abort();
+ i->flags = __le16_to_cpu(l->flags);
+ i->mode = __le16_to_cpu(l->mode);
+ i->uid = __le32_to_cpu(l->userid);
+ i->gid = __le32_to_cpu(l->groupid);
+ i->treeid = __le32_to_cpu(l->treeid);
+ i->creationtime = __le64_to_cpu(l->creationtime);
+ i->modifytime = __le64_to_cpu(l->modifytime);
+ i->ctime = __le64_to_cpu(l->ctime);
+ i->i_accesstime = __le64_to_cpu(l->accesstime);
+ i->accesstime = i->i_accesstime; /* FIXME load from accesstime file */
+ i->size = __le64_to_cpu(l->size);
+ i->parent = __le32_to_cpu(l->parent);
+ i->linkcount = __le32_to_cpu(l->linkcount);
+
+ if (ino->type == TypeDir)
+ i->seed = __le32_to_cpu(d->hash_seed);
+ if (ino->type == TypeSpecial) {
+ i->major = __le32_to_cpu(s->major);
+ i->minor = __le32_to_cpu(s->minor);
+ }
+
+ INIT_LIST_HEAD(&i->dirorphan);
+ break;
+ }
+ }
+ /* FIXME should I sanity-check anything? */
+ return 1;
+
+}
--- /dev/null
+
+/*
+ * Given an index block with some ->uninc blocks, add all the addresses
+ * of those blocks into the index info.
+ *
+ * FIXME
+ * For now we only support Indirect index in the inode as that is enough
+ * for mkfs
+ */
+
+#include <lafs/lafs.h>
+#include <stdlib.h>
+#include <stdio.h>
+
+#define encode16(p,n) ({ *(p++) = (n)&255; *(p++) = ((n)>>8) & 255; })
+#define encode32(p,n) ({ encode16(p,n); encode16(p, ((n)>>16)); })
+#define encode48(p,n) ({ encode32(p,n); encode16(p, ((n)>>32)); })
+
+void lafs_incorporate(struct lafs_iblk *ib)
+{
+ struct lafs *fs = ib->b.ino->fs;
+ unsigned char *ip;
+ int max_addr;
+
+ if (ib->uninc == NULL)
+ return;
+
+ if (ib->depth != 1)
+ abort();
+
+
+ ip = (void*)(ib->b.ino->dblock->b.data + ib->b.ino->metadata_size);
+ if (*(u16*)ip != __cpu_to_le16(IBLK_INDIRECT))
+ abort();
+ ip += 2;
+
+ max_addr = fs->blocksize - ib->b.ino->metadata_size;
+ max_addr -= 2; /* for type field */
+ max_addr /= 6; /* 6 bytes per address */
+
+ while (ib->uninc) {
+ struct lafs_blk *b = ib->uninc;
+ int addr;
+ unsigned char *ip2 = ip;
+
+ ib->uninc = b->chain;
+ b->chain = NULL;
+
+ addr = b->fileaddr - ib->b.fileaddr;
+ if (addr >= max_addr)
+ abort();
+ ip2 += addr * 6;
+ encode48(ip2, b->physaddr);
+ }
+ printf("incorp %d/%ld\n", ib->b.ino->dblock->b.ino->inum, ib->b.ino->dblock->b.fileaddr);
+ lafs_dirty_blk(&ib->b);
+}
+
+
--- /dev/null
+
+/*
+ * Fill a data buffer with info to look like an on-disk inode
+ */
+
+#include <stdlib.h>
+#include <lafs/lafs.h>
+#include <memory.h>
+#include <stdio.h>
+
+void lafs_inode_fillblock(struct lafs_ino *ino, char *buf)
+{
+ struct la_inode *lino = (void*)buf;
+
+ lino->data_blocks = __cpu_to_le32(ino->cblocks);
+ lino->index_blocks = __cpu_to_le32(ino->ciblocks);
+ lino->generation = __cpu_to_le16(ino->generation);
+ lino->metadata_size = __cpu_to_le16(ino->metadata_size);
+ lino->depth = ino->depth;
+ lino->trunc_gen = ino->trunc_gen;
+ lino->filetype = ino->type;
+ lino->flags = ino->flags;
+
+ printf("FILL %d\n", ino->type);
+ switch(ino->type) {
+ case TypeInodeFile:
+ {
+ struct fs_metadata *sfs = &lino->metadata[0].fs;
+ struct fs_md *mfs = &ino->md.fs;
+ sfs->update_time = __cpu_to_le64(mfs->update_time);
+ sfs->blocks_used = __cpu_to_le64(mfs->cblocks_used);
+ sfs->blocks_allowed = __cpu_to_le64(mfs->blocks_allowed);
+ sfs->creation_age = __cpu_to_le64(mfs->creation_age);
+ sfs->inodes_used = __cpu_to_le32(mfs->inodes_used);
+ sfs->snapshot_usage_table = __cpu_to_le16(mfs->usagetable);
+ sfs->quota_inodes[0] = __cpu_to_le32(mfs->quota_inums[0]);
+ sfs->quota_inodes[1] = __cpu_to_le32(mfs->quota_inums[1]);
+ sfs->quota_inodes[2] = __cpu_to_le32(mfs->quota_inums[2]);
+ memcpy(sfs->name, mfs->name, 64);
+ break;
+ }
+ case TypeInodeMap:
+ {
+ struct inodemap_metadata *sim = &lino->metadata[0].inodemap;
+ struct inodemap_md *mim = &ino->md.inodemap;
+ sim->size = __cpu_to_le32(mim->size);
+ break;
+ }
+ case TypeSegmentMap:
+ {
+ struct su_metadata *s = &lino->metadata[0].segmentusage;
+ struct su_md *m = &ino->md.segmentusage;
+ s->table_size = __cpu_to_le32(m->table_size);
+ break;
+ }
+ case TypeQuota:
+ {
+ struct quota_metadata *s = &lino->metadata[0].quota;
+ struct quota_md *m = &ino->md.quota;
+ s->gracetime = __cpu_to_le32(m->gracetime);
+ s->graceunits = __cpu_to_le32(m->graceunits);
+ break;
+ }
+ case TypeOrphanList:
+ case TypeAccessTime:
+ break;
+ default: /* external-file */
+ {
+ struct file_metadata *sfl = & lino->metadata[0].file;
+ struct file_md *mfl = & ino->md.file;
+ struct dir_metadata *sdr = & lino->metadata[0].dir;
+ struct special_metadata *ssp = & lino->metadata[0].special;
+
+ if (ino->type < TypeBase) abort();
+ sfl->flags = __cpu_to_le16(mfl->flags);
+ sfl->mode = __cpu_to_le16(mfl->mode);
+ sfl->userid = __cpu_to_le32(mfl->uid);
+ sfl->groupid = __cpu_to_le32(mfl->gid);
+ sfl->treeid = __cpu_to_le32(mfl->treeid);
+ sfl->creationtime = __cpu_to_le64(mfl->creationtime);
+ sfl->modifytime = __cpu_to_le64(mfl->modifytime);
+ sfl->ctime = __cpu_to_le64(mfl->ctime);
+ sfl->accesstime = __cpu_to_le64(mfl->i_accesstime);
+ /* FIXME should we zero the accesstime file at this point? */
+ sfl->size = __cpu_to_le64(mfl->size);
+ sfl->parent = __cpu_to_le32(mfl->parent);
+ sfl->linkcount = __cpu_to_le32(mfl->linkcount);
+ sfl->attrinode = 0;
+
+ if (ino->type == TypeDir)
+ sdr->hash_seed = __cpu_to_le32(mfl->seed);
+ if (ino->type == TypeSpecial) {
+ ssp->major = __cpu_to_le32(mfl->major);
+ ssp->minor = __cpu_to_le32(mfl->minor);
+ }
+
+ break;
+ }
+ }
+}
--- /dev/null
+
+/*
+ * Fill out a block to look like an on-disk inode
+ */
+
+#include <unistd.h>
+#include <stdlib.h>
+#include <lafs/lafs.h>
+#include <memory.h>
+#include <time.h>
+
+void lafs_inode_init(struct lafs *fs, char * buf, int type)
+{
+ struct lafs_ino ino;
+ unsigned char *ip;
+
+ memset(&ino, 0, sizeof(ino));
+
+ ino.generation = random();
+ ino.trunc_gen = random() & 0xff;
+ ino.flags = 0;
+ ino.type = type;
+
+ ino.depth = 1;
+ memset(&ino.md, 0, sizeof(ino.md));
+ switch(type) {
+ case TypeInodeFile:
+ ino.metadata_size = sizeof(struct la_inode) + sizeof(struct fs_metadata);
+ ino.depth = 1; /*never depth 0 for inode files */
+ ino.md.fs.update_time = time(0); /* FIXME */
+ break;
+
+ case TypeInodeMap:
+ ino.metadata_size = sizeof(struct la_inode) + sizeof(struct inodemap_metadata);
+ ino.md.inodemap.size = 0;
+ break;
+
+ case TypeSegmentMap:
+ ino.metadata_size = sizeof(struct la_inode) + sizeof(struct su_metadata);
+ ino.md.segmentusage.table_size = 0; /* FIXME */
+ break;
+
+ case TypeQuota:
+ ino.metadata_size = sizeof(struct la_inode) + sizeof(struct quota_metadata);
+ ino.md.quota.gracetime = 0;
+ ino.md.quota.graceunits = 0;
+ break;
+
+ case TypeOrphanList:
+ case TypeAccessTime:
+ ino.metadata_size = sizeof(struct la_inode);
+ break;
+
+ default: /* external file */
+ if (type < TypeBase) abort();
+ ino.metadata_size = sizeof(struct la_inode) + sizeof(struct file_metadata);
+ if (type == TypeDir)
+ ino.metadata_size = sizeof(struct la_inode) + sizeof(struct dir_metadata);
+ ino.md.file.flags = 0;
+ ino.md.file.mode = 0600;
+ ino.md.file.creationtime = time(0);
+ ino.md.file.modifytime = time(0);
+ ino.md.file.ctime = time(0);
+ ino.md.file.accesstime = time(0);
+ ino.md.file.i_accesstime = ino.md.file.accesstime;
+
+ ino.md.file.seed = (random() & ~7) | 1;
+ INIT_LIST_HEAD(&ino.md.file.dirorphan);
+ break;
+ }
+
+ ip = (void*)(buf + ino.metadata_size);
+ *(u16*)ip = __cpu_to_le16(IBLK_INDIRECT);
+
+ lafs_inode_fillblock(&ino, buf);
+}
--- /dev/null
+
+/*
+ * load the content of this dblk from the filesystem, or maybe
+ * from the inode
+ */
+#include <lafs/lafs.h>
+#include <memory.h>
+
+int lafs_load_dblk(struct lafs_dblk *db)
+{
+ struct lafs_ino *ino = db->b.ino;
+ struct lafs *fs = ino->fs;
+ struct lafs_dblk *inodb;
+ int offset;
+
+ if (db->b.flags & B_Valid)
+ return 0;
+
+ lafs_find_dblk(db);
+
+ db->b.flags |= B_Valid;
+ if (db->b.physaddr)
+ return lafs_read_virtual(fs, db->b.data,
+ db->b.physaddr);
+
+ memset(db->b.data, 0, fs->blocksize);
+ if (ino->depth > 0)
+ return 0;
+
+ /* content is in the inode dblock */
+ inodb = ino->dblock;
+ offset = ino->metadata_size;
+ memcpy(db->b.data, inodb->b.data + offset, fs->blocksize - offset);
+ return 0;
+}
--- /dev/null
+
+/*
+ * Allocate an iblock for the given inode
+ */
+#include <lafs/lafs.h>
+#include <memory.h>
+#include <talloc.h>
+#include <lafs/list.h>
+
+void lafs_make_iblock(struct lafs_ino *ino)
+{
+ struct lafs_iblk *ib;
+ if (ino->iblock)
+ return;
+
+ ib = talloc(ino, struct lafs_iblk);
+ memset(ib, 0, sizeof(*ib));
+
+ ib->depth = ino->depth;
+ INIT_LIST_HEAD(&ib->children);
+ ib->b.flags |= B_Index | B_InoIdx;
+ ib->b.ino = ino;
+ INIT_LIST_HEAD(&ib->b.hash);
+ ib->b.parent = ino->dblock->b.parent;
+ INIT_LIST_HEAD(&ib->b.siblings);
+ INIT_LIST_HEAD(&ib->b.leafs);
+
+ ino->iblock = ib;
+}
--- /dev/null
+
+#include <lafs/lafs.h>
+#include <unistd.h>
+#include <fcntl.h>
+
+int lafs_new(struct lafs *fs, int blockbytes)
+{
+ int fd;
+ int n;
+
+ if (fs->blocksize != 0)
+ return -1;
+
+ if (blockbytes < 512 ||
+ blockbytes > 4096 ||
+ (blockbytes & (blockbytes-1)) != 0)
+ return -1;
+
+
+ fd = open("/dev/urandom", O_RDONLY);
+ if (fd < 0)
+ return -1;
+
+ n = read(fd, fs->uuid, 16);
+ close(fd);
+ if (n != 16)
+ return -1;
+
+ fs->blocksize = blockbytes;
+
+ fs->seq = 1;
+ fs->statesize = 1024;
+
+ fs->wc[0].seq = 1;
+ fs->wc[1].seq = 1;
+
+ fs->youth_next = 1;
+
+ return 0;
+}
--- /dev/null
+
+/* allocate a new segment at assign it to a lafs_cluster
+ */
+
+#include <lafs/lafs.h>
+
+static void cluster_reset(struct lafs *fs, struct lafs_cluster *wc)
+{
+ wc->remaining = dev_by_num(fs, wc->seg.dev)->segment_size;
+ wc->chead_blocks = 1;
+ wc->remaining--;
+ wc->chead_size = sizeof(struct cluster_head);
+}
+
+void set_youth(struct lafs *fs, int dev, loff_t seg, int youth)
+{
+ struct lafs_device *dv;
+ struct lafs_dblk *db;
+ uint16_t *p;
+ loff_t addr;
+
+ dv = dev_by_num(fs, dev);
+ addr = seg / (fs->blocksize/2);
+
+ db = lafs_dblk(dv->segsum, addr);
+ lafs_load_dblk(db);
+ p = (void*)db->b.data;
+ p[seg % (fs->blocksize/2)] = __cpu_to_le16(youth);
+ lafs_dirty_blk(&db->b);
+}
+
+int lafs_new_segment(struct lafs *fs, int cnum)
+{
+ int dev;
+ loff_t seg;
+ struct lafs_cluster *wc = &fs->wc[cnum];
+
+ if (fs->free_head == fs->free_tail)
+ /* list is empty */
+ return 0;
+
+ dev = fs->freelist[fs->free_head].dev;
+ seg = fs->freelist[fs->free_head].seg;
+
+ fs->free_head++;
+ if (fs->free_head > 128)
+ fs->free_head -= 128;
+
+ wc->seg.dev = dev;
+ wc->seg.num = seg;
+ wc->seg.st_table = 0;
+ wc->seg.st_row = 0;
+ wc->seg.table = wc->seg.nxt_table = wc->seg.st_table;
+ wc->seg.row = wc->seg.nxt_row = wc->seg.st_row;
+ wc->seg.col = 0;
+
+ cluster_reset(fs, wc);
+
+ /* Need to set the youth for this segment */
+ set_youth(fs, dev, seg, fs->youth_next++);
+ return 1;
+}
--- /dev/null
+/*
+ * read a block given a virtual address
+ */
+#define _GNU_SOURCE
+#define _FILE_OFFSET_BITS 64
+#include <unistd.h>
+#include <lafs/lafs.h>
+
+
+int lafs_read_virtual(struct lafs *fs, char *buf, loff_t addr)
+{
+ struct lafs_device *d = lafs_dev_find(fs, addr);
+ int n;
+
+ if (!d)
+ return -1;
+ lseek64(d->fd, (addr - d->start) * fs->blocksize, SEEK_SET);
+ n = read(d->fd, buf, fs->blocksize);
+
+
+ if (n == fs->blocksize)
+ return 0;
+ return -1;
+}
--- /dev/null
+
+/*
+ * schedule a block for writeout in the next checkpoint
+ */
+
+#include <lafs/lafs.h>
+
+#include <stdio.h>
+
+int lafs_sched_blk(struct lafs_blk *blk)
+{
+ struct lafs *fs = blk->ino->fs;
+
+ if (blk->flags & B_Sched)
+ return 0;
+
+ if (!(blk->flags & B_Index)) {
+ struct lafs_dblk *dblk = container_of(blk, struct lafs_dblk, b);
+ if (lafs_find_dblk(dblk) < 0)
+ return -1;
+ }
+ blk->flags |= B_Sched;
+ if (blk->parent)
+ blk->parent->sched_cnt++;
+ printf("add %p\n", blk);
+ list_add(&blk->leafs, &fs->leafs);
+ return 0;
+}
--- /dev/null
+
+/* lafs_segment_count
+ * adjust the usage count for the segment containing a given block
+ */
+
+#include <lafs/lafs.h>
+
+static void delay_update(struct lafs *fs, int dev, loff_t seg, int diff)
+{
+ struct lafs_delayed *d = fs->delayed;
+
+ while (d && (d->dev != dev || d->seg != seg))
+ d = d->next;
+ if (!d) {
+ d = malloc(sizeof(*d));
+ d->next = fs->delayed;
+ fs->delayed = d;
+ d->dev = dev;
+ d->seg = seg;
+ d->diff = 0;
+ }
+ d->diff += diff;
+}
+
+void segment_count(struct lafs *fs, int dev, loff_t seg, int diff)
+{
+ struct lafs_device *dv;
+ struct lafs_dblk *db;
+ uint16_t *p;
+ int cnt;
+ loff_t addr;
+
+ dv = dev_by_num(fs, dev);
+ addr = dv->tablesize + seg / (fs->blocksize/2);
+
+ db = lafs_dblk(dv->segsum, addr);
+ lafs_load_dblk(db);
+ p = (void*)db->b.data;
+ cnt = __le16_to_cpu(p[seg % (fs->blocksize/2)]);
+ cnt += diff;
+ p[seg % (fs->blocksize/2)] = __cpu_to_le16(cnt);
+ lafs_dirty_blk(&db->b);
+}
+
+
+void lafs_segment_count(struct lafs *fs, loff_t addr, int diff)
+{
+ int dev;
+ loff_t seg;
+ loff_t offset;
+
+ virttoseg(fs, addr, &dev, &seg, &offset);
+
+ if (fs->flags & LAFS_DELAY_UPDATES) {
+ delay_update(fs, dev, seg, diff);
+ return;
+ }
+ segment_count(fs, dev, seg, diff);
+}
+
+void lafs_segment_apply_delayed(struct lafs *fs)
+{
+ struct lafs_delayed *delayed = fs->delayed;
+
+ if (fs->flags & LAFS_DELAY_UPDATES)
+ abort();
+ fs->delayed = NULL;
+
+ while (delayed) {
+ struct lafs_delayed *d = delayed;
+ delayed = d->next;
+ segment_count(fs, d->dev, d->seg, d->diff);
+ free(d);
+ }
+}
--- /dev/null
+
+/*
+ * Updates summary info for this newly written block.
+ * This includes:
+ * block count in the inode
+ * block count in the fs
+ * quotas
+ * segment usage.
+ */
+
+#include <lafs/lafs.h>
+
+void lafs_summary_update(struct lafs_ino *ino,
+ loff_t oldaddr, loff_t newaddr,
+ int is_index)
+{
+ struct lafs *fs = ino->fs;
+ int diff;
+
+ if (oldaddr && newaddr) {
+ /* Not a new allocation */
+ lafs_segment_count(fs, oldaddr, -1);
+ lafs_segment_count(fs, newaddr, 1);
+ return;
+ }
+ if (!oldaddr && !newaddr)
+ /* nothing changing! */
+ return;
+
+ if (oldaddr)
+ diff = -1;
+ else
+ diff = 1;
+
+ if (is_index)
+ ino->ciblocks += diff;
+ else
+ ino->cblocks += diff;
+
+ if (!is_index)
+ if (diff > 0)
+ ino->ablocks--;
+
+ ino = ino->filesys;
+ ino->md.fs.cblocks_used++;
+ if (!is_index)
+ if (diff > 0)
+ ino->md.fs.ablocks_used--;
+
+ if (oldaddr)
+ lafs_segment_count(fs, oldaddr, -1);
+ if (newaddr)
+ lafs_segment_count(fs, newaddr, 1);
+}
+
--- /dev/null
+
+#define _GNU_SOURCE
+#define _FILE_OFFSET_BITS 64
+#include <unistd.h>
+#include <lafs/lafs.h>
+#include <memory.h>
+
+static int size_bits(long long size)
+{
+ int bits = 0;
+ while (size > 1) {
+ size >>= 1;
+ bits++;
+ }
+ return bits;
+}
+
+int lafs_write_dev(struct lafs_device *dev)
+{
+ char buf[LAFS_DEVBLK_SIZE];
+ struct lafs_dev *pd = (void*)buf;
+ struct lafs *fs = dev->fs;
+ int i;
+ uint32_t csum;
+
+ memset(buf, 0, sizeof(buf));
+
+ memcpy(pd->idtag, "LaFS-DeviceBlock", 16);
+ memset(pd->version, ' ', 16);
+
+ memcpy(pd->uuid, fs->uuid, 16);
+ dev->seq++;
+ pd->seq = __cpu_to_le32(dev->seq);
+ pd->ctime = lafs_encode_timeval(&dev->ctime);
+ pd->start = __cpu_to_le64(dev->start);
+ pd->size = __cpu_to_le64(dev->size);
+ for (i=0; i<2; i++)
+ pd->devaddr[i] = __cpu_to_le64(dev->devaddr[i]);
+ for (i=0; i<4; i++)
+ pd->stateaddr[i] = __cpu_to_le64(dev->stateaddr[i]);
+
+ pd->statebits = size_bits(fs->statesize);
+ pd->blockbits = size_bits(fs->blocksize);
+ pd->width = __cpu_to_le16(dev->width);
+ pd->stride = __cpu_to_le32(dev->stride);
+ pd->segment_size = __cpu_to_le32(dev->segment_size);
+ pd->segment_offset = __cpu_to_le32(dev->segment_offset);
+ pd->segment_count = __cpu_to_le32(dev->segment_count);
+ pd->usage_inum = __cpu_to_le32(dev->usage_inum);
+ pd->level = __cpu_to_le32(1);
+
+ csum = crc32(0, (uint32_t*)buf, LAFS_DEVBLK_SIZE);
+ pd->checksum = csum;
+
+ for (i=0; i<2; i++) {
+ lseek64(dev->fd, dev->devaddr[i], SEEK_SET);
+ write(dev->fd, buf, LAFS_DEVBLK_SIZE);
+ }
+
+ return 0;
+}
--- /dev/null
+
+/*
+ * Write the state blocks twice to each device.
+ */
+
+#define _GNU_SOURCE
+#define _FILE_OFFSET_BITS 64
+#include <unistd.h>
+#include <lafs/lafs.h>
+#include <memory.h>
+
+int lafs_write_state(struct lafs *fs)
+{
+ char buf[4096];
+ struct lafs_state *st = (void*)buf;
+ struct lafs_device *dev;
+
+ memset(buf, 0, 4096);
+
+ fs->seq++;
+
+ memcpy(st->idtag, "LaFS-State-Block", 16);
+ memcpy(st->uuid, fs->uuid, 16);
+ memset(st->version, ' ', 16);
+
+ st->seq = __cpu_to_le32(fs->seq);
+ st->nextyouth = __cpu_to_le16(fs->youth_next);
+ st->checkpointcluster = __cpu_to_le64(fs->checkpoint_cluster);
+ st->root_inodes[0] = __cpu_to_le64(fs->ss.root_addr);
+ st->maxsnapshot = 1;
+ st->devices = __cpu_to_le32(fs->devices);
+
+ st->checksum = crc32(0, (uint32_t *)buf, fs->blocksize);
+
+ for (dev = fs->devs ; dev ; dev = dev->next) {
+ int n;
+ for (n = (fs->seq & 1); n < 4 ; n += 2) {
+ lseek64(dev->fd, dev->stateaddr[n], SEEK_SET);
+ write(dev->fd, buf, fs->statesize);
+ }
+ }
+ return 1;
+}
--- /dev/null
+/*
+ * write a block given a virtual address
+ */
+#define _GNU_SOURCE
+#define _FILE_OFFSET_BITS 64
+#include <unistd.h>
+#include <lafs/lafs.h>
+
+
+int lafs_write_virtual(struct lafs *fs, char *buf, loff_t addr)
+{
+ struct lafs_device *d = lafs_dev_find(fs, addr);
+ int n;
+
+ if (!d)
+ return -1;
+ lseek64(d->fd, (addr - d->start) * fs->blocksize, SEEK_SET);
+ n = write(d->fd, buf, fs->blocksize);
+
+
+ if (n == fs->blocksize)
+ return 0;
+ return -1;
+}
--- /dev/null
+#
+# lafs-utils - tools for working wit LAFS filesystems.
+#
+# Copyright (C) 2010 Neil Brown <neil@brown.name>
+#
+#
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+#
+# Author: Neil Brown
+# Email: <neil@brown.name>
+
+CPPFLAGS = -I../include
+CFLAGS = -Wall -Werror -g
+LDFLAGS = -L../lib
+LDLIBS = -llafs -ltalloc
+
+all : mkfs.lafs
+
+mkfs.lafs : mkfs.lafs.o ../lib/liblafs.a
--- /dev/null
+/*
+ * mkfs.lafs - Create an empty LAFS filesystem
+ *
+ * Copyright (C) 2010 NeilBrown <neil@brown.name>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ *
+ * Author: Neil Brown
+ * Email: <neil@brown.name>
+ *
+ */
+
+#include <sys/types.h>
+#include <unistd.h>
+#include <stdlib.h>
+#include <stdio.h>
+#include <getopt.h>
+#include <sys/stat.h>
+#include <sys/ioctl.h>
+#include <sys/mount.h>
+#include <fcntl.h>
+#include <errno.h>
+#include <string.h>
+#include <lafs/lafs.h>
+#include <talloc.h>
+
+/*
+ * A new filesystem must contain:
+ * inode 0 - TypeInodeFile with all the files listed here
+ * inode 1 - TypeInodeMap - empty
+ * inode 2 - TypeDir - empty
+ * inode 8 - TypeOrphanList - empty
+ * inode 16 - TypeSegmentMap - youth block and usage block for first segment
+ *
+ * These can all be in one checkpoint and could be in a single
+ * write-cluster, though that could be a bit awkward.
+ *
+ * As well as the write-clusters we need device-blocks and state-blocks.
+ *
+ * Configurable values are:
+ * - block size, 512 to 4096 -b or --block-size
+ * - segment size - up to 64k, but multiple of width --segment-size
+ * - width - spindles across which device is striped --width
+ * - stride - chunk-size when device is striped --stride
+ *
+ */
+
+enum {
+ opt_segsize = 1000,
+ opt_width,
+ opt_stride,
+};
+
+char short_options[] = "-b:Vvh";
+struct option long_options[] = {
+ {"block-size", 1, 0, 'b'},
+ {"segment-size", 1, 0, opt_segsize},
+ {"width", 1, 0, opt_width},
+ {"stride", 1, 0, opt_stride},
+ {"version", 0, 0, 'V'},
+ {"verbose", 0, 0, 'v'},
+ {"help", 0, 0, 'h'},
+ {0, 0, 0, 0}
+};
+
+char version_text[] = "mkfs.lafs - unreleased\n";
+
+char help_text[] =
+ "mkfs.lafs: create a lafs filesystem\n"
+ ;
+
+char usage_text[] =
+ "Usage: mkfs.lafs [options] device-name\n"
+ " Options:\n"
+ " --block-size (-b) size of basic block, up to 4096 (e.g. 2K)\n"
+ " --segment-size size of segments, up to 65536 blocks (e.g. 32M)\n"
+ " --width number of members of a striped device (e.g. 3)\n"
+ " --stride chunk size of a striped device (e.g. 64K)\n"
+ " --help (-h) This help message\n"
+ " --version (-V) Report version information\n"
+ " --verbose (-v) Be more verbose\n"
+ ;
+
+static void get_size(long *valp, char *arg, char *name)
+{
+ long val;
+ long scale = 1;
+ char *endp;
+
+ if (*valp != 0) {
+ fprintf(stderr, "mkfs.lafs: %s has already been given, value \"%s\" not permitted.\n",
+ name, arg);
+ exit(2);
+ }
+
+ val = strtol(arg, &endp, 0);
+
+ if (endp == arg) {
+ fprintf(stderr, "mkfs.lafs: Unrecognised size \"%s\" for %s\n", arg, name);
+ exit(2);
+ }
+
+ switch(*endp) {
+ case 'k':
+ case 'K':
+ scale = 1024;
+ break;
+ case 'M':
+ scale = 1024*1024;
+ break;
+ case 'G':
+ scale = 1024*1024*1024;
+ break;
+ case '\0':
+ scale = 1;
+ break;
+ default:
+ fprintf(stderr, "mkfs.lafs: unrecognised modifier \"%s\" for %s\n", endp, name);
+ exit(2);
+ }
+ if (val == 0) {
+ fprintf(stderr, "mkfs.lafs: 0 is not a valid number for %s\n", name);
+ exit(2);
+ }
+ val *= scale;
+ *valp = val;
+}
+
+void get_num(int *valp, char *arg, char *name)
+{
+ long val;
+ char *endp;
+
+ if (*valp != 0) {
+ fprintf(stderr, "mkfs.lafs: %s has already been given, value \"%s\" not permitted.\n",
+ name, arg);
+ exit(2);
+ }
+
+ val = strtol(arg, &endp, 0);
+ if (endp == arg || *endp) {
+ fprintf(stderr, "mkfs.lafs: Unrecognised number \"%s\" for %s\n", arg, name);
+ exit(2);
+ }
+ if (val == 0) {
+ fprintf(stderr, "mkfs.lafs: 0 is not a valid number for %s\n", name);
+ exit(2);
+ }
+ *valp = val;
+}
+
+int is_pow2(long num)
+{
+ return (num & (num-1)) == 0;
+}
+
+
+void validate_parameters(long *block_bytes, long *segment_bytes, long *stride_bytes, int *width)
+{
+
+ if (*block_bytes == 0)
+ *block_bytes = 1024;
+
+ if (*block_bytes < 512 || *block_bytes > 4096 ||
+ !is_pow2(*block_bytes)) {
+ fprintf(stderr, "lafs.mkfs: block size %ld is illegal - must be power of 2 in range 512..4096\n",
+ *block_bytes);
+ exit(2);
+ }
+
+ if (*width == 0)
+ *width = 1;
+ if (*width < 1 || *width > 128) {
+ fprintf(stderr, "lafs.mkfs: width %d is illegal - must be in range 1..128\n", *width);
+ exit(2);
+ }
+
+ if (*stride_bytes == 0)
+ *stride_bytes = *block_bytes;
+
+ if (*stride_bytes < 0 || (*stride_bytes % *block_bytes) != 0) {
+ fprintf(stderr, "lafs.mkfs: stride %ld is illegal - must be a multiple of block size\n",
+ *stride_bytes);
+ exit(2);
+ }
+
+ /* segment size must be a multiple of block size and of width.
+ * It must either be a multiple or a sub-multiple of stride size
+ */
+ if (*segment_bytes == 0) {
+ /* choose maximum size ?? */
+ long seg;
+ long blocks = 65536 / *width;
+ blocks *= *width;
+ seg = *block_bytes * blocks;
+ if (*stride_bytes < seg) {
+ seg /= *stride_bytes;
+ seg *= *stride_bytes;
+ } else {
+ if ((*stride_bytes % seg) != 0) {
+ fprintf(stderr, "lafs.mkfs: explicit segment size must be given with large stride\n");
+ exit(2);
+ }
+ }
+ *segment_bytes = seg;
+ }
+
+ if (*segment_bytes % *block_bytes) {
+ fprintf(stderr, "lafs.mkfs: segment size must be a multiple of block size\n");
+ exit(2);
+ }
+ if (*segment_bytes % *width) {
+ fprintf(stderr, "lafs.mkfs: segment size must be a multiple of width\n");
+ exit(2);
+ }
+ if (*segment_bytes > *stride_bytes &&
+ (*segment_bytes % *stride_bytes)) {
+ fprintf(stderr, "lafs.mkfs: segment size must be a multiple of stride size\n");
+ exit(2);
+ }
+ if (*segment_bytes < *stride_bytes &&
+ (*stride_bytes % *segment_bytes)) {
+ fprintf(stderr, "lafs.mkfs: segment size must be a divisor of stride size\n");
+ exit(2);
+ }
+}
+
+int open_device(char *devname, long long *device_bytes)
+{
+ /* must be able to get an exclusive open on the device and its size
+ * must be non-trivial
+ *
+ */
+ int fd;
+ struct stat stb;
+ unsigned long long size;
+
+ fd = open(devname, O_RDWR|O_EXCL);
+ if (fd < 0)
+ fprintf(stderr, "mkfs.lafs: cannot open device %s: %s\n",
+ devname, strerror(errno));
+ else if (fstat(fd, &stb) < 0 ||
+ (stb.st_mode & S_IFMT) != S_IFBLK)
+ fprintf(stderr, "mkfs.lafs: %s is not a block device\n",
+ devname);
+ else if (ioctl(fd, BLKGETSIZE64, &size) != 0)
+ fprintf(stderr, "mkfs.lafs: Cannot get size of %s\n",
+ devname);
+ else if (size < 64*1024)
+ fprintf(stderr, "mkfs.lafs: %s is too small for a LAFS filesystem\n",
+ devname);
+ else {
+ *device_bytes = size;
+ return fd;
+ }
+ exit(2);
+}
+
+int main(int argc, char *argv[])
+{
+ int verbose = 0;
+ long block_bytes = 0;
+ long segment_bytes = 0;
+ int width = 0;
+ long stride_bytes = 0;
+ long long device_bytes;
+ char *devname = NULL;
+ int opt;
+ int dev_fd;
+ struct lafs *lafs;
+ struct lafs_device *dev;
+ struct lafs_ino *ifile, *imfile, *rootdir, *orphans, *segmap;
+
+ while ((opt = getopt_long(argc, argv,
+ short_options, long_options,
+ NULL)) != -1) {
+ switch(opt) {
+ case 'h':
+ fputs(help_text, stdout);
+ fputs(usage_text, stdout);
+ exit(0);
+ case 'V':
+ fputs(version_text, stdout);
+ exit(0);
+ case 'v':
+ verbose++;
+ break;
+ case 'b':
+ get_size(&block_bytes, optarg, "block size");
+ break;
+ case opt_segsize:
+ get_size(&segment_bytes, optarg, "segment size");
+ break;
+ case opt_stride:
+ get_size(&stride_bytes, optarg, "stride size");
+ break;
+ case opt_width:
+ get_num(&width, optarg, "device width");
+ break;
+
+ case 1:
+ if (devname == NULL) {
+ devname = optarg;
+ break;
+ }
+ fprintf(stderr, "mkfs.lafs: multiple device names not supported: %s and %s\n",
+ devname, optarg);
+ exit(2);
+
+ case '?':
+ default:
+ fputs(usage_text, stderr);
+ exit(2);
+ }
+ }
+
+ if (devname == NULL) {
+ fputs("mkfs.lafs: no device name given\n", stderr);
+ fputs(usage_text, stderr);
+ exit(2);
+ }
+
+ /* Validate device */
+ dev_fd = open_device(devname, &device_bytes);
+
+ /* Validate parameters */
+ validate_parameters(&block_bytes, &segment_bytes, &stride_bytes, &width);
+
+ /* Create filesystem handle */
+ lafs = lafs_alloc();
+
+ /* Initialise filesystem */
+ lafs_new(lafs, block_bytes);
+
+ /* Add device */
+ dev = lafs_add_device(lafs, devname, dev_fd,
+ segment_bytes / block_bytes,
+ stride_bytes / block_bytes,
+ width,
+ 16);
+
+ /* Write device blocks */
+ lafs_write_dev(dev);
+
+ /* create files */
+ ifile = lafs_get_itable(lafs);
+ imfile = lafs_add_inode(ifile, 1, TypeInodeMap);
+ rootdir = lafs_add_inode(ifile, 2, TypeDir);
+ rootdir->md.file.linkcount = 2;
+ rootdir->md.file.mode = 0755;
+ rootdir->md.file.parent = 2;
+ lafs_dirty_inode(rootdir);
+ orphans = lafs_add_inode(ifile, 8, TypeOrphanList);
+ segmap = lafs_add_inode(ifile, 16, TypeSegmentMap);
+ lafs->devs->segsum = segmap;
+ segmap->md.segmentusage.table_size = lafs->devs->tablesize * 16;
+ lafs->devs->tablesize = segmap->md.segmentusage.table_size;
+ lafs_dirty_inode(segmap);
+
+ lafs_imap_set(imfile, 1);
+ lafs_imap_set(imfile, 2);
+ lafs_imap_set(imfile, 8);
+ lafs_imap_set(imfile, 16);
+
+ lafs_cluster_init(lafs, 0, 0, 0, 1);
+ lafs_add_free_seg(lafs, dev->devnum, 0);
+ /* Write checkpoint and state blocks */
+ lafs_checkpoint(lafs);
+ /* Write state blocks a second time, so all 4 copies are written */
+ lafs_write_state(lafs);
+
+ talloc_free(lafs);
+
+ exit(0);
+}