mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-12-26 04:25:27 +08:00
e2efa07966
Implement the ->free_disk method to free the msb_data structure only once the last gendisk reference goes away instead of keeping a local refcount. Signed-off-by: Christoph Hellwig <hch@lst.de> Link: https://lore.kernel.org/r/20220215094514.3828912-3-hch@lst.de Signed-off-by: Jens Axboe <axboe@kernel.dk>
287 lines
6.9 KiB
C
287 lines
6.9 KiB
C
/* SPDX-License-Identifier: GPL-2.0-only */
|
|
/*
|
|
* ms_block.h - Sony MemoryStick (legacy) storage support
|
|
|
|
* Copyright (C) 2013 Maxim Levitsky <maximlevitsky@gmail.com>
|
|
*
|
|
* Minor portions of the driver are copied from mspro_block.c which is
|
|
* Copyright (C) 2007 Alex Dubov <oakad@yahoo.com>
|
|
*
|
|
* Also ms structures were copied from old broken driver by same author
|
|
* These probably come from MS spec
|
|
*/
|
|
|
|
#ifndef MS_BLOCK_NEW_H
|
|
#define MS_BLOCK_NEW_H
|
|
|
|
#define MS_BLOCK_MAX_SEGS 32
|
|
#define MS_BLOCK_MAX_PAGES ((2 << 16) - 1)
|
|
|
|
#define MS_BLOCK_MAX_BOOT_ADDR 0x000c
|
|
#define MS_BLOCK_BOOT_ID 0x0001
|
|
#define MS_BLOCK_INVALID 0xffff
|
|
#define MS_MAX_ZONES 16
|
|
#define MS_BLOCKS_IN_ZONE 512
|
|
|
|
#define MS_BLOCK_MAP_LINE_SZ 16
|
|
#define MS_BLOCK_PART_SHIFT 3
|
|
|
|
|
|
#define MEMSTICK_UNCORR_ERROR (MEMSTICK_STATUS1_UCFG | \
|
|
MEMSTICK_STATUS1_UCEX | MEMSTICK_STATUS1_UCDT)
|
|
|
|
#define MEMSTICK_CORR_ERROR (MEMSTICK_STATUS1_FGER | MEMSTICK_STATUS1_EXER | \
|
|
MEMSTICK_STATUS1_DTER)
|
|
|
|
#define MEMSTICK_INT_ERROR (MEMSTICK_INT_CMDNAK | MEMSTICK_INT_ERR)
|
|
|
|
#define MEMSTICK_OVERWRITE_FLAG_NORMAL \
|
|
(MEMSTICK_OVERWRITE_PGST1 | \
|
|
MEMSTICK_OVERWRITE_PGST0 | \
|
|
MEMSTICK_OVERWRITE_BKST)
|
|
|
|
#define MEMSTICK_OV_PG_NORMAL \
|
|
(MEMSTICK_OVERWRITE_PGST1 | MEMSTICK_OVERWRITE_PGST0)
|
|
|
|
#define MEMSTICK_MANAGEMENT_FLAG_NORMAL \
|
|
(MEMSTICK_MANAGEMENT_SYSFLG | \
|
|
MEMSTICK_MANAGEMENT_SCMS1 | \
|
|
MEMSTICK_MANAGEMENT_SCMS0) \
|
|
|
|
struct ms_boot_header {
|
|
unsigned short block_id;
|
|
unsigned short format_reserved;
|
|
unsigned char reserved0[184];
|
|
unsigned char data_entry;
|
|
unsigned char reserved1[179];
|
|
} __packed;
|
|
|
|
|
|
struct ms_system_item {
|
|
unsigned int start_addr;
|
|
unsigned int data_size;
|
|
unsigned char data_type_id;
|
|
unsigned char reserved[3];
|
|
} __packed;
|
|
|
|
struct ms_system_entry {
|
|
struct ms_system_item disabled_block;
|
|
struct ms_system_item cis_idi;
|
|
unsigned char reserved[24];
|
|
} __packed;
|
|
|
|
struct ms_boot_attr_info {
|
|
unsigned char memorystick_class;
|
|
unsigned char format_unique_value1;
|
|
unsigned short block_size;
|
|
unsigned short number_of_blocks;
|
|
unsigned short number_of_effective_blocks;
|
|
unsigned short page_size;
|
|
unsigned char extra_data_size;
|
|
unsigned char format_unique_value2;
|
|
unsigned char assembly_time[8];
|
|
unsigned char format_unique_value3;
|
|
unsigned char serial_number[3];
|
|
unsigned char assembly_manufacturer_code;
|
|
unsigned char assembly_model_code[3];
|
|
unsigned short memory_manufacturer_code;
|
|
unsigned short memory_device_code;
|
|
unsigned short implemented_capacity;
|
|
unsigned char format_unique_value4[2];
|
|
unsigned char vcc;
|
|
unsigned char vpp;
|
|
unsigned short controller_number;
|
|
unsigned short controller_function;
|
|
unsigned char reserved0[9];
|
|
unsigned char transfer_supporting;
|
|
unsigned short format_unique_value5;
|
|
unsigned char format_type;
|
|
unsigned char memorystick_application;
|
|
unsigned char device_type;
|
|
unsigned char reserved1[22];
|
|
unsigned char format_uniqure_value6[2];
|
|
unsigned char reserved2[15];
|
|
} __packed;
|
|
|
|
struct ms_cis_idi {
|
|
unsigned short general_config;
|
|
unsigned short logical_cylinders;
|
|
unsigned short reserved0;
|
|
unsigned short logical_heads;
|
|
unsigned short track_size;
|
|
unsigned short page_size;
|
|
unsigned short pages_per_track;
|
|
unsigned short msw;
|
|
unsigned short lsw;
|
|
unsigned short reserved1;
|
|
unsigned char serial_number[20];
|
|
unsigned short buffer_type;
|
|
unsigned short buffer_size_increments;
|
|
unsigned short long_command_ecc;
|
|
unsigned char firmware_version[28];
|
|
unsigned char model_name[18];
|
|
unsigned short reserved2[5];
|
|
unsigned short pio_mode_number;
|
|
unsigned short dma_mode_number;
|
|
unsigned short field_validity;
|
|
unsigned short current_logical_cylinders;
|
|
unsigned short current_logical_heads;
|
|
unsigned short current_pages_per_track;
|
|
unsigned int current_page_capacity;
|
|
unsigned short mutiple_page_setting;
|
|
unsigned int addressable_pages;
|
|
unsigned short single_word_dma;
|
|
unsigned short multi_word_dma;
|
|
unsigned char reserved3[128];
|
|
} __packed;
|
|
|
|
|
|
struct ms_boot_page {
|
|
struct ms_boot_header header;
|
|
struct ms_system_entry entry;
|
|
struct ms_boot_attr_info attr;
|
|
} __packed;
|
|
|
|
struct msb_data {
|
|
struct memstick_dev *card;
|
|
struct gendisk *disk;
|
|
struct request_queue *queue;
|
|
spinlock_t q_lock;
|
|
struct blk_mq_tag_set tag_set;
|
|
struct hd_geometry geometry;
|
|
struct attribute_group attr_group;
|
|
struct request *req;
|
|
int caps;
|
|
int disk_id;
|
|
|
|
/* IO */
|
|
struct workqueue_struct *io_queue;
|
|
bool io_queue_stopped;
|
|
struct work_struct io_work;
|
|
bool card_dead;
|
|
|
|
/* Media properties */
|
|
struct ms_boot_page *boot_page;
|
|
u16 boot_block_locations[2];
|
|
int boot_block_count;
|
|
|
|
bool read_only;
|
|
unsigned short page_size;
|
|
int block_size;
|
|
int pages_in_block;
|
|
int zone_count;
|
|
int block_count;
|
|
int logical_block_count;
|
|
|
|
/* FTL tables */
|
|
unsigned long *used_blocks_bitmap;
|
|
unsigned long *erased_blocks_bitmap;
|
|
u16 *lba_to_pba_table;
|
|
int free_block_count[MS_MAX_ZONES];
|
|
bool ftl_initialized;
|
|
|
|
/* Cache */
|
|
unsigned char *cache;
|
|
unsigned long valid_cache_bitmap;
|
|
int cache_block_lba;
|
|
bool need_flush_cache;
|
|
struct timer_list cache_flush_timer;
|
|
|
|
/* Preallocated buffers */
|
|
unsigned char *block_buffer;
|
|
struct scatterlist prealloc_sg[MS_BLOCK_MAX_SEGS+1];
|
|
|
|
|
|
/* handler's local data */
|
|
struct ms_register_addr reg_addr;
|
|
bool addr_valid;
|
|
|
|
u8 command_value;
|
|
bool command_need_oob;
|
|
struct scatterlist *current_sg;
|
|
int current_sg_offset;
|
|
|
|
struct ms_register regs;
|
|
int current_page;
|
|
|
|
int state;
|
|
int exit_error;
|
|
bool int_polling;
|
|
unsigned long int_timeout;
|
|
|
|
};
|
|
|
|
enum msb_readpage_states {
|
|
MSB_RP_SEND_BLOCK_ADDRESS = 0,
|
|
MSB_RP_SEND_READ_COMMAND,
|
|
|
|
MSB_RP_SEND_INT_REQ,
|
|
MSB_RP_RECEIVE_INT_REQ_RESULT,
|
|
|
|
MSB_RP_SEND_READ_STATUS_REG,
|
|
MSB_RP_RECEIVE_STATUS_REG,
|
|
|
|
MSB_RP_SEND_OOB_READ,
|
|
MSB_RP_RECEIVE_OOB_READ,
|
|
|
|
MSB_RP_SEND_READ_DATA,
|
|
MSB_RP_RECEIVE_READ_DATA,
|
|
};
|
|
|
|
enum msb_write_block_states {
|
|
MSB_WB_SEND_WRITE_PARAMS = 0,
|
|
MSB_WB_SEND_WRITE_OOB,
|
|
MSB_WB_SEND_WRITE_COMMAND,
|
|
|
|
MSB_WB_SEND_INT_REQ,
|
|
MSB_WB_RECEIVE_INT_REQ,
|
|
|
|
MSB_WB_SEND_WRITE_DATA,
|
|
MSB_WB_RECEIVE_WRITE_CONFIRMATION,
|
|
};
|
|
|
|
enum msb_send_command_states {
|
|
MSB_SC_SEND_WRITE_PARAMS,
|
|
MSB_SC_SEND_WRITE_OOB,
|
|
MSB_SC_SEND_COMMAND,
|
|
|
|
MSB_SC_SEND_INT_REQ,
|
|
MSB_SC_RECEIVE_INT_REQ,
|
|
|
|
};
|
|
|
|
enum msb_reset_states {
|
|
MSB_RS_SEND,
|
|
MSB_RS_CONFIRM,
|
|
};
|
|
|
|
enum msb_par_switch_states {
|
|
MSB_PS_SEND_SWITCH_COMMAND,
|
|
MSB_PS_SWICH_HOST,
|
|
MSB_PS_CONFIRM,
|
|
};
|
|
|
|
struct chs_entry {
|
|
unsigned long size;
|
|
unsigned char sec;
|
|
unsigned short cyl;
|
|
unsigned char head;
|
|
};
|
|
|
|
static int msb_reset(struct msb_data *msb, bool full);
|
|
|
|
static int h_msb_default_bad(struct memstick_dev *card,
|
|
struct memstick_request **mrq);
|
|
|
|
#define __dbg(level, format, ...) \
|
|
do { \
|
|
if (debug >= level) \
|
|
pr_err(format "\n", ## __VA_ARGS__); \
|
|
} while (0)
|
|
|
|
|
|
#define dbg(format, ...) __dbg(1, format, ## __VA_ARGS__)
|
|
#define dbg_verbose(format, ...) __dbg(2, format, ## __VA_ARGS__)
|
|
|
|
#endif
|