cdrtools/libhfs_iso/block.c

226 lines
5.1 KiB
C
Raw Permalink Normal View History

2025-06-15 04:19:58 +08:00
/* @(#)block.c 1.8 09/07/13 joerg */
#include <schily/mconfig.h>
#ifndef lint
static UConst char sccsid[] =
"@(#)block.c 1.8 09/07/13 joerg";
#endif
/*
* hfsutils - tools for reading and writing Macintosh HFS volumes
* Copyright (C) 1996, 1997 Robert Leslie
*
* 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., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include <schily/string.h>
#include <schily/unistd.h>
#include <schily/errno.h>
#include "internal.h"
#include "data.h"
#include "block.h"
#include "low.h"
#ifdef DEBUG
#include <schily/stdio.h>
#endif /* DEBUG */
/*
* NAME: block->readlb()
* DESCRIPTION: read a logical block from a volume
*/
int b_readlb(vol, num, bp)
hfsvol *vol;
unsigned long num;
block *bp;
{
#ifndef APPLE_HYB
int bytes;
#endif
#ifdef APPLE_HYB
block *b;
hce_mem *hce;
#ifdef DEBUG
fprintf(stderr,"b_readlb: start block = %d\n", vol->vstart + num);
#endif /* DEBUG */
hce = vol->hce;
/* Check to see if requested block is in the HFS header or catalog/exents
files. If it is, read info from memory copy. If not, then something
has gone horribly wrong ... */
if (num < hce->hfs_hdr_size)
b = (block *)hce->hfs_hdr + num;
else if (num < hce->hfs_hdr_size + hce->hfs_ce_size)
b = (block *)hce->hfs_ce + num - hce->hfs_hdr_size;
else
{
ERROR(EIO, "should not happen!");
return -1;
}
memcpy(bp, b, HFS_BLOCKSZ);
#else
if (lseek(vol->fd, (vol->vstart + num) * HFS_BLOCKSZ, SEEK_SET) < 0)
{
ERROR(errno, "error seeking device");
return -1;
}
bytes = read(vol->fd, bp, HFS_BLOCKSZ);
if (bytes < 0)
{
ERROR(errno, "error reading from device");
return -1;
}
else if (bytes == 0)
{
ERROR(EIO, "read EOF on volume");
return -1;
}
else if (bytes != HFS_BLOCKSZ)
{
ERROR(EIO, "read incomplete block");
return -1;
}
#endif /* APPLE_HYB */
return 0;
}
/*
* NAME: block->writelb()
* DESCRIPTION: write a logical block to a volume
*/
int b_writelb(vol, num, bp)
hfsvol *vol;
unsigned long num;
block *bp;
{
#ifndef APPLE_HYB
int bytes;
#endif
#ifdef APPLE_HYB
block *b;
hce_mem *hce;
#ifdef DEBUG
fprintf(stderr,"b_writelb: start block = %d\n", vol->vstart + num);
#endif /* DEBUG */
hce = vol->hce;
/* Check to see if requested block is in the HFS header or catalog/exents
files. If it is, write info to memory copy. If not, then it's a block
for an ordinary file - and as we are writing the files later, then just
ignore and return OK */
if (num < hce->hfs_hdr_size)
b = (block *)hce->hfs_hdr + num;
else if (num < hce->hfs_hdr_size + hce->hfs_ce_size)
b = (block *)hce->hfs_ce + num - hce->hfs_hdr_size;
else
{
#ifdef DEBUG
fprintf(stderr,"b_writelb: ignoring\n");
#endif /* DEBUG */
return 0;
}
memcpy(b, bp, HFS_BLOCKSZ);
#else
if (lseek(vol->fd, (vol->vstart + num) * HFS_BLOCKSZ, SEEK_SET) < 0)
{
ERROR(errno, "error seeking device");
return -1;
}
bytes = write(vol->fd, bp, HFS_BLOCKSZ);
if (bytes < 0)
{
ERROR(errno, "error writing to device");
return -1;
}
else if (bytes != HFS_BLOCKSZ)
{
ERROR(EIO, "wrote incomplete block");
return -1;
}
#endif /* APPLE_HYB */
return 0;
}
/*
* NAME: block->readab()
* DESCRIPTION: read a block from an allocation block from a volume
*/
int b_readab(vol, anum, idx, bp)
hfsvol *vol;
unsigned int anum;
unsigned int idx;
block *bp;
{
/* verify the allocation block exists and is marked as in-use */
if (anum >= vol->mdb.drNmAlBlks)
{
ERROR(EIO, "read nonexistent block");
return -1;
}
else if (vol->vbm && ! BMTST(vol->vbm, anum))
{
ERROR(EIO, "read unallocated block");
return -1;
}
return b_readlb(vol, vol->mdb.drAlBlSt + anum * vol->lpa + idx, bp);
}
/*
* NAME: b->writeab()
* DESCRIPTION: write a block to an allocation block to a volume
*/
int b_writeab(vol, anum, idx, bp)
hfsvol *vol;
unsigned int anum;
unsigned int idx;
block *bp;
{
/* verify the allocation block exists and is marked as in-use */
if (anum >= vol->mdb.drNmAlBlks)
{
ERROR(EIO, "write nonexistent block");
return -1;
}
else if (vol->vbm && ! BMTST(vol->vbm, anum))
{
ERROR(EIO, "write unallocated block");
return -1;
}
vol->mdb.drAtrb &= ~HFS_ATRB_UMOUNTED;
vol->mdb.drLsMod = d_tomtime(time(0));
++vol->mdb.drWrCnt;
vol->flags |= HFS_UPDATE_MDB;
return b_writelb(vol, vol->mdb.drAlBlSt + anum * vol->lpa + idx, bp);
}