📄 delugemetadatap.nc
字号:
/* Copyright (c) 2007 Johns Hopkins University.* All rights reserved.** Permission to use, copy, modify, and distribute this software and its* documentation for any purpose, without fee, and without written* agreement is hereby granted, provided that the above copyright* notice, the (updated) modification history and the author appear in* all copies of this source code.** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 COPYRIGHT HOLDERS OR CONTRIBUTORS* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, LOSS OF USE, DATA,* OR PROFITS) 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.*//** * @author Razvan Musaloiu-E. <razvanm@cs.jhu.edu> * @author Chieh-Jan Mike Liang <cliang4@cs.jhu.edu> */#include "imgNum2volumeId.h"module DelugeMetadataP{ provides interface DelugeMetadata[uint8_t client]; uses { interface Boot; interface BlockRead[uint8_t volumeId]; interface BlockWrite[uint8_t volumeId]; interface StorageMap[uint8_t volumeId]; event void storageReady(); }}implementation{ enum { S_READ_IDENT, S_READ_CRC, S_CRC, S_READY, S_BUSY }; DelugeIdent ident; uint8_t state; uint8_t currentVolume; uint8_t currentImageIdx; uint8_t currentPage; nx_uint16_t currentCrc; uint8_t currentClient; void nextImage() { if (currentImageIdx < DELUGE_NUM_VOLUMES) { state = S_READ_IDENT; call BlockRead.read[currentVolume](0, &ident, sizeof(ident)); } else { signal storageReady(); state = S_READY; } } uint32_t calcCrcAddr() { return DELUGE_IDENT_SIZE + currentPage * sizeof(uint16_t); } uint32_t calcPageAddr() { return DELUGE_IDENT_SIZE + DELUGE_CRC_BLOCK_SIZE + currentPage * DELUGE_BYTES_PER_PAGE; } event void Boot.booted() { // We are going to iterate over all the images and verify their // integrity. For each image we first read the ident to find the // number of pages and then iterate over all of them, compute the // CRC and check it against the corresponding value from the CRCs // block. state = S_READ_IDENT; currentImageIdx = 0; currentVolume = _imgNum2volumeId[currentImageIdx]; nextImage(); } command error_t DelugeMetadata.read[uint8_t client](uint8_t imgNum) { error_t error; if (state != S_READY) { return FAIL; } currentClient = client; error = call BlockRead.read[imgNum](0, &ident, sizeof(ident)); state = error == SUCCESS ? S_BUSY : state; return error; } event void BlockRead.readDone[uint8_t imgNum](storage_addr_t addr, void* buf, storage_len_t len, error_t error) { switch (state) { case S_BUSY: state = S_READY; signal DelugeMetadata.readDone[currentClient](imgNum, buf, error); break; case S_READ_IDENT: if (error == SUCCESS) { if (ident.uidhash != DELUGE_INVALID_UID) { currentPage = 0; state = S_READ_CRC; call BlockRead.read[currentVolume](calcCrcAddr(), ¤tCrc, sizeof(currentCrc)); } else { currentImageIdx++; nextImage(); } } break; case S_READ_CRC: state = S_CRC; call BlockRead.computeCrc[currentVolume](calcPageAddr(), DELUGE_BYTES_PER_PAGE, 0); break; } } event void BlockRead.computeCrcDone[uint8_t imgNum](storage_addr_t addr, storage_len_t len, uint16_t crc, error_t error) { switch (state) { case S_CRC: if (crc != currentCrc) {// printf("%04x %04x\n", crc, currentCrc); // invalidate the image by erasing it call BlockWrite.erase[currentVolume](); } else { currentPage++; if (currentPage < ident.numPgs) { state = S_READ_CRC; call BlockRead.read[currentVolume](calcCrcAddr(), ¤tCrc, sizeof(currentCrc)); } else { currentImageIdx++; currentVolume = _imgNum2volumeId[currentImageIdx]; nextImage(); } } break; } } default command error_t BlockRead.read[uint8_t imgNum](storage_addr_t addr, void* buf, storage_len_t len) { return FAIL; } default command error_t BlockRead.computeCrc[uint8_t imgNum](storage_addr_t addr, storage_len_t len, uint16_t crc) { return FAIL; } event void BlockWrite.writeDone[uint8_t imgNum](storage_addr_t addr, void* buf, storage_len_t len, error_t error) {} event void BlockWrite.eraseDone[uint8_t imgNum](error_t error) { switch (state) { case S_READY: signal BlockWrite.eraseDone[imgNum](error); break; case S_CRC: currentImageIdx++; currentVolume = _imgNum2volumeId[currentImageIdx]; nextImage(); break; } } event void BlockWrite.syncDone[uint8_t imgNum](error_t error) {} default command error_t BlockWrite.erase[uint8_t imgNum]() { return FAIL; } default event void DelugeMetadata.readDone[uint8_t client](uint8_t imgNum, DelugeIdent* i, error_t error) {} default event void storageReady() {}}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -