I'm know about big-endian and little-endian. All my PCs and virtual machines are little-endian, there are not any troubles with bit order. There are some troubles with my realisation, may by.
May be i'm need to align address to 4096 bytes when port rebase, my malloc returns me not-aligned address of memory, in EHCI i'm allocate to queue pool, td pool, framelist and say to malloc to allocate more then need by 8192 bytes and i'm do that: qhPool = ((uint)qhPool)/4096)*4096 + 4096; for td pool, qh pool, and framelist
There is my code of read and other:
Code: Select all
void port_rebase(HBA_PORT *port, int portno)
{
stop_cmd(port); // Stop command engine
// Command list offset: 1K*portno
// Command list entry size = 32
// Command list entry maxim count = 32
// Command list maxim size = 32*32 = 1K per port
port->clb = malloc(1024);
port->clbu = 0;
memset((void*)(port->clb), 0, 1024);
// FIS offset: 32K+256*portno
// FIS entry size = 256 unsigned chars per port
port->fb = malloc(256);
port->fbu = 0;
memset((void*)(port->fb), 0, 256);
// Command table offset: 40K + 8K*portno
// Command table size = 256*32 = 8K per port
HBA_CMD_HEADER *cmdheader = (HBA_CMD_HEADER*)(port->clb);
for (int i = 0; i<32; i++)
{
cmdheader[i].prdtl = 8; // 8 prdt entries per command table
// 256 unsigned chars per command table, 64+16+48+16*8
// Command table offset: 40K + 8K*portno + cmdheader_index*256
cmdheader[i].ctba = malloc(256);
cmdheader[i].ctbau = 0;
memset((void*)cmdheader[i].ctba, 0, 256);
}
start_cmd(port); // Start command engine
}
// Start command engine
void start_cmd(HBA_PORT *port)
{
// Wait until CR (bit15) is cleared
while (port->cmd & HBA_PxCMD_CR)
//kprintf("cmd: %d %d\n", port->cmd, port->cmd & HBA_PxCMD_CR);
;
port->cmd |= HBA_PxCMD_FRE;
port->cmd |= HBA_PxCMD_ST;
}
// Stop command engine
void stop_cmd(HBA_PORT *port)
{
// kprintf("%x\n", port->cmd);
// Clear ST (bit0)
port->cmd &= ~HBA_PxCMD_ST;
port->cmd &= ~HBA_PxCMD_FRE;
//port->cmd &= ~HBA_PxCMD_FRE;
// Wait until FR (bit14), CR (bit15) are cleared
while (1)
{
if (port->cmd & HBA_PxCMD_FR)
continue;
if (port->cmd & HBA_PxCMD_CR)
continue;
break;
}
// Clear FRE (bit4)
}
#define HBA_PxIS_TFES (1 << 30)
bool read(HBA_PORT *port, unsigned long long starth, uint32_t count, uint16_t *buf)
{
port->is = 0xffff; // Clear pending interrupt bits
int slot = find_cmdslot(port);
if (slot == -1)
return 0;
uint64_t addr = 0;
addr = (((addr | port->clbu) << 32) | port->clb);
HBA_CMD_HEADER *cmdheader = (HBA_CMD_HEADER*)(KERNBASE + addr);
cmdheader += slot;
cmdheader->cfl = sizeof(FIS_REG_H2D) / sizeof(unsigned int); // Command FIS size
cmdheader->w = 0; // Read from device
cmdheader->c = 1; // Read from device
cmdheader->p = 1; // Read from device
cmdheader->prdtl = (unsigned short)((count - 1) >> 4) + 1; // PRDT entries count
addr = 0;
addr = (((addr | cmdheader->ctbau) << 32) | cmdheader->ctba);
HBA_CMD_TBL *cmdtbl = (HBA_CMD_TBL*)(KERNBASE + addr);
int i = 0;
for (i = 0; i < cmdheader->prdtl - 1; i++)
{
cmdtbl->prdt_entry[i].dba = (unsigned int)(buf) & 0xFFFFFFFF;
cmdtbl->prdt_entry[i].dbau = 0;
cmdtbl->prdt_entry[i].dbc = 8 * 1024 - 1; // 8K unsigned chars
cmdtbl->prdt_entry[i].i = 0;
buf += 4 * 1024; // 4K unsigned shorts
count -= 16; // 16 sectors
}
cmdtbl->prdt_entry[i].dba = (unsigned int)(buf) & 0xFFFFFFFF;
cmdtbl->prdt_entry[i].dbau = 0;
cmdtbl->prdt_entry[i].dbc = count << 9; // 512 unsigned chars per sector
cmdtbl->prdt_entry[i].i = 0;
FIS_REG_H2D *cmdfis = (FIS_REG_H2D*)(&cmdtbl->cfis);
cmdfis->fis_type = FIS_TYPE_REG_H2D;
cmdfis->c = 1; // Command
cmdfis->command = 0x25;
cmdfis->lba0 = (unsigned char)starth;
cmdfis->lba1 = (unsigned char)(starth >> 8);
cmdfis->lba2 = (unsigned char)(starth >> 16);
cmdfis->device = 1 << 6; // LBA mode
cmdfis->lba3 = (unsigned char)(starth >> 24);
cmdfis->lba4 = (unsigned char)(starth >> 32);
cmdfis->lba5 = (unsigned char)(starth >> 40);
cmdfis->countl = count & 0xff;
cmdfis->counth = count >> 8;
//kprintf("[slot]{%d}", slot);
port->ci = 1 << slot; // Issue command
port->is = ~0; // Issue command
while (1)
{
kprintf("[%x,%x]", port->ci, port->is);
if ((port->ci & (1 << slot)) == 0)
break;
if ((port->is & (1<<26)) != 0)
{ // Task file error
kprintf("SATA Error\n");
return 0;
}
if (port->is & HBA_PxIS_TFES)
{ // Task file error
kprintf("Read disk error\n");
return 0;
}
}
if ((port->is & (1 << 26)) != 0)
{ // Task file error
kprintf("SATA Error\n");
return 0;
}
if (port->is & HBA_PxIS_TFES)
{
kprintf("Read disk error\n");
return 0;
}
int k = 0;
while (port->ci != 0)
{
k++;
if (k > 10)
return 0;
PitWait(1000);
kprintf("[%d]", port->ci);
}
return 1;
}
// Find a free command list slot
int find_cmdslot(HBA_PORT *m_port)
{
// If not set in SACT and CI, the slot is free
uint32_t slots = (m_port->sact | m_port->ci);
uint qq = (abar->cap & 0x0f00) >> 8;
for (int i = 0; i<qq; i++)
{
if ((slots & 1) == 0)
return i;
slots >>= 1;
}
kprintf("Cannot find free command list entry\n");
return -1;
}
void _probe_port(HBA_MEM *abar_temp)
{
// Search disk in impelemented ports
uint pi = abar_temp->pi;
int i = 0;
while (i < 32)
{
if (pi & 1)
{
int dt = check_type((HBA_PORT *)&abar_temp->ports[i]);
if (dt == AHCI_DEV_SATA)
{
kprintf("SATA drive found at port %d\n", i);
abar = abar_temp;
port_rebase(&abar_temp->ports[i], i);
kprintf("DONE AHCI INITIALISATION");
unsigned short * buf = malloc(16 * 512);
buf[0] = 1;
kprintf("qqqss");
int res = 0;
int jj = 0;
while (res == 0) {
res = read(&abar->ports[i], 16*jj, 1, buf);
if (res != 0)
for (int z = 0; z < 256; z++)
{
kprintf(" %x", buf[z]);
}
PitWait(10000);
jj++;
}
}
else if (dt == AHCI_DEV_SATAPI)
{
//kprintf("\nSATAPI drive found at port %d\n", i);
}
else if (dt == AHCI_DEV_SEMB)
{
//kprintf("\nSEMB drive found at port %d\n", i);
}
else if (dt == AHCI_DEV_PM)
{
//kprintf("\nPM drive found at port %d\n", i);
}
else
{
//kprintf("\nNo drive found at port %d\n", i);
}
}
pi >>= 1;
i++;
}
kprintf("probe_port complete\n");
}
With best regards,
Aleksandr