/* * Copyright (c) 2009, Swedish Institute of Computer Science * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of the Institute nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * * This file is part of the Contiki operating system. * * $Id: CoffeeHeader.java,v 1.2 2009/08/10 12:51:52 nvt-se Exp $ * * @author Nicolas Tsiftes * */ package se.sics.coffee; class CoffeeHeader { private CoffeeFS coffeeFS; private CoffeeConfiguration conf; private int page; private static final int HDR_FLAG_VALID = 0x1; private static final int HDR_FLAG_ALLOCATED = 0x2; private static final int HDR_FLAG_OBSOLETE = 0x4; private static final int HDR_FLAG_MODIFIED = 0x8; private static final int HDR_FLAG_LOG = 0x10; private static final int HDR_FLAG_ISOLATED = 0x20; int logPage; int logRecords; int logRecordSize; int maxPages; String name; private boolean valid; private boolean allocated; private boolean obsolete; private boolean modified; private boolean log; private boolean isolated; public CoffeeHeader(CoffeeFS coffeeFS, int page) { this.coffeeFS = coffeeFS; this.page = page; conf = coffeeFS.getConfiguration(); } public CoffeeHeader(CoffeeFS coffeeFS, int page, byte[] bytes) { this.coffeeFS = coffeeFS; this.page = page; processRawHeader(bytes); } private void processRawHeader(byte[] bytes) { int index = 0; logPage = getPageValue(bytes, 0); index += conf.pageTypeSize; logRecords = bytes[index] + (bytes[index + 1] << 8); index += 2; logRecordSize = bytes[index] + (bytes[index + 1] << 8); index += 2; maxPages = getPageValue(bytes, index); index += conf.pageTypeSize; index++; // Skip deprecated EOF hint field. processFlags((int)bytes[index]); index++; name = new String(bytes).substring(index, index + conf.NAME_LENGTH); int nullCharOffset = name.indexOf(0); if (nullCharOffset >= 0) { name = name.substring(0, nullCharOffset); } } private byte composeFlags() { byte flags = 0; if (valid) { flags |= HDR_FLAG_VALID; } if (allocated) { flags |= HDR_FLAG_ALLOCATED; } if (obsolete) { flags |= HDR_FLAG_OBSOLETE; } if (modified) { flags |= HDR_FLAG_MODIFIED; } if (log) { flags |= HDR_FLAG_LOG; } if (isolated) { flags |= HDR_FLAG_ISOLATED; } return flags; } private void processFlags(int flags) { if ((flags & HDR_FLAG_VALID) != 0) { valid = true; } if ((flags & HDR_FLAG_ALLOCATED) != 0) { allocated = true; } if ((flags & HDR_FLAG_OBSOLETE) != 0) { obsolete = true; } if ((flags & HDR_FLAG_MODIFIED) != 0) { modified = true; } if ((flags & HDR_FLAG_LOG) != 0) { log = true; } if ((flags & HDR_FLAG_ISOLATED) != 0) { isolated = true; } } private byte[] setPageValue(int page) { byte[] bytes = new byte[conf.pageTypeSize]; for (int i = conf.pageTypeSize - 1; i >= 0; i--) { bytes[i] = (byte) (page >> (8 * i)); } return bytes; } private int getPageValue(byte[] bytes, int offset) { int page = 0; for (int i = 0; i < conf.pageTypeSize; i++) { page |= bytes[offset + i] << (8 * i); } return page; } public byte[] toRawHeader() { byte[] bytes = new byte[2 * conf.pageTypeSize + conf.NAME_LENGTH + 6]; int index = 0; System.arraycopy(setPageValue(logPage), 0, bytes, 0, conf.pageTypeSize); index += conf.pageTypeSize; bytes[index++] = (byte) (logRecords >> 8); bytes[index++] = (byte) logRecords; bytes[index++] = (byte) (logRecordSize >> 8); bytes[index++] = (byte) logRecordSize; System.arraycopy(setPageValue(maxPages), 0, bytes, index, conf.pageTypeSize); index += conf.pageTypeSize; bytes[index++] = 0; // Deprecated EOF hint field. bytes[index++] = composeFlags(); byte[] nameBytes = name.getBytes(); int copyLength = nameBytes.length > conf.NAME_LENGTH ? conf.NAME_LENGTH : nameBytes.length; System.arraycopy(nameBytes, 0, bytes, index, copyLength); return bytes; } public int rawLength() { return 2 * conf.pageTypeSize + conf.NAME_LENGTH + 6; } public int getPage() { return page; } public boolean isValid() { return valid; } public boolean isAllocated() { return allocated; } public boolean isObsolete() { return obsolete; } public boolean isModified() { return modified; } public boolean isIsolated() { return isolated; } public boolean isLog() { return log; } public boolean isFree() { return !isAllocated(); } public boolean isActive() { return isAllocated() && !isObsolete() && !isIsolated(); } public void allocate() { allocated = true; } public void makeObsolete() { obsolete = true; } public void setName(String name) { this.name = name; } public void setReservedSize(int pages) { maxPages = pages; } public int getReservedSize() { return maxPages; } }