/* JPC: An x86 PC Hardware Emulator for a pure Java Virtual Machine Release Version 2.4 A project from the Physics Dept, The University of Oxford Copyright (C) 2007-2010 The University of Oxford This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License version 2 as published by the Free Software Foundation. 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. Details (including contact information) can be found at: jpc.sourceforge.net or the developer website sourceforge.net/projects/jpc/ Conceived and Developed by: Rhys Newman, Ian Preston, Chris Dennis End of licence header */ package org.jpc.emulator.pci.peripheral; import org.jpc.emulator.Hibernatable; import org.jpc.emulator.pci.IOPortIORegion; import org.jpc.emulator.memory.Memory; import java.io.*; /** * * @author Chris Dennis */ class BMDMAIORegion implements IOPortIORegion, Hibernatable { public static final int BM_STATUS_DMAING = 0x01; private static final int BM_STATUS_ERROR = 0x02; private static final int BM_STATUS_INT = 0x04; private static final int BM_CMD_START = 0x01; private static final int BM_CMD_READ = 0x08; private int baseAddress; private long size; private BMDMAIORegion next; private byte command; private byte status; private int address, dtpr; /* current transfer state */ private IDEChannel.IDEState ideDevice; private int ideDMAFunction; private Memory physicalMemory; public BMDMAIORegion(BMDMAIORegion next) { this.baseAddress = -1; this.next = next; } public void saveState(DataOutput output) throws IOException { output.writeInt(baseAddress); output.writeLong(size); output.writeByte(command); output.writeByte(status); output.writeInt(address); output.writeInt(ideDMAFunction); if (ideDevice != null) ideDevice.saveState(output); } public void loadState(DataInput input) throws IOException { baseAddress = input.readInt(); size = input.readLong(); command = input.readByte(); status = input.readByte(); address = input.readInt(); ideDMAFunction = input.readInt(); //ideDevice.loadState(input); } void setAddressSpace(Memory memory) { physicalMemory = memory; } void writeMemory(int address, byte[] buffer, int offset, int length) { physicalMemory.copyArrayIntoContents(address, buffer, offset, length); } void setIDEDevice(IDEChannel.IDEState device) { this.ideDevice = device; } void setDMAFunction(int function) { ideDMAFunction = function; } void setIRQ() { status |= BM_STATUS_INT; } public int getAddress() { return baseAddress; } public long getSize() { return 0x10; } public int getType() { return PCI_ADDRESS_SPACE_IO; } byte getStatus() { return status; } public int getRegionNumber() { return 4; } public int getDtpr() { return dtpr; } public int getCommand() { return command; } public void setAddress(int address) { this.baseAddress = address; if (next != null) next.setAddress(address + 8); } public void ioPortWrite8(int address, int data) { if ((address - this.baseAddress) > 7) next.ioPortWrite8(address, data); switch(address - this.baseAddress) { case 0: this.writeCommand(data); return; case 2: this.writeStatus(data); return; default: } } public void ioPortWrite16(int address, int data) { this.ioPortWrite8(address, 0xff & data); this.ioPortWrite8(address + 1, 0xff & (data >>> 8)); } public void ioPortWrite32(int address, int data) { if ((address - this.baseAddress) > 7) next.ioPortWrite32(address, data); switch(address - this.baseAddress) { case 4: case 5: case 6: case 7: this.writeAddress(data); return; default: this.ioPortWrite16(address, 0xffff & data); this.ioPortWrite16(address + 2, data >>> 16); } } public int ioPortRead8(int address) { if ((address - this.baseAddress) > 7) return next.ioPortRead8(address); switch(address - this.baseAddress) { case 0: return this.command; case 2: return this.status; default: return 0xff; } } public int ioPortRead16(int address) { return (ioPortRead8(address) & 0xff) | (0xff00 & (ioPortRead8(address + 1) << 8)); } public int ioPortRead32(int address) { if ((address - this.baseAddress) > 7) return next.ioPortRead32(address); switch (address - this.baseAddress) { case 4: case 5: case 6: case 7: return this.address; default: return (ioPortRead16(address) & 0xffff) | (0xffff0000 & (ioPortRead16(address + 1) << 8)); } } public int[] ioPortsRequested() { return new int[]{baseAddress, baseAddress + 2, baseAddress + 4, baseAddress + 5, baseAddress + 6, baseAddress + 7, baseAddress + 8, baseAddress + 10, baseAddress + 12, baseAddress + 13, baseAddress + 14, baseAddress + 15}; } private void writeCommand(int data) { if ((data & BM_CMD_START) == 0) { /* XXX: do it better */ this.status &= ~BM_STATUS_DMAING; this.command = (byte)(data & 0x09); } else { this.status |= BM_STATUS_DMAING; this.command = (byte)(data & 0x09); /* start dma transfer if possible */ if (this.ideDMAFunction != IDEChannel.IDEState.IDF_NONE) this.ideDMALoop(); } } private void writeStatus(int data) { status = (byte)((data & 0x60) | (status & 1) | (status & ~data & 0x06)); } private void writeAddress(int data) { this.address = data & ~3; } void ideDMALoop() { int currentAddress = this.address; /* at most one page to avoid hanging if erroneous parameters */ for (int i = 0 ; i < 512; i++) { int prdAddress = physicalMemory.getDoubleWord(currentAddress); int prdSize = physicalMemory.getDoubleWord(currentAddress + 4); int length = prdSize & 0xfffe; if (length == 0) length = 0x10000; while (length > 0) { int lengthOne = this.ideDevice.dmaCallback(ideDMAFunction, prdAddress, length); if (lengthOne == 0) { /* end of transfer */ this.status &= ~BM_STATUS_DMAING; this.status |= BM_STATUS_INT; this.ideDMAFunction = IDEChannel.IDEState.IDF_NONE; this.ideDevice = null; return; } prdAddress += lengthOne; length -= lengthOne; } /* end of transfer */ if ((prdSize & 0x80000000) != 0) break; currentAddress += 8; } /* end of transfer */ this.status &= ~BM_STATUS_DMAING; this.status |= BM_STATUS_INT; this.ideDMAFunction = IDEChannel.IDEState.IDF_NONE; this.ideDevice = null; } }