* Added CHECKSUM_DEVICE_IOCTL_GET_CHECK_SUM ioctl.

* Added some debug output.
* CheckSumCache::_GetBlock(): Fixed incorrect check. The method could return
  an arbitrary block.


git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@37638 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Ingo Weinhold 2010-07-20 23:16:46 +00:00
parent 7514b57ebf
commit 8d591ecc41
2 changed files with 54 additions and 7 deletions

View File

@ -32,6 +32,14 @@
#include "CheckSum.h" #include "CheckSum.h"
//#define TRACE_CHECK_SUM_DEVICE
#ifdef TRACE_CHECK_SUM_DEVICE
# define TRACE(x...) dprintf(x)
#else
# define TRACE(x) do {} while (false)
#endif
// parameters for the DMA resource // parameters for the DMA resource
static const uint32 kDMAResourceBufferCount = 16; static const uint32 kDMAResourceBufferCount = 16;
static const uint32 kDMAResourceBounceBufferCount = 16; static const uint32 kDMAResourceBounceBufferCount = 16;
@ -138,6 +146,14 @@ struct CheckSumCache {
block->checkSums[blockIndex % kCheckSumsPerBlock] = checkSum; block->checkSums[blockIndex % kCheckSumsPerBlock] = checkSum;
block->dirty = true; block->dirty = true;
#ifdef TRACE_CHECK_SUM_DEVICE
TRACE("checksum_device: setting check sum of block %" B_PRIu64 " to: ",
blockIndex);
for (size_t i = 0; i < kCheckSumLength; i++)
TRACE("%02x", checkSum.Data()[i]);
TRACE("\n");
#endif
return B_OK; return B_OK;
} }
@ -150,7 +166,7 @@ private:
// check whether we have already cached the block // check whether we have already cached the block
for (BlockList::Iterator it = fBlocks.GetIterator(); for (BlockList::Iterator it = fBlocks.GetIterator();
CheckSumBlock* block = it.Next();) { CheckSumBlock* block = it.Next();) {
if (block->used && block->blockIndex) { if (block->used && blockIndex == block->blockIndex) {
// we know it -- requeue and return // we know it -- requeue and return
it.Remove(); it.Remove();
fBlocks.Add(block); fBlocks.Add(block);
@ -434,6 +450,11 @@ struct RawDevice : Device, DoublyLinkedListLinkImpl<RawDevice> {
kRawDeviceModuleName); kRawDeviceModuleName);
} }
status_t GetBlockCheckSum(uint64 blockIndex, CheckSum& checkSum)
{
return fCheckSumCache->GetCheckSum(blockIndex, checkSum);
}
status_t SetBlockCheckSum(uint64 blockIndex, const CheckSum& checkSum) status_t SetBlockCheckSum(uint64 blockIndex, const CheckSum& checkSum)
{ {
return fCheckSumCache->SetCheckSum(blockIndex, checkSum); return fCheckSumCache->SetCheckSum(blockIndex, checkSum);
@ -552,7 +573,9 @@ private:
fSHA256.Update(fTransferBuffer, B_PAGE_SIZE); fSHA256.Update(fTransferBuffer, B_PAGE_SIZE);
if (expectedCheckSum != fSHA256.Digest()) if (expectedCheckSum != fSHA256.Digest())
panic("Check sum mismatch for block %" B_PRIu64, blockIndex); panic("Check sum mismatch for block %" B_PRIu64 " (exptected at %p"
", actual at %p)", blockIndex, &expectedCheckSum,
fSHA256.Digest());
} }
private: private:
@ -1085,10 +1108,33 @@ checksum_raw_device_control(void* _cookie, uint32 op, void* buffer,
case B_FLUSH_DRIVE_CACHE: case B_FLUSH_DRIVE_CACHE:
return B_OK; return B_OK;
case CHECKSUM_DEVICE_IOCTL_GET_CHECK_SUM:
{
if (IS_USER_ADDRESS(buffer)) {
checksum_device_ioctl_check_sum getCheckSum;
if (user_memcpy(&getCheckSum, buffer, sizeof(getCheckSum))
!= B_OK) {
return B_BAD_ADDRESS;
}
status_t error = device->GetBlockCheckSum(
getCheckSum.blockIndex, getCheckSum.checkSum);
if (error != B_OK)
return error;
return user_memcpy(buffer, &getCheckSum, sizeof(getCheckSum));
}
checksum_device_ioctl_check_sum* getCheckSum
= (checksum_device_ioctl_check_sum*)buffer;
return device->GetBlockCheckSum(getCheckSum->blockIndex,
getCheckSum->checkSum);
}
case CHECKSUM_DEVICE_IOCTL_SET_CHECK_SUM: case CHECKSUM_DEVICE_IOCTL_SET_CHECK_SUM:
{ {
if (IS_USER_ADDRESS(buffer)) { if (IS_USER_ADDRESS(buffer)) {
checksum_device_ioctl_set_check_sum setCheckSum; checksum_device_ioctl_check_sum setCheckSum;
if (user_memcpy(&setCheckSum, buffer, sizeof(setCheckSum)) if (user_memcpy(&setCheckSum, buffer, sizeof(setCheckSum))
!= B_OK) { != B_OK) {
return B_BAD_ADDRESS; return B_BAD_ADDRESS;
@ -1098,8 +1144,8 @@ checksum_raw_device_control(void* _cookie, uint32 op, void* buffer,
setCheckSum.checkSum); setCheckSum.checkSum);
} }
checksum_device_ioctl_set_check_sum* setCheckSum checksum_device_ioctl_check_sum* setCheckSum
= (checksum_device_ioctl_set_check_sum*)buffer; = (checksum_device_ioctl_check_sum*)buffer;
return device->SetBlockCheckSum(setCheckSum->blockIndex, return device->SetBlockCheckSum(setCheckSum->blockIndex,
setCheckSum->checkSum); setCheckSum->checkSum);
} }

View File

@ -12,11 +12,12 @@
enum { enum {
CHECKSUM_DEVICE_IOCTL_SET_CHECK_SUM = B_DEVICE_OP_CODES_END + 0x666 CHECKSUM_DEVICE_IOCTL_GET_CHECK_SUM = B_DEVICE_OP_CODES_END + 0x666,
CHECKSUM_DEVICE_IOCTL_SET_CHECK_SUM
}; };
struct checksum_device_ioctl_set_check_sum { struct checksum_device_ioctl_check_sum {
uint64 blockIndex; uint64 blockIndex;
CheckSum checkSum; CheckSum checkSum;
}; };