rockbox/utils/imxtools/scsitools/stmp_scsi.h

389 lines
15 KiB
C
Raw Normal View History

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2012 Amaury Pouly
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
#ifndef __STMP_SCSI__
#define __STMP_SCSI__
#include <stdint.h>
#include <stdbool.h>
#include "rbscsi.h"
#if defined(_WIN32) || defined(__WIN32__)
/* Mingw has a curious behaviour: it packs only the last field, see
* https://gcc.gnu.org/bugzilla/show_bug.cgi?id=52991 */
#pragma pack(push)
#pragma pack(1)
#define RB_POP_PACK
#define RB_PACKED
#else
#define RB_PACKED __attribute__((packed))
#endif
/**
* Low-Level SCSI stuff
*/
#define SCSI_STMP_READ 0xc0
#define SCSI_STMP_WRITE 0xc1
/** STMP: Command */
#define SCSI_STMP_CMD_GET_PROTOCOL_VERSION 0x0
#define SCSI_STMP_CMD_GET_LOGICAL_MEDIA_INFO 0x2
#define SCSI_STMP_CMD_GET_LOGICAL_TABLE 0x5
#define SCSI_STMP_CMD_ALLOCATE_LOGICAL_MEDIA 0x6
#define SCSI_STMP_CMD_ERASE LOGICAL MEDIA 0x7
#define SCSI_STMP_CMD_GET_LOGICAL_DRIVE_INFO 0x12
#define SCSI_STMP_CMD_READ_LOGICAL_DRIVE_SECTOR 0x13
#define SCSI_STMP_CMD_SET_LOGICAL_DRIVE_INFO 0x20
#define SCSI_STMP_CMD_WRITE_LOGICAL_DRIVE_SECTOR 0x23
#define SCSI_STMP_CMD_ERASE_LOGICAL_DRIVE 0x2f
#define SCSI_STMP_CMD_GET_CHIP_MAJOR_REV_ID 0x30
#define SCSI_STMP_CMD_CHIP_RESET 0x31
#define SCSI_STMP_CMD_GET_CHIP_SERIAL_NUMBER 0x32
#define SCSI_STMP_CMD_GET_ROM_REV_ID 0x37
#define SCSI_STMP_CMD_GET_JANUS_STATUS 0x40
#define SCSI_STMP_CMD_INITIALIZE_STATUS 0x41
#define SCSI_STMP_CMD_RESET_TO_RECOVERY 0x42
#define SCSI_STMP_CMD_INITIALIZE_DATA_STORE 0x43
#define SCSI_STMP_CMD_RESET_TO_UPDATER 0x44
#define SCSI_STMP_CMD_GET_DEVICE_INFO 0x45
struct scsi_stmp_protocol_version_t
{
uint8_t major;
uint8_t minor;
} RB_PACKED;
struct scsi_stmp_rom_rev_id_t
{
uint16_t rev; /* big-endian */
} RB_PACKED;
struct scsi_stmp_chip_major_rev_id_t
{
uint16_t rev; /* big-endian */
} RB_PACKED;
struct scsi_stmp_logical_table_entry_t
{
uint8_t drive_no;
uint8_t type;
uint8_t tag;
uint64_t size; /* big-endian */
} RB_PACKED;
#define SCSI_STMP_DRIVE_TYPE_USER 0
#define SCSI_STMP_DRIVE_TYPE_SYSTEM 1
#define SCSI_STMP_DRIVE_TYPE_JANUS 2
#define SCSI_STMP_DRIVE_TAG_USER_STORAGE 0xa
#define SCSI_STMP_DRIVE_TAG_SYSTEM_BOOT 0x50
struct scsi_stmp_logical_table_header_t
{
uint16_t count; /* big-endian */
} RB_PACKED;
#define SCSI_STMP_MEDIA_INFO_NR_DRIVES 0 /** Number of drives (obsolete) */
#define SCSI_STMP_MEDIA_INFO_SIZE 1 /** Total size (bytes) */
#define SCSI_STMP_MEDIA_INFO_ALLOC_UNIT_SIZE 2 /** Allocation unit size (bytes) */
#define SCSI_STMP_MEDIA_INFO_IS_INITIALISED 3 /** Is initialised ? */
#define SCSI_STMP_MEDIA_INFO_STATE 4 /** Media state */
#define SCSI_STMP_MEDIA_INFO_IS_WRITE_PROTECTED 5 /** Is write protected ? */
#define SCSI_STMP_MEDIA_INFO_TYPE 6 /** Physical media type */
#define SCSI_STMP_MEDIA_INFO_SERIAL_NUMBER_SIZE 7 /** Serial number size (bytes) */
#define SCSI_STMP_MEDIA_INFO_SERIAL_NUMBER 8 /** Serial number */
#define SCSI_STMP_MEDIA_INFO_IS_SYSTEM_MEDIA 9 /** Is system media ? */
#define SCSI_STMP_MEDIA_INFO_IS_MEDIA_PRESENT 10 /** Is media present ? */
#define SCSI_STMP_MEDIA_INFO_PAGE_SIZE 11 /** Page size (bytes) */
#define SCSI_STMP_MEDIA_INFO_VENDOR 12 /** Vendor ID */
#define SCSI_STMP_MEDIA_INFO_NAND_ID 13 /** Full NAND ID */
#define SCSI_STMP_MEDIA_INFO_NR_DEVICES 14 /** Number of physical devices */
#define SCSI_STMP_MEDIA_STATE_UNKNOWN 0
#define SCSI_STMP_MEDIA_STATE_ERASED 1
#define SCSI_STMP_MEDIA_STATE_ALLOCATED 2
#define SCSI_STMP_MEDIA_TYPE_NAND 0
#define SCSI_STMP_MEDIA_TYPE_SDMMC 1
#define SCSI_STMP_MEDIA_TYPE_HDD 2
#define SCSI_STMP_MEDIA_TYPE_RAM 3
#define SCSI_STMP_MEDIA_TYPE_iNAND 4
#define SCSI_STMP_MEDIA_VENDOR_SAMSUNG 0xEC
#define SCSI_STMP_MEDIA_VENDOR_STMICRO 0x20
#define SCSI_STMP_MEDIA_VENDOR_HYNIX 0xAD
#define SCSI_STMP_MEDIA_VENDOR_MICRON 0x2C
#define SCSI_STMP_MEDIA_VENDOR_TOSHIBA 0x98
#define SCSI_STMP_MEDIA_VENDOR_RENESAS 0x07
#define SCSI_STMP_MEDIA_VENDOR_SANDISK 0x45
#define SCSI_STMP_MEDIA_VENDOR_INTEL 0x89
struct scsi_stmp_logical_media_info_type_t
{
uint8_t type;
} RB_PACKED;
struct scsi_stmp_logical_media_info_manufacturer_t
{
uint32_t type; /* big-endian */
} RB_PACKED;
#define SCSI_STMP_DRIVE_INFO_SECTOR_SIZE 0 /** Sector Size (bytes) */
#define SCSI_STMP_DRIVE_INFO_ERASE_SIZE 1 /** Erase Size (bytes) */
#define SCSI_STMP_DRIVE_INFO_SIZE 2 /** Total Size (bytes) */
#define SCSI_STMP_DRIVE_INFO_SIZE_MEGA 3 /** Total Size (mega-bytes) */
#define SCSI_STMP_DRIVE_INFO_SECTOR_COUNT 4 /** Sector Count */
#define SCSI_STMP_DRIVE_INFO_TYPE 5 /** Drive Type */
#define SCSI_STMP_DRIVE_INFO_TAG 6 /** Drive Tag */
#define SCSI_STMP_DRIVE_INFO_COMPONENT_VERSION 7 /** Component Version */
#define SCSI_STMP_DRIVE_INFO_PROJECT_VERSION 8 /** Project Version */
#define SCSI_STMP_DRIVE_INFO_IS_WRITE_PROTETED 9 /** Is Write Protected */
#define SCSI_STMP_DRIVE_INFO_SERIAL_NUMBER_SIZE 10 /** Serial Number Size */
#define SCSI_STMP_DRIVE_INFO_SERIAL_NUMBER 11 /** Serial Number */
#define SCSI_STMP_DRIVE_INFO_MEDIA_PRESENT 12 /** Is Media Present */
#define SCSI_STMP_DRIVE_INFO_MEDIA_CHANGE 13 /** Media Change */
#define SCSI_STMP_DRIVE_INFO_SECTOR_ALLOCATION 14 /** Sector Allocation */
#define SCSI_STMP_DRIVE_TYPE_DATA 0
#define SCSI_STMP_DRIVE_TYPE_SYSTEM 1
#define SCSI_STMP_DRIVE_TYPE_HIDDEN 2
#define SCSI_STMP_DRIVE_TYPE_UNKNOWN 3
#define SCSI_STMP_DRIVE_TAG_STMPSYS_S 0x00 /** Player drive */
#define SCSI_STMP_DRIVE_TAG_HOSTLINK_S 0x01 /** USB MSC/MTP drive */
#define SCSI_STMP_DRIVE_TAG_RESOURCE_BIN 0x02 /** Resource drive */
#define SCSI_STMP_DRIVE_TAG_EXTRA_S 0x03 /** Extra system drive */
#define SCSI_STMP_DRIVE_TAG_RESOURCE1_BIN 0x04 /** Extra resource drive */
#define SCSI_STMP_DRIVE_TAG_OTGHOST_S 0x05 /** OTG drive */
#define SCSI_STMP_DRIVE_TAG_HOSTRSC_BIN 0x06 /** USB MSC/MTP resource drive */
#define SCSI_STMP_DRIVE_TAG_DATA 0x0a /** Data drive */
#define SCSI_STMP_DRIVE_TAG_HIDDEN 0x0b /** Hidden data drive */
#define SCSI_STMP_DRIVE_TAG_BOOTMANAGER_S 0x50 /** Boot manager drive */
#define SCSI_STMP_DRIVE_TAG_UPDATER_S 0xff /** Recovery drive */
struct scsi_stmp_logical_drive_info_sector_t
{
uint32_t size; /* big-endian */
} RB_PACKED;
struct scsi_stmp_logical_drive_info_count_t
{
uint64_t count; /* big-endian */
} RB_PACKED;
struct scsi_stmp_logical_drive_info_size_t
{
uint64_t size; /* big-endian */
} RB_PACKED;
struct scsi_stmp_logical_drive_info_type_t
{
uint8_t type;
} RB_PACKED;
struct scsi_stmp_logical_drive_info_version_t
{
uint16_t major;
uint16_t minor;
uint16_t revision;
} RB_PACKED;
struct stmp_device_t;
typedef struct stmp_device_t *stmp_device_t;
typedef void (*stmp_printf_t)(void *user, const char *fmt, ...);
/* open flags */
#define STMP_DEBUG (1 << 0)
/* scsi flags */
#define STMP_READ (1 << 1)
#define STMP_WRITE (1 << 2)
uint16_t stmp_fix_endian16be(uint16_t w);
uint32_t stmp_fix_endian32be(uint32_t w);
uint64_t stmp_fix_endian64be(uint64_t w);
/* returns NULL on error */
stmp_device_t stmp_open(rb_scsi_device_t dev, unsigned flags, void *user, stmp_printf_t printf);
void stmp_close(stmp_device_t dev);
/* returns <0 on error and status otherwise */
int stmp_scsi(stmp_device_t dev, uint8_t *cdb, int cdb_size, unsigned flags,
void *sense, int *sense_size, void *buffer, int *buf_size);
/* returns != 0 on error */
int stmp_sense_analysis(stmp_device_t dev, int status, uint8_t *sense, int sense_size);
void stmp_printf(rb_scsi_device_t dev, const char *fmt, ...);
void stmp_debugf(rb_scsi_device_t dev, const char *fmt, ...);
/**
* Mid-level API
*/
/* returns !=0 on error */
int stmp_scsi_inquiry(stmp_device_t dev, uint8_t *dev_type, char vendor[9], char product[17]);
int stmp_scsi_get_protocol_version(stmp_device_t dev, void *buf, int *len);
int stmp_scsi_get_chip_major_rev_id(stmp_device_t dev, void *buf, int *len);
int stmp_scsi_get_rom_rev_id(stmp_device_t dev, void *buf, int *len);
int stmp_scsi_read_logical_drive_sectors(stmp_device_t dev, uint8_t drive, uint64_t address,
uint32_t count, void *buffer, int *buffer_size);
int stmp_scsi_write_logical_drive_sectors(stmp_device_t dev, uint8_t drive, uint64_t address,
uint32_t count, void *buffer, int *buffer_size);
/* the following functions *DO NOT* fix the endianness of the structures */
int stmp_scsi_get_logical_table(stmp_device_t dev, int entry_count, void *buf, int *len);
int stmp_scsi_get_serial_number(stmp_device_t dev, uint8_t info, void *data, int *len);
int stmp_scsi_get_logical_media_info(stmp_device_t dev, uint8_t info, void *data, int *len);
int stmp_scsi_get_logical_drive_info(stmp_device_t dev, uint8_t drive, uint8_t info, void *data, int *len);
int stmp_scsi_get_device_info(stmp_device_t dev, uint8_t info, void *data, int *len);
/* these helper functions fix the endianness for the previous calls, or returns != 0
* if they don't know about this info or the size doesn't match */
int stmp_fix_logical_media_info(uint8_t info, void *data, int len);
int stmp_fix_logical_drive_info(uint8_t info, void *data, int len);
int stmp_fix_device_info(uint8_t info, void *data, int len);
/**
* High-Level API
*/
struct stmp_logical_media_table_t
{
struct scsi_stmp_logical_table_header_t header;
struct scsi_stmp_logical_table_entry_t entry[];
}RB_PACKED table;
#ifdef RB_POP_PACK
#pragma pack(pop)
#endif
struct stmp_logical_media_info_t
{
struct
{
bool nr_drives;
bool size;
bool alloc_size;
bool initialised;
bool state;
bool write_protected;
bool type;
bool serial_len;
bool serial;
bool system;
bool present;
bool page_size;
bool vendor;
bool nand_id;
bool nr_devices;
}has;
uint16_t nr_drives;
uint64_t size;
uint32_t alloc_size;
uint8_t initialised;
uint8_t state;
uint8_t write_protected;
uint8_t type;
uint32_t serial_len;
uint8_t *serial; /* must be released with free() */
uint8_t system;
uint8_t present;
uint32_t page_size;
uint32_t vendor;
uint8_t nand_id[6];
uint32_t nr_devices;
};
#define SCSI_STMP_DRIVE_INFO_SECTOR_SIZE 0 /** Sector Size (bytes) */
#define SCSI_STMP_DRIVE_INFO_ERASE_SIZE 1 /** Erase Size (bytes) */
#define SCSI_STMP_DRIVE_INFO_SIZE 2 /** Total Size (bytes) */
#define SCSI_STMP_DRIVE_INFO_SIZE_MEGA 3 /** Total Size (mega-bytes) */
#define SCSI_STMP_DRIVE_INFO_SECTOR_COUNT 4 /** Sector Count */
#define SCSI_STMP_DRIVE_INFO_TYPE 5 /** Drive Type */
#define SCSI_STMP_DRIVE_INFO_TAG 6 /** Drive Tag */
#define SCSI_STMP_DRIVE_INFO_COMPONENT_VERSION 7 /** Component Version */
#define SCSI_STMP_DRIVE_INFO_PROJECT_VERSION 8 /** Project Version */
#define SCSI_STMP_DRIVE_INFO_IS_WRITE_PROTECTED 9 /** Is Write Protected */
#define SCSI_STMP_DRIVE_INFO_SERIAL_NUMBER_SIZE 10 /** Serial Number Size */
#define SCSI_STMP_DRIVE_INFO_SERIAL_NUMBER 11 /** Serial Number */
#define SCSI_STMP_DRIVE_INFO_MEDIA_PRESENT 12 /** Is Media Present */
#define SCSI_STMP_DRIVE_INFO_MEDIA_CHANGE 13 /** Media Change */
#define SCSI_STMP_DRIVE_INFO_SECTOR_ALLOCATION 14 /** Sector Allocation */
struct stmp_logical_drive_info_t
{
struct
{
bool sector_size;
bool erase_size;
bool size;
bool sector_count;
bool type;
bool tag;
bool component_version;
bool project_version;
bool write_protected;
bool serial_len;
bool serial;
bool present;
bool change;
bool sector_alloc;
}has;
uint32_t sector_size;
uint32_t erase_size;
uint64_t size;
uint64_t sector_count;
uint32_t type;
uint8_t tag;
struct scsi_stmp_logical_drive_info_version_t component_version;
struct scsi_stmp_logical_drive_info_version_t project_version;
uint8_t write_protected;
uint32_t serial_len;
uint8_t *serial;
uint8_t present;
uint8_t change;
uint32_t sector_alloc;
};
int stmp_get_protocol_version(stmp_device_t dev, struct scsi_stmp_protocol_version_t *ver);
int stmp_get_chip_major_rev_id(stmp_device_t dev, uint16_t *ver);
int stmp_get_rom_rev_id(stmp_device_t dev, uint16_t *ver);
/* return 0 on success, buffers must be released with free() */
int stmp_get_device_serial(stmp_device_t dev, uint8_t **buffer, int *len);
int stmp_get_logical_media_info(stmp_device_t dev, struct stmp_logical_media_info_t *info);
int stmp_get_logical_media_table(stmp_device_t dev, struct stmp_logical_media_table_t **table);
int stmp_get_logical_drive_info(stmp_device_t dev, uint8_t drive, struct stmp_logical_drive_info_t *info);
int stmp_read_logical_drive_sectors(stmp_device_t dev, uint8_t drive, uint64_t address,
uint32_t count, void *buffer, int buffer_size);
int stmp_write_logical_drive_sectors(stmp_device_t dev, uint8_t drive, uint64_t address,
uint32_t count, void *buffer, int buffer_size);
/* return <0 on error, or firmware size in bytes otherwise,
* if not NULL, the read/write function will be called as many times as needed to provide
* the entire firmware, it should return number of bytes read/written on success or -1 on error
* in all cases, the total size of the firmware is based on the header
* if NULL for read, return firmware size */
typedef int (*stmp_fw_rw_fn_t)(void *user, void *buf, size_t size);
int stmp_read_firmware(stmp_device_t dev, void *user, stmp_fw_rw_fn_t fn);
int stmp_write_firmware(stmp_device_t dev, void *user, stmp_fw_rw_fn_t fn);
/* string helpers */
const char *stmp_get_logical_media_type_string(uint32_t type);
const char *stmp_get_logical_media_vendor_string(uint32_t type);
const char *stmp_get_logical_drive_type_string(uint32_t type);
const char *stmp_get_logical_drive_tag_string(uint32_t type);
const char *stmp_get_logical_media_state_string(uint8_t state);
#endif /* __STMP_SCSI__ */