- int retval, copydata=0;
- struct ata_ioc_request request;
- unsigned char buff[512];
-
- bzero(buff,512);
- bzero(&request,sizeof(struct ata_ioc_request));
- bzero(buff,512);
-
- request.u.ata.command=ATA_SMART_CMD;
- request.timeout=SCSI_TIMEOUT_DEFAULT;
- switch (command){
- case READ_VALUES:
- request.u.ata.feature=ATA_SMART_READ_VALUES;
- request.u.ata.lba=0xc24f<<8;
- request.flags=ATA_CMD_READ;
- request.data=(char *)buff;
- request.count=512;
- copydata=1;
- break;
- case READ_THRESHOLDS:
- request.u.ata.feature=ATA_SMART_READ_THRESHOLDS;
- request.u.ata.count=1;
- request.u.ata.lba=1|(0xc24f<<8);
- request.flags=ATA_CMD_READ;
- request.data=(char *)buff;
- request.count=512;
- copydata=1;
- break;
- case READ_LOG:
- request.u.ata.feature=ATA_SMART_READ_LOG_SECTOR;
- request.u.ata.lba=select|(0xc24f<<8);
- request.u.ata.count=1;
- request.flags=ATA_CMD_READ;
- request.data=(char *)buff;
- request.count=512;
- copydata=1;
- break;
- case IDENTIFY:
- request.u.ata.command=ATA_IDENTIFY_DEVICE;
- request.flags=ATA_CMD_READ;
- request.data=(char *)buff;
- request.count=512;
- copydata=1;
- break;
- case PIDENTIFY:
- request.u.ata.command=ATA_IDENTIFY_PACKET_DEVICE;
- request.flags=ATA_CMD_READ;
- request.data=(char *)buff;
- request.count=512;
- copydata=1;
- break;
- case ENABLE:
- request.u.ata.feature=ATA_SMART_ENABLE;
- request.u.ata.lba=0xc24f<<8;
- request.flags=ATA_CMD_CONTROL;
- break;
- case DISABLE:
- request.u.ata.feature=ATA_SMART_DISABLE;
- request.u.ata.lba=0xc24f<<8;
- request.flags=ATA_CMD_CONTROL;
- break;
- case AUTO_OFFLINE:
- // NOTE: According to ATAPI 4 and UP, this command is obsolete
- request.u.ata.feature=ATA_SMART_AUTO_OFFLINE;
- request.u.ata.lba=0xc24f<<8;
- request.u.ata.count=select;
- request.flags=ATA_CMD_CONTROL;
- break;
- case AUTOSAVE:
- request.u.ata.feature=ATA_SMART_AUTOSAVE;
- request.u.ata.lba=0xc24f<<8;
- request.u.ata.count=select;
- request.flags=ATA_CMD_CONTROL;
- break;
- case IMMEDIATE_OFFLINE:
- request.u.ata.feature=ATA_SMART_IMMEDIATE_OFFLINE;
- request.u.ata.lba = select|(0xc24f<<8); // put test in sector
- request.flags=ATA_CMD_CONTROL;
- break;
- case STATUS_CHECK: // same command, no HDIO in FreeBSD
- case STATUS:
- // this command only says if SMART is working. It could be
- // replaced with STATUS_CHECK below.
- request.u.ata.feature=ATA_SMART_STATUS;
- request.u.ata.lba=0xc24f<<8;
- request.flags=ATA_CMD_CONTROL;
- break;
- case CHECK_POWER_MODE:
- request.u.ata.command=ATA_CHECK_POWER_MODE;
- request.u.ata.feature=0;
- request.flags=ATA_CMD_CONTROL;
- break;
- case WRITE_LOG:
- memcpy(buff, data, 512);
- request.u.ata.feature=ATA_SMART_WRITE_LOG_SECTOR;
- request.u.ata.lba=select|(0xc24f<<8);
- request.u.ata.count=1;
- request.flags=ATA_CMD_WRITE;
- request.data=(char *)buff;
- request.count=512;
- break;
- default:
- pout("Unrecognized command %d in ata_command_interface()\n"
- "Please contact " PACKAGE_BUGREPORT "\n", command);
- errno=ENOSYS;
- return -1;
- }
-
- if (command==STATUS_CHECK){
- unsigned const char normal_lo=0x4f, normal_hi=0xc2;
- unsigned const char failed_lo=0xf4, failed_hi=0x2c;
- unsigned char low,high;
-
- if ((retval=do_cmd(&request)) || request.error)
- return -1;
-
-#if (FREEBSDVER < 502000)
- printwarning(NO_RETURN,NULL);
-#endif