mirror of
https://github.com/RGBCube/serenity
synced 2025-05-30 10:15:07 +00:00
Kernel: Support reading/writing PATADiskDevices directly via /dev/hdX
This commit is contained in:
parent
9b5e0b6247
commit
a189285658
2 changed files with 94 additions and 4 deletions
|
@ -24,8 +24,11 @@
|
||||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
//#define PATA_DEVICE_DEBUG
|
||||||
|
|
||||||
#include <Kernel/Devices/PATAChannel.h>
|
#include <Kernel/Devices/PATAChannel.h>
|
||||||
#include <Kernel/Devices/PATADiskDevice.h>
|
#include <Kernel/Devices/PATADiskDevice.h>
|
||||||
|
#include <Kernel/FileSystem/FileDescription.h>
|
||||||
|
|
||||||
NonnullRefPtr<PATADiskDevice> PATADiskDevice::create(PATAChannel& channel, DriveType type, int major, int minor)
|
NonnullRefPtr<PATADiskDevice> PATADiskDevice::create(PATAChannel& channel, DriveType type, int major, int minor)
|
||||||
{
|
{
|
||||||
|
@ -73,6 +76,93 @@ void PATADiskDevice::set_drive_geometry(u16 cyls, u16 heads, u16 spt)
|
||||||
m_sectors_per_track = spt;
|
m_sectors_per_track = spt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ssize_t PATADiskDevice::read(FileDescription& fd, u8* outbuf, ssize_t len)
|
||||||
|
{
|
||||||
|
unsigned index = fd.offset() / block_size();
|
||||||
|
u16 whole_blocks = len / block_size();
|
||||||
|
ssize_t remaining = len % block_size();
|
||||||
|
|
||||||
|
unsigned blocks_per_page = PAGE_SIZE / block_size();
|
||||||
|
|
||||||
|
// PATAChannel will chuck a wobbly if we try to read more than PAGE_SIZE
|
||||||
|
// at a time, because it uses a single page for its DMA buffer.
|
||||||
|
if (whole_blocks >= blocks_per_page) {
|
||||||
|
whole_blocks = blocks_per_page;
|
||||||
|
remaining = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef PATA_DEVICE_DEBUG
|
||||||
|
kprintf("PATADiskDevice::read() index=%d whole_blocks=%d remaining=%d\n", index, whole_blocks, remaining);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (whole_blocks > 0) {
|
||||||
|
if (!read_blocks(index, whole_blocks, outbuf))
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
off_t pos = whole_blocks * block_size();
|
||||||
|
|
||||||
|
if (remaining > 0) {
|
||||||
|
auto buf = ByteBuffer::create_uninitialized(block_size());
|
||||||
|
if (!read_blocks(index + whole_blocks, 1, buf.data()))
|
||||||
|
return pos;
|
||||||
|
memcpy(&outbuf[pos], buf.data(), remaining);
|
||||||
|
}
|
||||||
|
|
||||||
|
return pos + remaining;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PATADiskDevice::can_read(const FileDescription& fd) const
|
||||||
|
{
|
||||||
|
return static_cast<unsigned>(fd.offset()) < (m_cylinders * m_heads * m_sectors_per_track * block_size());
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t PATADiskDevice::write(FileDescription& fd, const u8* inbuf, ssize_t len)
|
||||||
|
{
|
||||||
|
unsigned index = fd.offset() / block_size();
|
||||||
|
u16 whole_blocks = len / block_size();
|
||||||
|
ssize_t remaining = len % block_size();
|
||||||
|
|
||||||
|
unsigned blocks_per_page = PAGE_SIZE / block_size();
|
||||||
|
|
||||||
|
// PATAChannel will chuck a wobbly if we try to write more than PAGE_SIZE
|
||||||
|
// at a time, because it uses a single page for its DMA buffer.
|
||||||
|
if (whole_blocks >= blocks_per_page) {
|
||||||
|
whole_blocks = blocks_per_page;
|
||||||
|
remaining = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef PATA_DEVICE_DEBUG
|
||||||
|
kprintf("PATADiskDevice::write() index=%d whole_blocks=%d remaining=%d\n", index, whole_blocks, remaining);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (whole_blocks > 0) {
|
||||||
|
if (!write_blocks(index, whole_blocks, inbuf))
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
off_t pos = whole_blocks * block_size();
|
||||||
|
|
||||||
|
// since we can only write in block_size() increments, if we want to do a
|
||||||
|
// partial write, we have to read the block's content first, modify it,
|
||||||
|
// then write the whole block back to the disk.
|
||||||
|
if (remaining > 0) {
|
||||||
|
auto buf = ByteBuffer::create_zeroed(block_size());
|
||||||
|
if (!read_blocks(index + whole_blocks, 1, buf.data()))
|
||||||
|
return pos;
|
||||||
|
memcpy(buf.data(), &inbuf[pos], remaining);
|
||||||
|
if (!write_blocks(index + whole_blocks, 1, buf.data()))
|
||||||
|
return pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
return pos + remaining;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PATADiskDevice::can_write(const FileDescription& fd) const
|
||||||
|
{
|
||||||
|
return static_cast<unsigned>(fd.offset()) < (m_cylinders * m_heads * m_sectors_per_track * block_size());
|
||||||
|
}
|
||||||
|
|
||||||
bool PATADiskDevice::read_sectors_with_dma(u32 lba, u16 count, u8* outbuf)
|
bool PATADiskDevice::read_sectors_with_dma(u32 lba, u16 count, u8* outbuf)
|
||||||
{
|
{
|
||||||
return m_channel.ata_read_sectors_with_dma(lba, count, outbuf, is_slave());
|
return m_channel.ata_read_sectors_with_dma(lba, count, outbuf, is_slave());
|
||||||
|
|
|
@ -59,10 +59,10 @@ public:
|
||||||
void set_drive_geometry(u16, u16, u16);
|
void set_drive_geometry(u16, u16, u16);
|
||||||
|
|
||||||
// ^BlockDevice
|
// ^BlockDevice
|
||||||
virtual ssize_t read(FileDescription&, u8*, ssize_t) override { return 0; }
|
virtual ssize_t read(FileDescription&, u8*, ssize_t) override;
|
||||||
virtual bool can_read(const FileDescription&) const override { return true; }
|
virtual bool can_read(const FileDescription&) const override;
|
||||||
virtual ssize_t write(FileDescription&, const u8*, ssize_t) override { return 0; }
|
virtual ssize_t write(FileDescription&, const u8*, ssize_t) override;
|
||||||
virtual bool can_write(const FileDescription&) const override { return true; }
|
virtual bool can_write(const FileDescription&) const override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
explicit PATADiskDevice(PATAChannel&, DriveType, int, int);
|
explicit PATADiskDevice(PATAChannel&, DriveType, int, int);
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue