mirror of
https://github.com/ksherlock/profuse.git
synced 2025-01-22 23:30:54 +00:00
120 lines
2.6 KiB
C++
120 lines
2.6 KiB
C++
|
|
||
|
#include <cstring>
|
||
|
#include <memory>
|
||
|
|
||
|
#include "Bitmap.h"
|
||
|
#include "BlockDevice.h"
|
||
|
#include "Buffer.h"
|
||
|
#include "Endian.h"
|
||
|
#include "Entry.h"
|
||
|
#include "Exception.h"
|
||
|
|
||
|
using namespace ProFUSE;
|
||
|
using namespace LittleEndian;
|
||
|
|
||
|
|
||
|
#pragma mark VolumeDirectory
|
||
|
|
||
|
VolumeDirectory::VolumeDirectory(const char *name, BlockDevice *device) :
|
||
|
Directory(VolumeHeader, name)
|
||
|
{
|
||
|
_totalBlocks = device->blocks();
|
||
|
_bitmapPointer = 6;
|
||
|
|
||
|
_entryBlocks.push_back(2);
|
||
|
_entryBlocks.push_back(3);
|
||
|
_entryBlocks.push_back(4);
|
||
|
_entryBlocks.push_back(5);
|
||
|
|
||
|
std::auto_ptr<Bitmap> bitmap(new Bitmap(_totalBlocks));
|
||
|
|
||
|
Buffer buffer(512);
|
||
|
|
||
|
// 2 bootcode blocks
|
||
|
for (unsigned i = 0; i < 2; ++i)
|
||
|
{
|
||
|
bitmap->allocBlock(i);
|
||
|
device->zeroBlock(i);
|
||
|
}
|
||
|
|
||
|
//4 volume header blocks.
|
||
|
for (unsigned i = 2; i < 6; ++i)
|
||
|
{
|
||
|
bitmap->allocBlock(i);
|
||
|
|
||
|
buffer.clear();
|
||
|
// prev block, next block
|
||
|
buffer.push16le(i == 2 ? 0 : i - 1);
|
||
|
buffer.push16le(i == 5 ? 0 : i + 1);
|
||
|
|
||
|
if (i == 2)
|
||
|
{
|
||
|
// create the volume header.
|
||
|
// all ivars must be set.
|
||
|
write(&buffer);
|
||
|
}
|
||
|
|
||
|
buffer.resize(512);
|
||
|
device->write(i, buffer.buffer());
|
||
|
}
|
||
|
|
||
|
// TODO -- create/write the volume entry....
|
||
|
|
||
|
// allocate blocks for the bitmap itself
|
||
|
unsigned bb = bitmap->bitmapBlocks();
|
||
|
for (unsigned i = 0; i < bb; ++i)
|
||
|
bitmap->allocBlock(i);
|
||
|
|
||
|
// now write the bitmap...
|
||
|
const uint8_t *bm = (const uint8_t *)bitmap->bitmap();
|
||
|
for (unsigned i = 0; i < bb; ++i)
|
||
|
{
|
||
|
device->write(_bitmapPointer + i, 512 * i + bm);
|
||
|
}
|
||
|
|
||
|
_device = device;
|
||
|
_bitmap = bitmap.release();
|
||
|
}
|
||
|
|
||
|
VolumeDirectory::~VolumeDirectory()
|
||
|
{
|
||
|
if (_device)
|
||
|
{
|
||
|
_device->sync();
|
||
|
delete _device;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void VolumeDirectory::write(Buffer *out)
|
||
|
{
|
||
|
out->push8((VolumeHeader << 4 ) | nameLength());
|
||
|
out->pushBytes(namei(), 15);
|
||
|
|
||
|
// reserved. SOS uses 0x75 for the first byte [?]
|
||
|
out->push8(0);
|
||
|
out->push8(0);
|
||
|
|
||
|
// last mod
|
||
|
out->push16le(_modification.date());
|
||
|
out->push16le(_modification.time());
|
||
|
|
||
|
// filename case bits
|
||
|
out->push16le(caseFlag());
|
||
|
|
||
|
// creation
|
||
|
out->push16le(creation().date());
|
||
|
out->push16le(creation().time());
|
||
|
|
||
|
out->push8(version());
|
||
|
out->push8(minVersion());
|
||
|
|
||
|
|
||
|
out->push8(access());
|
||
|
out->push8(entryLength());
|
||
|
out->push8(entriesPerBlock());
|
||
|
out->push16le(fileCount());
|
||
|
|
||
|
out->push16le(_bitmapPointer);
|
||
|
out->push16le(_totalBlocks);
|
||
|
}
|