Page 1 of 1
Need help with the data returned by AHCI on IDENTIFY
Posted: Thu Feb 18, 2016 2:10 am
by ashishkumar4
Was reading this thing :
http://kurtqiao.github.io/uefi/2014/12/ ... -mode.html and made a 'driver' on its basis. it identifies sata drives and ahci controller on qemu. Then I send an Identify command to the drive and get the information from the device. But I don't know the structure of this 'info' I got from the device. The info or better dump is shown in the screenshot attached. Please help me know exactly what did the device returned and in what order (structure) like there is a structure for PATA Drives which we get while initializing them via IDENTIFY Command. I want a similar struct.
EDIT: After a few tests, I concluded that the first QWORD has something to do with the size of the SATA disk as I made several virtual disks and tried it with them. The qword I got were related to the size of the hard disks but there are a few more things in that qword along with the size. Please help me.
The code I made to do this is:
Code: Select all
int IDENTIFYdrive(HBA_PORT *port)
{
/***Make the Command Header***/
HBA_CMD_HEADER *cmdhead=(HBA_CMD_HEADER*)port->clb;//kmalloc(sizeof(HBA_CMD_HEADER));
//port->clb = (DWORD)cmdhead;
cmdhead->cfl = 5;
//cmdhead->a=0;
cmdhead->w = 0;
cmdhead->prdtl = 1;
cmdhead->p = 1;
cmdhead->c = 1;
/***Make the Command Table***/
HBA_CMD_TBL *cmdtbl = (HBA_CMD_TBL*)kmalloc(sizeof(HBA_CMD_TBL));
cmdhead->ctba = (DWORD)cmdtbl;
memset((void*)cmdtbl, 0, sizeof(HBA_CMD_TBL));
cmdtbl->prdt_entry[0].dba = kmalloc(4096);
cmdtbl->prdt_entry[0].dbc = 0x1FF;
cmdtbl->prdt_entry[0].i = 1; // interrupt when identify complete
data_base=cmdtbl->prdt_entry[0].dba;
memset((void*)data_base,0,4096);
/***Make the IDENTIFY DEVICE h2d FIS***/
FIS_REG_H2D *cmdfis = (FIS_REG_H2D*)cmdtbl->cfis;
//printf("cmdfis %x ",cmdfis);
memset((void*)cmdfis,0,sizeof(FIS_REG_H2D));
cmdfis->fis_type = FIS_TYPE_REG_H2D;
cmdfis->c = 1;
cmdfis->command = ATA_CMD_IDENTIFY;
/***Send the Command***/
port->ci=1;
/***Wait for a reply***/
while(1)
{
if(port->ci == 0)
{
break;
}
printf(" %x",port->ci);
}
uint32_t *baddr=data_base;
printf("\n\t The Data we Recieved from identify command:\n\t");
for(;*baddr;)
{
printf("%x ",*baddr);
++baddr;
}
return 1;
Please help me understand the thing device gave me, those numbers marked in red given to the iobuff after issuing the command.
Re: Need help with the data returned by AHCI on IDENTIFY
Posted: Thu Feb 18, 2016 3:09 pm
by Brendan
Hi,
ashishkumar4 wrote:Code: Select all
uint32_t *baddr=data_base;
printf("\n\t The Data we Recieved from identify command:\n\t");
for(;*baddr;)
{
printf("%x ",*baddr);
++baddr;
}
This doesn't make sense. The IDENTIFY command typically returns a block of 512 bytes, where any of the fields in the middle can be zeros and where most fields are 16-bit (and not 32-bit).
To find the format of the data return by the IDENTIFY command, you need to find the relevant documentation (e.g. google search for "serial ATA specification filetype:pdf"). Sadly, this specification only gives you part of the information and says "Set as indicated in ATA/ATAPI-5" for the rest, so you also need to find the ATA/ATAPI-5 specification.
Cheers,
Brendan
Re: Need help with the data returned by AHCI on IDENTIFY
Posted: Thu Feb 18, 2016 9:03 pm
by ashishkumar4
what? ._. that's sad. yeah I got to know various fields are 0 when I replaced that part with a byte sized array. but if there is no proper documentation, how can I make correctly that which particular information refers to what ._. Sir, how did you do it for your OS or someone you know did it? because I searched Linux kernel. its way to complex. then I searched some microkernels. that also didn't helped.
Re: Need help with the data returned by AHCI on IDENTIFY
Posted: Thu Feb 18, 2016 10:38 pm
by Brendan
Hi,
ashishkumar4 wrote:what? ._. that's sad. yeah I got to know various fields are 0 when I replaced that part with a byte sized array. but if there is no proper documentation, how can I make correctly that which particular information refers to what ._. Sir, how did you do it for your OS or someone you know did it? because I searched Linux kernel. its way to complex. then I searched some microkernels. that also didn't helped.
Brendan wrote:To find the format of the data return by the IDENTIFY command, you need to find the relevant documentation (e.g. google search for "serial ATA specification filetype:pdf"). Sadly, this specification only gives you part of the information and says "Set as indicated in ATA/ATAPI-5" for the rest, so you also need to find the ATA/ATAPI-5 specification.
Cheers,
Brendan
Re: Need help with the data returned by AHCI on IDENTIFY
Posted: Fri Feb 19, 2016 12:35 am
by ashishkumar4
:/ ok
Re: Need help with the data returned by AHCI on IDENTIFY
Posted: Fri Feb 19, 2016 8:25 am
by ashishkumar4
Thank you sir, atlast I did it too. After reading the ATA-CS and SATA specs, I have made the information structure. It looks like this:
(got some help from other sources as well) For those who are also seeking this:
Code: Select all
typedef struct SATA_ident
{
unsigned short config; /* lots of obsolete bit flags */
unsigned short cyls; /* obsolete */
unsigned short reserved2; /* special config */
unsigned short heads; /* "physical" heads */
unsigned short track_bytes; /* unformatted bytes per track */
unsigned short sector_bytes; /* unformatted bytes per sector */
unsigned short sectors; /* "physical" sectors per track */
unsigned short vendor0; /* vendor unique */
unsigned short vendor1; /* vendor unique */
unsigned short vendor2; /* vendor unique */
unsigned char serial_no[20]; /* 0 = not_specified */
unsigned short buf_type;
unsigned short buf_size; /* 512 byte increments; 0 = not_specified */
unsigned short ecc_bytes; /* for r/w long cmds; 0 = not_specified */
unsigned char fw_rev[8]; /* 0 = not_specified */
unsigned char model[40]; /* 0 = not_specified */
unsigned short multi_count; /* Multiple Count */
unsigned short dword_io; /* 0=not_implemented; 1=implemented */
unsigned short capability1; /* vendor unique */
unsigned short capability2; /* bits 0:DMA 1:LBA 2:IORDYsw 3:IORDYsup word: 50 */
unsigned char vendor5; /* vendor unique */
unsigned char tPIO; /* 0=slow, 1=medium, 2=fast */
unsigned char vendor6; /* vendor unique */
unsigned char tDMA; /* 0=slow, 1=medium, 2=fast */
unsigned short field_valid; /* bits 0:cur_ok 1:eide_ok */
unsigned short cur_cyls; /* logical cylinders */
unsigned short cur_heads; /* logical heads word 55*/
unsigned short cur_sectors; /* logical sectors per track */
unsigned short cur_capacity0; /* logical total sectors on drive */
unsigned short cur_capacity1; /* (2 words, misaligned int) */
unsigned char multsect; /* current multiple sector count */
unsigned char multsect_valid; /* when (bit0==1) multsect is ok */
unsigned int lba_capacity; /* total number of sectors */
unsigned short dma_1word; /* single-word dma info */
unsigned short dma_mword; /* multiple-word dma info */
unsigned short eide_pio_modes; /* bits 0:mode3 1:mode4 */
unsigned short eide_dma_min; /* min mword dma cycle time (ns) */
unsigned short eide_dma_time; /* recommended mword dma cycle time (ns) */
unsigned short eide_pio; /* min cycle time (ns), no IORDY */
unsigned short eide_pio_iordy; /* min cycle time (ns), with IORDY */
unsigned short words69_70[2]; /* reserved words 69-70 */
unsigned short words71_74[4]; /* reserved words 71-74 */
unsigned short queue_depth; /* */
unsigned short sata_capability; /* SATA Capabilities word 76*/
unsigned short sata_additional; /* Additional Capabilities */
unsigned short sata_supported; /* SATA Features supported */
unsigned short features_enabled; /* SATA features enabled */
unsigned short major_rev_num; /* Major rev number word 80 */
unsigned short minor_rev_num; /* */
unsigned short command_set_1; /* bits 0:Smart 1:Security 2:Removable 3:PM */
unsigned short command_set_2; /* bits 14:Smart Enabled 13:0 zero */
unsigned short cfsse; /* command set-feature supported extensions */
unsigned short cfs_enable_1; /* command set-feature enabled */
unsigned short cfs_enable_2; /* command set-feature enabled */
unsigned short csf_default; /* command set-feature default */
unsigned short dma_ultra; /* */
unsigned short word89; /* reserved (word 89) */
unsigned short word90; /* reserved (word 90) */
unsigned short CurAPMvalues; /* current APM values */
unsigned short word92; /* reserved (word 92) */
unsigned short comreset; /* should be cleared to 0 */
unsigned short accoustic; /* accoustic management */
unsigned short min_req_sz; /* Stream minimum required size */
unsigned short transfer_time_dma; /* Streaming Transfer Time-DMA */
unsigned short access_latency; /* Streaming access latency-DMA & PIO WORD 97*/
unsigned int perf_granularity; /* Streaming performance granularity */
unsigned int total_usr_sectors[2]; /* Total number of user addressable sectors */
unsigned short transfer_time_pio; /* Streaming Transfer time PIO */
unsigned short reserved105; /* Word 105 */
unsigned short sector_sz; /* Puysical Sector size / Logical sector size */
unsigned short inter_seek_delay; /* In microseconds */
unsigned short words108_116[9]; /* */
unsigned int words_per_sector; /* words per logical sectors */
unsigned short supported_settings; /* continued from words 82-84 */
unsigned short command_set_3; /* continued from words 85-87 */
unsigned short words121_126[6]; /* reserved words 121-126 */
unsigned short word127; /* reserved (word 127) */
unsigned short security_status; /* device lock function
* 15:9 reserved
* 8 security level 1:max 0:high
* 7:6 reserved
* 5 enhanced erase
* 4 expire
* 3 frozen
* 2 locked
* 1 en/disabled
* 0 capability
*/
unsigned short csfo; /* current set features options
* 15:4 reserved
* 3 auto reassign
* 2 reverting
* 1 read-look-ahead
* 0 write cache
*/
unsigned short words130_155[26];/* reserved vendor words 130-155 */
unsigned short word156;
unsigned short words157_159[3];/* reserved vendor words 157-159 */
unsigned short cfa; /* CFA Power mode 1 */
unsigned short words161_175[15]; /* Reserved */
unsigned char media_serial[60]; /* words 176-205 Current Media serial number */
unsigned short sct_cmd_transport; /* SCT Command Transport */
unsigned short words207_208[2]; /* reserved */
unsigned short block_align; /* Alignement of logical blocks in larger physical blocks */
unsigned int WRV_sec_count; /* Write-Read-Verify sector count mode 3 only */
unsigned int verf_sec_count; /* Verify Sector count mode 2 only */
unsigned short nv_cache_capability; /* NV Cache capabilities */
unsigned short nv_cache_sz; /* NV Cache size in logical blocks */
unsigned short nv_cache_sz2; /* NV Cache size in logical blocks */
unsigned short rotation_rate; /* Nominal media rotation rate */
unsigned short reserved218; /* */
unsigned short nv_cache_options; /* NV Cache options */
unsigned short words220_221[2]; /* reserved */
unsigned short transport_major_rev; /* */
unsigned short transport_minor_rev; /* */
unsigned short words224_233[10]; /* Reserved */
unsigned short min_dwnload_blocks; /* Minimum number of 512byte units per DOWNLOAD MICROCODE
command for mode 03h */
unsigned short max_dwnload_blocks; /* Maximum number of 512byte units per DOWNLOAD MICROCODE
command for mode 03h */
unsigned short words236_254[19]; /* Reserved */
unsigned short integrity; /* Cheksum, Signature */
} SATA_ident_t;