kernel_samsung_a34x-permissive/drivers/scsi/ufs/ufshpb_skh.c
2024-04-28 15:49:01 +02:00

3241 lines
88 KiB
C
Executable file

// SPDX-License-Identifier: GPL-2.0
/*
* Copyright (C) 2017-2018 Samsung Electronics Co., Ltd.
* Modified work Copyright (C) 2018, Google, Inc.
* Modified work Copyright (C) 2019 SK hynix
*/
#include <linux/slab.h>
#include <linux/blkdev.h>
#include <scsi/scsi.h>
#include <linux/sysfs.h>
#include <linux/blktrace_api.h>
#include "../../../block/blk.h"
#include "ufs.h"
#include "ufshcd.h"
#include "ufshpb_skh.h"
#include <asm/unaligned.h>
u32 skhpb_debug_mask = SKHPB_LOG_ERR | SKHPB_LOG_INFO;
//u32 skhpb_debug_mask = SKHPB_LOG_ERR | SKHPB_LOG_INFO | SKHPB_LOG_DEBUG | SKHPB_LOG_HEX;
int debug_map_req = SKHPB_MAP_RSP_DISABLE;
/*
* debug variables
*/
static int skhpb_alloc_mctx;
/*
* define global constants
*/
static int skhpb_sects_per_blk_shift;
static int skhpb_bits_per_dword_shift;
static int skhpb_bits_per_dword_mask;
static int skhpb_create_sysfs(struct ufs_hba *hba, struct skhpb_lu *hpb);
static int skhpb_check_lru_evict(struct skhpb_lu *hpb, struct skhpb_region *cb);
static void skhpb_error_handler(struct work_struct *work);
static void skhpb_evict_region(struct skhpb_lu *hpb,
struct skhpb_region *cb);
static void skhpb_purge_active_block(struct skhpb_lu *hpb);
static int skhpb_set_map_req(struct skhpb_lu *hpb,
int region, int subregion, struct skhpb_map_ctx *mctx,
struct skhpb_rsp_info *rsp_info,
enum SKHPB_BUFFER_MODE flag);
static int skhpb_rsp_map_cmd_req(struct skhpb_lu *hpb,
struct skhpb_rsp_info *rsp_info);
static void skhpb_map_loading_trigger(struct skhpb_lu *hpb,
bool only_pinned, bool do_work_handler);
static inline void skhpb_purge_active_page(struct skhpb_lu *hpb,
struct skhpb_subregion *cp, int state);
static void skhpb_hit_lru_info(struct skhpb_victim_select_info *lru_info,
struct skhpb_region *cb);
static inline void skhpb_get_bit_offset(
struct skhpb_lu *hpb, int subregion_offset,
int *dword, int *offset)
{
*dword = subregion_offset >> skhpb_bits_per_dword_shift;
*offset = subregion_offset & skhpb_bits_per_dword_mask;
}
/* called with hpb_lock (irq) */
static bool skhpb_ppn_dirty_check(struct skhpb_lu *hpb,
struct skhpb_subregion *cp, int subregion_offset)
{
bool is_dirty = false;
unsigned int bit_dword, bit_offset;
if (!cp->mctx->ppn_dirty)
return true;
skhpb_get_bit_offset(hpb, subregion_offset,
&bit_dword, &bit_offset);
is_dirty = cp->mctx->ppn_dirty[bit_dword] & (1 << bit_offset) ? true : false;
return is_dirty;
}
static void skhpb_ppn_prep(struct skhpb_lu *hpb,
struct ufshcd_lrb *lrbp, skhpb_t ppn,
unsigned int sector_len)
{
unsigned char *cmd = lrbp->cmd->cmnd;
if (hpb->hpb_ver < 0x0200) {
cmd[0] = READ_16;
cmd[1] = 0x0;
cmd[15] = 0x01; //NAND V5,V6 : Use Control field for HPB_READ
} else {
cmd[0] = READ_16;
cmd[1] = 0x5; // NAND V7 : Use reseved fields for HPB_READ
cmd[15] = 0x0;
}
put_unaligned(ppn, (u64 *)&cmd[6]);
cmd[14] = (u8)(sector_len >> skhpb_sects_per_blk_shift); //Transfer length
lrbp->cmd->cmd_len = MAX_CDB_SIZE;
//To verify the values within READ command
/* SKHPB_DRIVER_HEXDUMP("[HPB] HPB READ ", 16, 1, cmd, sizeof(cmd), 1); */
}
static inline void skhpb_set_dirty_bits(struct skhpb_lu *hpb,
struct skhpb_region *cb, struct skhpb_subregion *cp,
int dword, int offset, unsigned int count)
{
const unsigned long mask = ((1UL << count) - 1) & 0xffffffff;
if (cb->region_state == SKHPB_REGION_INACTIVE)
return;
BUG_ON(!cp->mctx);
cp->mctx->ppn_dirty[dword] |= (mask << offset);
}
/* called with hpb_lock (irq) */
static void skhpb_set_dirty(struct skhpb_lu *hpb,
struct ufshcd_lrb *lrbp, int region,
int subregion, int subregion_offset)
{
struct skhpb_region *cb;
struct skhpb_subregion *cp;
int count;
int bit_count, bit_dword, bit_offset;
count = blk_rq_sectors(lrbp->cmd->request) >> skhpb_sects_per_blk_shift;
skhpb_get_bit_offset(hpb, subregion_offset,
&bit_dword, &bit_offset);
do {
bit_count = min(count, SKHPB_BITS_PER_DWORD - bit_offset);
cb = hpb->region_tbl + region;
cp = cb->subregion_tbl + subregion;
skhpb_set_dirty_bits(hpb, cb, cp,
bit_dword, bit_offset, bit_count);
bit_offset = 0;
bit_dword++;
if (bit_dword == hpb->dwords_per_subregion) {
bit_dword = 0;
subregion++;
if (subregion == hpb->subregions_per_region) {
subregion = 0;
region++;
}
}
count -= bit_count;
} while (count);
}
#if 0
static inline bool skhpb_is_encrypted_lrbp(struct ufshcd_lrb *lrbp)
{
return (lrbp->utr_descriptor_ptr->header.dword_0 & UTRD_CRYPTO_ENABLE);
}
#endif
static inline enum SKHPB_CMD skhpb_get_cmd(struct ufshcd_lrb *lrbp)
{
unsigned char cmd = lrbp->cmd->cmnd[0];
if (cmd == READ_10 || cmd == READ_16)
return SKHPB_CMD_READ;
if (cmd == WRITE_10 || cmd == WRITE_16)
return SKHPB_CMD_WRITE;
if (cmd == UNMAP)
return SKHPB_CMD_DISCARD;
return SKHPB_CMD_OTHERS;
}
static inline void skhpb_get_pos_from_lpn(struct skhpb_lu *hpb,
unsigned int lpn, int *region, int *subregion, int *offset)
{
int region_offset;
*region = lpn >> hpb->entries_per_region_shift;
region_offset = lpn & hpb->entries_per_region_mask;
*subregion = region_offset >> hpb->entries_per_subregion_shift;
*offset = region_offset & hpb->entries_per_subregion_mask;
}
static inline bool skhpb_check_region_subregion_validity(struct skhpb_lu *hpb,
int region, int subregion)
{
struct skhpb_region *cb;
if (region >= hpb->regions_per_lu) {
SKHPB_DRIVER_E("[HCM] Out of REGION range - region[%d]:MAX[%d]\n",
region, hpb->regions_per_lu);
return false;
}
cb = hpb->region_tbl + region;
if (subregion >= cb->subregion_count) {
SKHPB_DRIVER_E("[HCM] Out of SUBREGION range - subregion[i]:MAX[%d]\n",
subregion, cb->subregion_count);
return false;
}
return true;
}
static inline skhpb_t skhpb_get_ppn(struct skhpb_map_ctx *mctx, int pos)
{
skhpb_t *ppn_table;
int index, offset;
index = pos / SKHPB_ENTREIS_PER_OS_PAGE;
offset = pos % SKHPB_ENTREIS_PER_OS_PAGE;
ppn_table = page_address(mctx->m_page[index]);
return ppn_table[offset];
}
#if defined(SKHPB_READ_LARGE_CHUNK_SUPPORT)
static bool skhpb_subregion_dirty_check(struct skhpb_lu *hpb, struct skhpb_subregion *cp,
int subregion_offset, int reqBlkCnt)
{
unsigned int bit_dword, bit_offset;
unsigned int tmp;
int checkCnt;
if (!cp->mctx)
return true;
if (!cp->mctx->ppn_dirty)
return true;
skhpb_get_bit_offset(hpb, subregion_offset,
&bit_dword, &bit_offset);
while (true) {
checkCnt = SKHPB_BITS_PER_DWORD - bit_offset;
if (cp->mctx->ppn_dirty[bit_dword]) {
tmp = cp->mctx->ppn_dirty[bit_dword] << bit_offset;
if (SKHPB_BITS_PER_DWORD - reqBlkCnt > 0)
tmp = tmp >> (SKHPB_BITS_PER_DWORD - reqBlkCnt);
if (tmp)
return true;
}
reqBlkCnt -= checkCnt;
if (reqBlkCnt <= 0)
break;
bit_dword++;
if (bit_dword >= hpb->ppn_dirties_per_subregion)
break;
bit_offset = 0;
}
return false;
}
static bool skhpb_lc_dirty_check(struct skhpb_lu *hpb, unsigned int lpn, unsigned int rq_sectors)
{
int reg;
int subReg;
struct skhpb_region *cb;
struct skhpb_subregion *cp;
unsigned long cur_lpn = lpn;
int subRegOffset;
int reqBlkCnt = rq_sectors >> skhpb_sects_per_blk_shift;
do {
skhpb_get_pos_from_lpn(hpb, cur_lpn, &reg, &subReg, &subRegOffset);
if (!skhpb_check_region_subregion_validity(hpb, reg, subReg))
return true;
cb = hpb->region_tbl + reg;
cp = cb->subregion_tbl + subReg;
if (cb->region_state == SKHPB_REGION_INACTIVE ||
cp->subregion_state != SKHPB_SUBREGION_CLEAN) {
atomic64_inc(&hpb->lc_reg_subreg_miss);
return true;
}
if (skhpb_subregion_dirty_check(hpb, cp, subRegOffset, reqBlkCnt)) {
//SKHPB_DRIVER_D("[NORMAL READ] DIRTY: Region(%d), SubRegion(%d) \n", reg, subReg);
atomic64_inc(&hpb->lc_entry_dirty_miss);
return true;
}
if (hpb->entries_per_subregion < subRegOffset + reqBlkCnt) {
reqBlkCnt -= (hpb->entries_per_subregion - subRegOffset);
} else {
reqBlkCnt = 0;
}
cur_lpn += reqBlkCnt;
} while (reqBlkCnt);
return false;
}
#endif
void skhpb_prep_fn(struct ufs_hba *hba, struct ufshcd_lrb *lrbp)
{
struct skhpb_lu *hpb;
struct skhpb_region *cb;
struct skhpb_subregion *cp;
unsigned int lpn;
skhpb_t ppn = 0;
int region, subregion, subregion_offset;
const struct request *rq;
unsigned long long rq_pos;
unsigned int rq_sectors;
unsigned char cmd;
unsigned long flags;
/* WKLU could not be HPB-LU */
if (lrbp->lun >= UFS_UPIU_MAX_GENERAL_LUN)
return;
hpb = hba->skhpb_lup[lrbp->lun];
if (!hpb || !hpb->lu_hpb_enable)
return;
if (hpb->force_hpb_read_disable)
return;
cmd = skhpb_get_cmd(lrbp);
if (cmd == SKHPB_CMD_OTHERS)
return;
if (blk_rq_is_scsi(lrbp->cmd->request))
return;
/*
* TODO: check if ICE is not supported or not.
*
* if (cmd==SKHPB_CMD_READ && skhpb_is_encrypted_lrbp(lrbp))
* return;
*/
rq = lrbp->cmd->request;
rq_pos = blk_rq_pos(rq);
rq_sectors = blk_rq_sectors(rq);
lpn = rq_pos / SKHPB_SECTORS_PER_BLOCK;
skhpb_get_pos_from_lpn(
hpb, lpn, &region, &subregion, &subregion_offset);
if (!skhpb_check_region_subregion_validity(hpb, region, subregion))
return;
cb = hpb->region_tbl + region;
cp = cb->subregion_tbl + subregion;
if (cmd == SKHPB_CMD_WRITE ||
cmd == SKHPB_CMD_DISCARD) {
if (cb->region_state == SKHPB_REGION_INACTIVE) {
return;
}
spin_lock_irqsave(&hpb->hpb_lock, flags);
skhpb_set_dirty(hpb, lrbp, region, subregion,
subregion_offset);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
return;
}
#if defined(SKHPB_READ_LARGE_CHUNK_SUPPORT)
if (((rq_sectors & (SKHPB_SECTORS_PER_BLOCK - 1)) != 0) ||
rq_sectors >
(SKHPB_READ_LARGE_CHUNK_MAX_BLOCK_COUNT << skhpb_sects_per_blk_shift)) {
#else
if (rq_sectors != SKHPB_SECTORS_PER_BLOCK) {
#endif
atomic64_inc(&hpb->size_miss);
return;
}
spin_lock_irqsave(&hpb->hpb_lock, flags);
if (cb->region_state == SKHPB_REGION_INACTIVE) {
atomic64_inc(&hpb->region_miss);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
return;
} else if (cp->subregion_state != SKHPB_SUBREGION_CLEAN) {
atomic64_inc(&hpb->subregion_miss);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
return;
}
if (rq_sectors <= SKHPB_SECTORS_PER_BLOCK) {
if (skhpb_ppn_dirty_check(hpb, cp, subregion_offset)) {
atomic64_inc(&hpb->entry_dirty_miss);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
return;
}
}
#if defined(SKHPB_READ_LARGE_CHUNK_SUPPORT)
else {
if (skhpb_lc_dirty_check(hpb, lpn, rq_sectors)) {
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
return;
}
atomic64_inc(&hpb->lc_hit);
}
#endif
ppn = skhpb_get_ppn(cp->mctx, subregion_offset);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
//SKHPB_DRIVER_D("XXX ufs ppn %016llx, lba %u\n", (unsigned long long)ppn, lpn);
skhpb_ppn_prep(hpb, lrbp, ppn, rq_sectors);
atomic64_inc(&hpb->hit);
return;
}
static int skhpb_clean_dirty_bitmap(
struct skhpb_lu *hpb, struct skhpb_subregion *cp)
{
struct skhpb_region *cb;
cb = hpb->region_tbl + cp->region;
/* if mctx is null, active block had been evicted out */
if (cb->region_state == SKHPB_REGION_INACTIVE || !cp->mctx) {
SKHPB_DRIVER_D("%d - %d error already evicted\n",
cp->region, cp->subregion);
return -EINVAL;
}
memset(cp->mctx->ppn_dirty, 0x00,
hpb->entries_per_subregion / BITS_PER_BYTE);
return 0;
}
static void skhpb_clean_active_subregion(
struct skhpb_lu *hpb, struct skhpb_subregion *cp)
{
struct skhpb_region *cb;
cb = hpb->region_tbl + cp->region;
/* if mctx is null, active block had been evicted out */
if (cb->region_state == SKHPB_REGION_INACTIVE || !cp->mctx) {
SKHPB_DRIVER_D("%d - %d clean already evicted\n",
cp->region, cp->subregion);
return;
}
cp->subregion_state = SKHPB_SUBREGION_CLEAN;
}
static void skhpb_error_active_subregion(
struct skhpb_lu *hpb, struct skhpb_subregion *cp)
{
struct skhpb_region *cb;
cb = hpb->region_tbl + cp->region;
/* if mctx is null, active block had been evicted out */
if (cb->region_state == SKHPB_REGION_INACTIVE || !cp->mctx) {
SKHPB_DRIVER_E("%d - %d evicted\n", cp->region, cp->subregion);
return;
}
cp->subregion_state = SKHPB_SUBREGION_DIRTY;
}
static void skhpb_map_compl_process(struct skhpb_lu *hpb,
struct skhpb_map_req *map_req)
{
unsigned long flags;
SKHPB_MAP_REQ_TIME(map_req, map_req->RSP_end, 1);
SKHPB_DRIVER_D("SKHPB RB COMPL BUFFER %d - %d\n", map_req->region, map_req->subregion);
spin_lock_irqsave(&hpb->hpb_lock, flags);
skhpb_clean_active_subregion(hpb,
hpb->region_tbl[map_req->region].subregion_tbl +
map_req->subregion);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
}
static void skhpb_map_inactive_compl_process(struct skhpb_lu *hpb,
struct skhpb_map_req *map_req)
{
SKHPB_DRIVER_D("SKHPB WB COMPL BUFFER %d\n", map_req->region);
SKHPB_MAP_REQ_TIME(map_req, map_req->RSP_end, 1);
}
/*
* Must held rsp_list_lock before enter this function
*/
static struct skhpb_rsp_info *skhpb_get_req_info(struct skhpb_lu *hpb)
{
struct skhpb_rsp_info *rsp_info =
list_first_entry_or_null(&hpb->lh_rsp_info_free,
struct skhpb_rsp_info,
list_rsp_info);
if (!rsp_info) {
SKHPB_DRIVER_E("there is no rsp_info");
return NULL;
}
list_del(&rsp_info->list_rsp_info);
memset(rsp_info, 0x00, sizeof(struct skhpb_rsp_info));
INIT_LIST_HEAD(&rsp_info->list_rsp_info);
return rsp_info;
}
static void skhpb_map_req_compl_fn(struct request *req, blk_status_t error)
{
struct skhpb_map_req *map_req = req->end_io_data;
struct ufs_hba *hba;
struct skhpb_lu *hpb;
struct scsi_sense_hdr sshdr;
struct skhpb_region *cb;
struct scsi_request *scsireq = scsi_req(req);
unsigned long flags;
/* shut up "bio leak" warning */
memcpy(map_req->sense, scsireq->sense, SCSI_SENSE_BUFFERSIZE);
req->bio = NULL;
__blk_put_request(req->q, req);
hpb = map_req->hpb;
hba = hpb->hba;
cb = hpb->region_tbl + map_req->region;
if (hba->skhpb_state != SKHPB_PRESENT)
goto free_map_req;
if (!error) {
skhpb_map_compl_process(hpb, map_req);
goto free_map_req;
}
SKHPB_DRIVER_E("error number %d ( %d - %d )\n",
error, map_req->region, map_req->subregion);
scsi_normalize_sense(map_req->sense,
SCSI_SENSE_BUFFERSIZE, &sshdr);
SKHPB_DRIVER_E("code %x sense_key %x asc %x ascq %x\n",
sshdr.response_code,
sshdr.sense_key, sshdr.asc, sshdr.ascq);
SKHPB_DRIVER_E("byte4 %x byte5 %x byte6 %x additional_len %x\n",
sshdr.byte4, sshdr.byte5,
sshdr.byte6, sshdr.additional_length);
atomic64_inc(&hpb->rb_fail);
if (sshdr.sense_key == ILLEGAL_REQUEST) {
if (sshdr.asc == 0x00 && sshdr.ascq == 0x16) {
/* OPERATION IN PROGRESS */
SKHPB_DRIVER_E("retry rb %d - %d",
map_req->region, map_req->subregion);
spin_lock_irqsave(&hpb->map_list_lock, flags);
INIT_LIST_HEAD(&map_req->list_map_req);
list_add_tail(&map_req->list_map_req, &hpb->lh_map_req_retry);
spin_unlock_irqrestore(&hpb->map_list_lock, flags);
schedule_delayed_work(&hpb->skhpb_map_req_retry_work,
msecs_to_jiffies(5000));
return;
}
}
// Only change subregion status at here.
// Do not put map_ctx, it will re-use when it is activated again.
spin_lock_irqsave(&hpb->hpb_lock, flags);
skhpb_error_active_subregion(hpb, cb->subregion_tbl + map_req->subregion);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
free_map_req:
spin_lock_irqsave(&hpb->map_list_lock, flags);
INIT_LIST_HEAD(&map_req->list_map_req);
list_add_tail(&map_req->list_map_req, &hpb->lh_map_req_free);
spin_unlock_irqrestore(&hpb->map_list_lock, flags);
atomic64_dec(&hpb->alloc_map_req_cnt);
}
static void skhpb_map_inactive_req_compl_fn(
struct request *req, blk_status_t error)
{
struct skhpb_map_req *map_req = req->end_io_data;
struct ufs_hba *hba;
struct skhpb_lu *hpb;
struct scsi_sense_hdr sshdr;
struct skhpb_region *cb;
struct scsi_request *scsireq = scsi_req(req);
unsigned long flags;
/* shut up "bio leak" warning */
memcpy(map_req->sense, scsireq->sense, SCSI_SENSE_BUFFERSIZE);
req->bio = NULL;
__blk_put_request(req->q, req);
hpb = map_req->hpb;
hba = hpb->hba;
cb = hpb->region_tbl + map_req->region;
if (hba->skhpb_state != SKHPB_PRESENT)
goto free_map_req;
if (!error) {
skhpb_map_inactive_compl_process(hpb, map_req);
goto free_map_req;
}
SKHPB_DRIVER_E("error number %d ( %d - %d )",
error, map_req->region, map_req->subregion);
scsi_normalize_sense(map_req->sense, SCSI_SENSE_BUFFERSIZE,
&sshdr);
SKHPB_DRIVER_E("code %x sense_key %x asc %x ascq %x",
sshdr.response_code,
sshdr.sense_key, sshdr.asc, sshdr.ascq);
SKHPB_DRIVER_E("byte4 %x byte5 %x byte6 %x additional_len %x",
sshdr.byte4, sshdr.byte5,
sshdr.byte6, sshdr.additional_length);
atomic64_inc(&hpb->rb_fail);
if (sshdr.sense_key == ILLEGAL_REQUEST) {
if (cb->is_pinned) {
SKHPB_DRIVER_E("WRITE_BUFFER is not allowed on pinned area: region#%d",
cb->region);
} else {
if (sshdr.asc == 0x00 && sshdr.ascq == 0x16) {
/* OPERATION IN PROGRESS */
SKHPB_DRIVER_E("retry wb %d", map_req->region);
spin_lock(&hpb->map_list_lock);
INIT_LIST_HEAD(&map_req->list_map_req);
list_add_tail(&map_req->list_map_req, &hpb->lh_map_req_retry);
spin_unlock(&hpb->map_list_lock);
schedule_delayed_work(&hpb->skhpb_map_req_retry_work,
msecs_to_jiffies(5000));
return;
}
}
}
// Only change subregion status at here.
// Do not put map_ctx, it will re-use when it is activated again.
spin_lock_irqsave(&hpb->hpb_lock, flags);
skhpb_error_active_subregion(hpb, cb->subregion_tbl + map_req->subregion);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
free_map_req:
spin_lock(&hpb->map_list_lock);
INIT_LIST_HEAD(&map_req->list_map_req);
list_add_tail(&map_req->list_map_req, &hpb->lh_map_req_free);
spin_unlock(&hpb->map_list_lock);
atomic64_dec(&hpb->alloc_map_req_cnt);
}
static int skhpb_execute_req_dev_ctx(struct skhpb_lu *hpb,
unsigned char *cmd, void *buf, int length)
{
unsigned long flags;
struct scsi_sense_hdr sshdr;
struct scsi_device *sdp;
struct ufs_hba *hba = hpb->hba;
int ret = 0;
spin_lock_irqsave(hba->host->host_lock, flags);
sdp = hba->sdev_ufs_lu[hpb->lun];
if (sdp) {
ret = scsi_device_get(sdp);
if (!ret && !scsi_device_online(sdp)) {
ret = -ENODEV;
scsi_device_put(sdp);
} else if (!ret) {
hba->issue_ioctl = true;
}
} else {
ret = -ENODEV;
}
spin_unlock_irqrestore(hba->host->host_lock, flags);
if (ret)
return ret;
ret = scsi_execute_req(sdp, cmd, DMA_FROM_DEVICE,
buf, length, &sshdr,
msecs_to_jiffies(30000), 3, NULL);
spin_lock_irqsave(hba->host->host_lock, flags);
hba->issue_ioctl = false;
spin_unlock_irqrestore(hba->host->host_lock, flags);
scsi_device_put(sdp);
return ret;
}
static inline void skhpb_set_read_dev_ctx_cmd(unsigned char *cmd, int lba,
int length)
{
cmd[0] = READ_10;
cmd[1] = 0x02;
cmd[2] = SKHPB_GET_BYTE_3(lba);
cmd[3] = SKHPB_GET_BYTE_2(lba);
cmd[4] = SKHPB_GET_BYTE_1(lba);
cmd[5] = SKHPB_GET_BYTE_0(lba);
cmd[6] = SKHPB_GET_BYTE_2(length);
cmd[7] = SKHPB_GET_BYTE_1(length);
cmd[8] = SKHPB_GET_BYTE_0(length);
}
int skhpb_issue_req_dev_ctx(struct skhpb_lu *hpb, unsigned char *buf,
int buf_length)
{
unsigned char cmd[10] = { 0 };
int cmd_len = buf_length >> PAGE_SHIFT;
int ret = 0;
skhpb_set_read_dev_ctx_cmd(cmd, 0x48504230, cmd_len);
ret = skhpb_execute_req_dev_ctx(hpb, cmd, buf, buf_length);
if (ret < 0)
SKHPB_DRIVER_E("failed with err %d\n", ret);
return ret;
}
static inline void skhpb_set_write_buf_cmd(unsigned char *cmd, int region)
{
cmd[0] = SKHPB_WRITE_BUFFER;
cmd[1] = 0x01;
cmd[2] = SKHPB_GET_BYTE_1(region);
cmd[3] = SKHPB_GET_BYTE_0(region);
cmd[4] = 0x00;
cmd[5] = 0x00;
cmd[6] = 0x00;
cmd[7] = 0x00;
cmd[8] = 0x00;
cmd[9] = 0x00;
//To verify the values within WRITE_BUFFER command
SKHPB_DRIVER_HEXDUMP("[HPB] WRITE BUFFER ", 16, 1, cmd, 10, 1);
}
static inline void skhpb_set_read_buf_cmd(unsigned char *cmd,
int region, int subregion, int subregion_mem_size)
{
cmd[0] = SKHPB_READ_BUFFER;
cmd[1] = 0x01;
cmd[2] = SKHPB_GET_BYTE_1(region);
cmd[3] = SKHPB_GET_BYTE_0(region);
cmd[4] = SKHPB_GET_BYTE_1(subregion);
cmd[5] = SKHPB_GET_BYTE_0(subregion);
cmd[6] = SKHPB_GET_BYTE_2(subregion_mem_size);
cmd[7] = SKHPB_GET_BYTE_1(subregion_mem_size);
cmd[8] = SKHPB_GET_BYTE_0(subregion_mem_size);
cmd[9] = 0x00;
//To verify the values within READ_BUFFER command
SKHPB_DRIVER_HEXDUMP("[HPB] READ BUFFER ", 16, 1, cmd, 10, 1);
}
static int skhpb_add_bio_page(struct skhpb_lu *hpb,
struct request_queue *q, struct bio *bio, struct bio_vec *bvec,
struct skhpb_map_ctx *mctx)
{
struct page *page = NULL;
int i, ret;
bio_init(bio, bvec, hpb->mpages_per_subregion);
for (i = 0; i < hpb->mpages_per_subregion; i++) {
page = mctx->m_page[i];
if (!page)
return -ENOMEM;
ret = bio_add_pc_page(q, bio, page, hpb->mpage_bytes, 0);
if (ret != hpb->mpage_bytes) {
SKHPB_DRIVER_E("error ret %d\n", ret);
return -EINVAL;
}
}
return 0;
}
static int skhpb_map_req_issue(
struct skhpb_lu *hpb, struct skhpb_map_req *map_req)
{
struct request_queue *q = hpb->hba->sdev_ufs_lu[hpb->lun]->request_queue;
struct request *req;
struct scsi_request *scsireq;
unsigned char cmd[10] = { 0 };
int ret;
unsigned long flags;
if (map_req->rwbuffer_flag == W_BUFFER)
skhpb_set_write_buf_cmd(cmd, map_req->region);
else
skhpb_set_read_buf_cmd(cmd, map_req->region, map_req->subregion,
map_req->subregion_mem_size);
if (map_req->rwbuffer_flag == W_BUFFER)
req = blk_get_request(q, REQ_OP_SCSI_OUT, BLK_MQ_REQ_PREEMPT);
else
req = blk_get_request(q, REQ_OP_SCSI_IN, BLK_MQ_REQ_PREEMPT);
if (IS_ERR(req)) {
int rv = PTR_ERR(req);
if (map_req->rwbuffer_flag == W_BUFFER)
SKHPB_DRIVER_E("blk_get_request errno %d, \
retry #%d, WRITE BUFFER %d\n",
rv, map_req->retry_cnt, map_req->region);
else
SKHPB_DRIVER_E("blk_get_request errno %d, \
retry #%d, READ BUFFER %d:%d\n",
rv, map_req->retry_cnt,
map_req->region, map_req->subregion);
if (map_req->retry_cnt == 10) {
/* give up */
return rv;
}
spin_lock_irqsave(&hpb->map_list_lock, flags);
list_add_tail(&map_req->list_map_req, &hpb->lh_map_req_retry);
spin_unlock_irqrestore(&hpb->map_list_lock, flags);
schedule_delayed_work(&hpb->skhpb_map_req_retry_work, msecs_to_jiffies(10));
return 0;
}
scsireq = scsi_req(req);
scsireq->cmd_len = COMMAND_SIZE(cmd[0]);
BUG_ON(scsireq->cmd_len > sizeof(scsireq->__cmd));
scsireq->cmd = scsireq->__cmd;
memcpy(scsireq->cmd, cmd, scsireq->cmd_len);
req->rq_flags |= RQF_QUIET | RQF_PREEMPT;
req->timeout = msecs_to_jiffies(30000);
req->end_io_data = map_req;
if (map_req->rwbuffer_flag == R_BUFFER) {
ret = skhpb_add_bio_page(
hpb, q, &map_req->bio, map_req->bvec, map_req->mctx);
if (ret) {
SKHPB_DRIVER_E("skhpb_add_bio_page_error %d\n", ret);
goto out_put_request;
}
map_req->pbio = &map_req->bio;
blk_rq_append_bio(req, &map_req->pbio);
}
SKHPB_DRIVER_D("issue map_request: %d - %d\n",
map_req->region, map_req->subregion);
SKHPB_MAP_REQ_TIME(map_req, map_req->RSP_issue, 0);
if (hpb->hpb_control_mode == HOST_CTRL_MODE) {
if (map_req->rwbuffer_flag == W_BUFFER)
blk_execute_rq_nowait(
q, NULL, req, 0, skhpb_map_inactive_req_compl_fn);
else
blk_execute_rq_nowait(q, NULL, req, 0, skhpb_map_req_compl_fn);
} else
blk_execute_rq_nowait(q, NULL, req, 1, skhpb_map_req_compl_fn);
if (map_req->rwbuffer_flag == W_BUFFER)
atomic64_inc(&hpb->w_map_req_cnt);
else
atomic64_inc(&hpb->map_req_cnt);
return 0;
out_put_request:
blk_put_request(req);
return ret;
}
static int skhpb_set_map_req(struct skhpb_lu *hpb,
int region, int subregion, struct skhpb_map_ctx *mctx,
struct skhpb_rsp_info *rsp_info,
enum SKHPB_BUFFER_MODE flag)
{
bool last = hpb->region_tbl[region].subregion_tbl[subregion].last;
struct skhpb_map_req *map_req;
unsigned long flags;
spin_lock_irqsave(&hpb->map_list_lock, flags);
map_req = list_first_entry_or_null(&hpb->lh_map_req_free,
struct skhpb_map_req,
list_map_req);
if (!map_req) {
SKHPB_DRIVER_D("There is no map_req\n");
spin_unlock_irqrestore(&hpb->map_list_lock, flags);
return -EAGAIN;
}
list_del(&map_req->list_map_req);
spin_unlock_irqrestore(&hpb->map_list_lock, flags);
atomic64_inc(&hpb->alloc_map_req_cnt);
memset(map_req, 0x00, sizeof(struct skhpb_map_req));
map_req->hpb = hpb;
map_req->region = region;
map_req->subregion = subregion;
map_req->subregion_mem_size =
last ? hpb->last_subregion_mem_size : hpb->subregion_mem_size;
map_req->mctx = mctx;
map_req->lun = hpb->lun;
map_req->RSP_start = rsp_info->RSP_start;
if (flag == W_BUFFER) {
map_req->rwbuffer_flag = W_BUFFER;
} else
map_req->rwbuffer_flag = R_BUFFER;
if (skhpb_map_req_issue(hpb, map_req)) {
SKHPB_DRIVER_E("issue Failed!!!\n");
return -ENOMEM;
}
return 0;
}
static struct skhpb_map_ctx *skhpb_get_map_ctx(struct skhpb_lu *hpb)
{
struct skhpb_map_ctx *mctx;
mctx = list_first_entry_or_null(&hpb->lh_map_ctx,
struct skhpb_map_ctx, list_table);
if (mctx) {
list_del_init(&mctx->list_table);
hpb->debug_free_table--;
return mctx;
}
return ERR_PTR(-ENOMEM);
}
static inline void skhpb_add_lru_info(struct skhpb_victim_select_info *lru_info,
struct skhpb_region *cb)
{
cb->region_state = SKHPB_REGION_ACTIVE;
list_add_tail(&cb->list_region, &lru_info->lru);
atomic64_inc(&lru_info->active_count);
}
static inline int skhpb_add_region(struct skhpb_lu *hpb,
struct skhpb_region *cb)
{
struct skhpb_victim_select_info *lru_info;
int subregion;
int err = 0;
lru_info = &hpb->lru_info;
//SKHPB_DRIVER_D("E->active region: %d", cb->region);
for (subregion = 0; subregion < cb->subregion_count; subregion++) {
struct skhpb_subregion *cp;
cp = cb->subregion_tbl + subregion;
cp->mctx = skhpb_get_map_ctx(hpb);
if (IS_ERR(cp->mctx)) {
err = PTR_ERR(cp->mctx);
goto out;
}
cp->subregion_state = SKHPB_SUBREGION_DIRTY;
}
if (!cb->is_pinned)
skhpb_add_lru_info(lru_info, cb);
atomic64_inc(&hpb->region_add);
out:
if (err)
SKHPB_DRIVER_E("get mctx failed. err %d subregion %d free_table %d\n",
err, subregion, hpb->debug_free_table);
return err;
}
static inline void skhpb_put_map_ctx(
struct skhpb_lu *hpb, struct skhpb_map_ctx *mctx)
{
list_add(&mctx->list_table, &hpb->lh_map_ctx);
hpb->debug_free_table++;
}
static inline void skhpb_purge_active_page(struct skhpb_lu *hpb,
struct skhpb_subregion *cp, int state)
{
if (state == SKHPB_SUBREGION_UNUSED) {
skhpb_put_map_ctx(hpb, cp->mctx);
cp->mctx = NULL;
}
cp->subregion_state = state;
}
static inline void skhpb_cleanup_lru_info(
struct skhpb_victim_select_info *lru_info,
struct skhpb_region *cb)
{
list_del_init(&cb->list_region);
cb->region_state = SKHPB_REGION_INACTIVE;
cb->hit_count = 0;
atomic64_dec(&lru_info->active_count);
}
static inline void skhpb_evict_region(struct skhpb_lu *hpb,
struct skhpb_region *cb)
{
struct skhpb_victim_select_info *lru_info;
struct skhpb_subregion *cp;
int subregion;
// If the maximum value is exceeded at the time of region addition,
// it may have already been processed.
if (cb->region_state == SKHPB_REGION_INACTIVE) {
SKHPB_DRIVER_D("Region:%d was already inactivated.\n",
cb->region);
return;
}
lru_info = &hpb->lru_info;
//SKHPB_DRIVER_D("C->EVICT region: %d\n", cb->region);
skhpb_cleanup_lru_info(lru_info, cb);
atomic64_inc(&hpb->region_evict);
for (subregion = 0; subregion < cb->subregion_count; subregion++) {
cp = cb->subregion_tbl + subregion;
skhpb_purge_active_page(hpb, cp, SKHPB_SUBREGION_UNUSED);
}
}
static void skhpb_hit_lru_info(struct skhpb_victim_select_info *lru_info,
struct skhpb_region *cb)
{
switch (lru_info->selection_type) {
case TYPE_LRU:
list_move_tail(&cb->list_region, &lru_info->lru);
break;
case TYPE_LFU:
if (cb->hit_count != 0xffffffff)
cb->hit_count++;
list_move_tail(&cb->list_region, &lru_info->lru);
break;
default:
break;
}
}
static struct skhpb_region *skhpb_victim_lru_info(
struct skhpb_victim_select_info *lru_info)
{
struct skhpb_region *cb;
struct skhpb_region *victim_cb = NULL;
u32 hit_count = 0xffffffff;
switch (lru_info->selection_type) {
case TYPE_LRU:
victim_cb = list_first_entry(&lru_info->lru,
struct skhpb_region, list_region);
break;
case TYPE_LFU:
list_for_each_entry(cb, &lru_info->lru, list_region) {
if (hit_count > cb->hit_count) {
hit_count = cb->hit_count;
victim_cb = cb;
}
}
break;
default:
break;
}
return victim_cb;
}
static int skhpb_check_lru_evict(struct skhpb_lu *hpb, struct skhpb_region *cb)
{
struct skhpb_victim_select_info *lru_info = &hpb->lru_info;
struct skhpb_region *victim_cb;
unsigned long flags;
if (cb->is_pinned)
return 0;
spin_lock_irqsave(&hpb->hpb_lock, flags);
if (!list_empty(&cb->list_region)) {
skhpb_hit_lru_info(lru_info, cb);
goto out;
}
if (cb->region_state != SKHPB_REGION_INACTIVE)
goto out;
if (atomic64_read(&lru_info->active_count) == lru_info->max_lru_active_count) {
victim_cb = skhpb_victim_lru_info(lru_info);
if (!victim_cb) {
SKHPB_DRIVER_E("SKHPB victim_cb is NULL\n");
goto unlock_error;
}
SKHPB_DRIVER_D("max lru case. victim : %d\n", victim_cb->region);
skhpb_evict_region(hpb, victim_cb);
}
if (skhpb_add_region(hpb, cb)) {
SKHPB_DRIVER_E("SKHPB memory allocation failed\n");
goto unlock_error;
}
out:
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
return 0;
unlock_error:
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
return -ENOMEM;
}
static int skhpb_evict_load_region(struct skhpb_lu *hpb,
struct skhpb_rsp_info *rsp_info)
{
struct skhpb_region *cb;
int region, iter;
unsigned long flags;
for (iter = 0; iter < rsp_info->inactive_cnt; iter++) {
region = rsp_info->inactive_list.region[iter];
cb = hpb->region_tbl + region;
if (cb->is_pinned) {
/*
* Pinned active-block should not drop-out.
* But if so, it would treat error as critical,
* and it will run skhpb_eh_work
*/
SKHPB_DRIVER_E("SKHPB pinned active-block drop-out error\n");
return -ENOMEM;
}
if (list_empty(&cb->list_region))
continue;
spin_lock_irqsave(&hpb->hpb_lock, flags);
skhpb_evict_region(hpb, cb);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
}
for (iter = 0; iter < rsp_info->active_cnt; iter++) {
region = rsp_info->active_list.region[iter];
cb = hpb->region_tbl + region;
if (skhpb_check_lru_evict(hpb, cb))
return -ENOMEM;
}
return 0;
}
static inline struct skhpb_rsp_field *skhpb_get_hpb_rsp(
struct ufshcd_lrb *lrbp)
{
return (struct skhpb_rsp_field *)&lrbp->ucd_rsp_ptr->sr.sense_data_len;
}
static int skhpb_rsp_map_cmd_req(struct skhpb_lu *hpb,
struct skhpb_rsp_info *rsp_info)
{
struct skhpb_region *cb;
struct skhpb_subregion *cp;
int region, subregion;
int iter;
int ret;
unsigned long flags;
/*
* Before Issue read buffer CMD for active active block,
* prepare the memory from memory pool.
*/
ret = skhpb_evict_load_region(hpb, rsp_info);
if (ret) {
SKHPB_DRIVER_E("region evict/load failed. ret %d\n", ret);
goto wakeup_ee_worker;
}
for (iter = 0; iter < rsp_info->active_cnt; iter++) {
region = rsp_info->active_list.region[iter];
subregion = rsp_info->active_list.subregion[iter];
cb = hpb->region_tbl + region;
if (region >= hpb->regions_per_lu ||
subregion >= cb->subregion_count) {
SKHPB_DRIVER_E("skhpb issue-map %d - %d range error\n",
region, subregion);
goto wakeup_ee_worker;
}
cp = cb->subregion_tbl + subregion;
/*
* if subregion_state set SKHPB_SUBREGION_ISSUED,
* active_page has already been added to list,
* so it just ends function.
*/
spin_lock_irqsave(&hpb->hpb_lock, flags);
if (cp->subregion_state == SKHPB_SUBREGION_ISSUED) {
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
continue;
}
cp->subregion_state = SKHPB_SUBREGION_ISSUED;
ret = skhpb_clean_dirty_bitmap(hpb, cp);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
if (ret)
continue;
if (!hpb->hba->sdev_ufs_lu[hpb->lun] ||
!hpb->hba->sdev_ufs_lu[hpb->lun]->request_queue)
return -ENODEV;
ret = skhpb_set_map_req(hpb, region, subregion,
cp->mctx, rsp_info, R_BUFFER);
SKHPB_DRIVER_D("SEND READ_BUFFER - Region:%d, SubRegion:%d\n",
region, subregion);
if (ret) {
if (ret == -EAGAIN) {
spin_lock_irqsave(&hpb->hpb_lock, flags);
cp->subregion_state = SKHPB_SUBREGION_DIRTY;
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
rsp_info->inactive_cnt = 0;
if (iter) {
rsp_info->active_list.region[0] = region;
rsp_info->active_list.subregion[0] = subregion;
rsp_info->active_cnt = 1;
}
return ret;
}
SKHPB_DRIVER_E("skhpb_set_map_req error %d\n", ret);
goto wakeup_ee_worker;
}
}
return 0;
wakeup_ee_worker:
hpb->hba->skhpb_state = SKHPB_FAILED;
schedule_work(&hpb->hba->skhpb_eh_work);
return -ENODEV;
}
/* routine : isr (ufs) */
void skhpb_rsp_upiu(struct ufs_hba *hba, struct ufshcd_lrb *lrbp)
{
struct skhpb_lu *hpb;
struct skhpb_rsp_field *rsp_field;
struct skhpb_rsp_info *rsp_info;
int data_seg_len, num, blk_idx, update_alert;
update_alert = be32_to_cpu(lrbp->ucd_rsp_ptr->header.dword_2)
& MASK_RSP_UPIU_HPB_UPDATE_ALERT;
data_seg_len = be32_to_cpu(lrbp->ucd_rsp_ptr->header.dword_2)
& MASK_RSP_UPIU_DATA_SEG_LEN;
if (!update_alert || !data_seg_len) {
bool do_tasklet = false;
if (lrbp->lun >= UFS_UPIU_MAX_GENERAL_LUN)
return;
hpb = hba->skhpb_lup[lrbp->lun];
if (!hpb)
return;
spin_lock(&hpb->rsp_list_lock);
do_tasklet = !list_empty(&hpb->lh_rsp_info);
spin_unlock(&hpb->rsp_list_lock);
if (do_tasklet)
schedule_work(&hpb->skhpb_rsp_work);
return;
}
rsp_field = skhpb_get_hpb_rsp(lrbp);
if ((get_unaligned_be16(rsp_field->sense_data_len + 0)
!= SKHPB_DEV_SENSE_SEG_LEN) ||
rsp_field->desc_type != SKHPB_DEV_DES_TYPE ||
rsp_field->additional_len != SKHPB_DEV_ADDITIONAL_LEN ||
rsp_field->hpb_type == SKHPB_RSP_NONE ||
rsp_field->active_region_cnt > SKHPB_MAX_ACTIVE_NUM ||
rsp_field->inactive_region_cnt > SKHPB_MAX_INACTIVE_NUM)
return;
if (rsp_field->lun >= UFS_UPIU_MAX_GENERAL_LUN) {
SKHPB_DRIVER_E("lun is not general = %d", rsp_field->lun);
return;
}
hpb = hba->skhpb_lup[rsp_field->lun];
if (!hpb) {
SKHPB_DRIVER_E("UFS-LU%d is not SKHPB LU\n", rsp_field->lun);
return;
}
if (hpb->force_map_req_disable)
return;
SKHPB_DRIVER_D("HPB-Info Noti: %d LUN: %d Seg-Len %d, Req_type = %d\n",
rsp_field->hpb_type, rsp_field->lun,
data_seg_len, rsp_field->hpb_type);
if (!hpb->lu_hpb_enable) {
SKHPB_DRIVER_E("LU(%d) not HPB-LU\n", rsp_field->lun);
return;
}
//To verify the values within RESPONSE UPIU.
SKHPB_DRIVER_HEXDUMP("[HPB] RESP UPIU ", 16, 1,
lrbp->ucd_rsp_ptr, sizeof(struct utp_upiu_rsp), 1);
//If wLUMaxActiveHPBRegions == wNumHPBPinnedRegions
if (hpb->lru_info.max_lru_active_count == 0) {
SKHPB_DRIVER_D("max_lru_active_count is 0");
return;
}
switch (rsp_field->hpb_type) {
case SKHPB_RSP_REQ_REGION_UPDATE:
atomic64_inc(&hpb->rb_noti_cnt);
WARN_ON(data_seg_len != SKHPB_DEV_DATA_SEG_LEN);
spin_lock(&hpb->rsp_list_lock);
rsp_info = skhpb_get_req_info(hpb);
spin_unlock(&hpb->rsp_list_lock);
if (!rsp_info)
return;
rsp_info->type = SKHPB_RSP_REQ_REGION_UPDATE;
for (num = 0; num < rsp_field->active_region_cnt; num++) {
blk_idx = num * SKHPB_PER_ACTIVE_INFO_BYTES;
rsp_info->active_list.region[num] =
get_unaligned_be16(rsp_field->hpb_active_field + blk_idx);
rsp_info->active_list.subregion[num] =
get_unaligned_be16(rsp_field->hpb_active_field
+ blk_idx + 2);
SKHPB_DRIVER_D("active num: %d, #block: %d, page#: %d\n",
num + 1,
rsp_info->active_list.region[num],
rsp_info->active_list.subregion[num]);
}
rsp_info->active_cnt = num;
for (num = 0; num < rsp_field->inactive_region_cnt; num++) {
blk_idx = num * SKHPB_PER_INACTIVE_INFO_BYTES;
rsp_info->inactive_list.region[num] =
get_unaligned_be16(rsp_field->hpb_inactive_field + blk_idx);
SKHPB_DRIVER_D("inactive num: %d, #block: %d\n",
num + 1, rsp_info->inactive_list.region[num]);
}
rsp_info->inactive_cnt = num;
SKHPB_DRIVER_D("active cnt: %d, inactive cnt: %d\n",
rsp_info->active_cnt, rsp_info->inactive_cnt);
SKHPB_DRIVER_D("add_list %p -> %p\n",
rsp_info, &hpb->lh_rsp_info);
spin_lock(&hpb->rsp_list_lock);
list_add_tail(&rsp_info->list_rsp_info, &hpb->lh_rsp_info);
spin_unlock(&hpb->rsp_list_lock);
schedule_work(&hpb->skhpb_rsp_work);
break;
case SKHPB_RSP_HPB_RESET:
for (num = 0 ; num < UFS_UPIU_MAX_GENERAL_LUN ; num++) {
hpb = hba->skhpb_lup[num];
if (!hpb || !hpb->lu_hpb_enable)
continue;
atomic64_inc(&hpb->reset_noti_cnt);
spin_lock(&hpb->rsp_list_lock);
rsp_info = skhpb_get_req_info(hpb);
spin_unlock(&hpb->rsp_list_lock);
if (!rsp_info)
return;
rsp_info->type = SKHPB_RSP_HPB_RESET;
spin_lock(&hpb->rsp_list_lock);
list_add_tail(&rsp_info->list_rsp_info, &hpb->lh_rsp_info);
spin_unlock(&hpb->rsp_list_lock);
schedule_work(&hpb->skhpb_rsp_work);
}
break;
default:
SKHPB_DRIVER_E("hpb_type is not available : %d\n",
rsp_field->hpb_type);
break;
}
}
static int skhpb_read_desc(struct ufs_hba *hba,
u8 desc_id, u8 desc_index, u8 *desc_buf, u32 size)
{
int err = 0;
err = ufshcd_query_descriptor_retry(hba, UPIU_QUERY_OPCODE_READ_DESC,
desc_id, desc_index, 0, desc_buf, &size);
if (err) {
SKHPB_DRIVER_E("reading Device Desc failed. err = %d\n", err);
}
return err;
}
static int skhpb_read_device_desc(
struct ufs_hba *hba, u8 *desc_buf, u32 size)
{
return skhpb_read_desc(hba, QUERY_DESC_IDN_DEVICE, 0, desc_buf, size);
}
static int skhpb_read_geo_desc(struct ufs_hba *hba, u8 *desc_buf, u32 size)
{
return skhpb_read_desc(hba, QUERY_DESC_IDN_GEOMETRY, 0,
desc_buf, size);
}
static int skhpb_read_unit_desc(struct ufs_hba *hba, int lun,
u8 *desc_buf, u32 size)
{
return skhpb_read_desc(hba, QUERY_DESC_IDN_UNIT,
lun, desc_buf, size);
}
static inline void skhpb_add_subregion_to_req_list(struct skhpb_lu *hpb,
struct skhpb_subregion *cp)
{
list_add_tail(&cp->list_subregion, &hpb->lh_subregion_req);
cp->subregion_state = SKHPB_SUBREGION_DIRTY;
}
static int skhpb_execute_req(struct skhpb_lu *hpb, unsigned char *cmd,
struct skhpb_subregion *cp)
{
struct ufs_hba *hba = hpb->hba;
struct scsi_device *sdp;
struct request_queue *q;
struct request *req;
struct scsi_request *scsireq;
struct bio bio;
struct bio *pbio = &bio;
struct scsi_sense_hdr sshdr;
unsigned long flags;
int ret = 0;
spin_lock_irqsave(hba->host->host_lock, flags);
sdp = hba->sdev_ufs_lu[hpb->lun];
if (sdp) {
ret = scsi_device_get(sdp);
if (!ret && !scsi_device_online(sdp)) {
ret = -ENODEV;
scsi_device_put(sdp);
}
} else {
ret = -ENODEV;
}
spin_unlock_irqrestore(hba->host->host_lock, flags);
if (ret)
return ret;
q = sdp->request_queue;
req = blk_get_request(q, REQ_OP_SCSI_IN, BLK_MQ_REQ_PREEMPT);
if (IS_ERR(req)) {
ret = PTR_ERR(req);
goto out_put;
}
scsireq = scsi_req(req);
scsireq->cmd_len = COMMAND_SIZE(cmd[0]);
if (scsireq->cmd_len > sizeof(scsireq->__cmd)) {
scsireq->cmd = kmalloc(scsireq->cmd_len, __GFP_RECLAIM);
if (!scsireq->cmd) {
ret = -ENOMEM;
goto out_put_request;
}
} else {
scsireq->cmd = scsireq->__cmd;
}
memcpy(scsireq->cmd, cmd, scsireq->cmd_len);
scsireq->retries = 3;
req->timeout = msecs_to_jiffies(10000);
req->rq_flags |= RQF_QUIET | RQF_PREEMPT;
ret = skhpb_add_bio_page(hpb, q, &bio, hpb->bvec, cp->mctx);
if (ret)
goto out_put_scsi_req;
blk_rq_append_bio(req, &pbio);
blk_execute_rq(q, NULL, req, 1);
if (scsireq->result) {
scsi_normalize_sense(scsireq->sense, SCSI_SENSE_BUFFERSIZE, &sshdr);
SKHPB_DRIVER_E("code %x sense_key %x asc %x ascq %x",
sshdr.response_code, sshdr.sense_key, sshdr.asc,
sshdr.ascq);
SKHPB_DRIVER_E("byte4 %x byte5 %x byte6 %x additional_len %x",
sshdr.byte4, sshdr.byte5, sshdr.byte6,
sshdr.additional_length);
spin_lock_irqsave(&hpb->hpb_lock, flags);
skhpb_error_active_subregion(hpb, cp);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
ret = -EIO;
} else {
spin_lock_irqsave(&hpb->hpb_lock, flags);
ret = skhpb_clean_dirty_bitmap(hpb, cp);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
if (ret) {
SKHPB_DRIVER_E("skhpb_clean_dirty_bitmap error %d", ret);
}
ret = 0;
}
out_put_scsi_req:
scsi_req_free_cmd(scsireq);
out_put_request:
blk_put_request(req);
out_put:
scsi_device_put(sdp);
return ret;
}
static int skhpb_issue_map_req_from_list(struct skhpb_lu *hpb)
{
struct skhpb_subregion *cp, *next_cp;
int ret;
unsigned long flags;
LIST_HEAD(req_list);
spin_lock_irqsave(&hpb->hpb_lock, flags);
list_splice_init(&hpb->lh_subregion_req, &req_list);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
list_for_each_entry_safe(cp, next_cp, &req_list, list_subregion) {
int subregion_mem_size =
cp->last ? hpb->last_subregion_mem_size : hpb->subregion_mem_size;
unsigned char cmd[10] = { 0 };
skhpb_set_read_buf_cmd(cmd, cp->region, cp->subregion,
subregion_mem_size);
SKHPB_DRIVER_D("issue map_request: %d - %d/%d\n",
cp->region, cp->subregion, subregion_mem_size);
ret = skhpb_execute_req(hpb, cmd, cp);
if (ret < 0) {
SKHPB_DRIVER_E("region %d sub %d failed with err %d",
cp->region, cp->subregion, ret);
continue;
}
spin_lock_irqsave(&hpb->hpb_lock, flags);
skhpb_clean_active_subregion(hpb, cp);
list_del_init(&cp->list_subregion);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
}
return 0;
}
static void skhpb_pinned_work_handler(struct work_struct *work)
{
struct delayed_work *dwork = to_delayed_work(work);
struct skhpb_lu *hpb =
container_of(dwork, struct skhpb_lu, skhpb_pinned_work);
int ret;
SKHPB_DRIVER_D("worker start\n");
if (!list_empty(&hpb->lh_subregion_req)) {
pm_runtime_get_sync(SKHPB_DEV(hpb));
ret = skhpb_issue_map_req_from_list(hpb);
/*
* if its function failed at init time,
* skhpb-device will request map-req,
* so it is not critical-error, and just finish work-handler
*/
if (ret)
SKHPB_DRIVER_E("failed map-issue. ret %d\n", ret);
pm_runtime_mark_last_busy(SKHPB_DEV(hpb));
pm_runtime_put_noidle(SKHPB_DEV(hpb));
}
SKHPB_DRIVER_D("worker end\n");
}
static void skhpb_map_req_retry_work_handler(struct work_struct *work)
{
struct skhpb_lu *hpb;
struct delayed_work *dwork = to_delayed_work(work);
struct skhpb_map_req *map_req;
int ret = 0;
unsigned long flags;
LIST_HEAD(retry_list);
hpb = container_of(dwork, struct skhpb_lu, skhpb_map_req_retry_work);
SKHPB_DRIVER_D("retry worker start");
spin_lock_irqsave(&hpb->map_list_lock, flags);
list_splice_init(&hpb->lh_map_req_retry, &retry_list);
spin_unlock_irqrestore(&hpb->map_list_lock, flags);
while (1) {
map_req = list_first_entry_or_null(&retry_list,
struct skhpb_map_req, list_map_req);
if (!map_req) {
SKHPB_DRIVER_D("There is no map_req");
break;
}
list_del(&map_req->list_map_req);
map_req->retry_cnt++;
ret = skhpb_map_req_issue(hpb, map_req);
if (ret) {
SKHPB_DRIVER_E("skhpb_map_req_issue error %d", ret);
goto wakeup_ee_worker;
}
}
SKHPB_DRIVER_D("worker end");
return;
wakeup_ee_worker:
hpb->hba->skhpb_state = SKHPB_FAILED;
schedule_work(&hpb->hba->skhpb_eh_work);
}
static void skhpb_delayed_rsp_work_handler(struct work_struct *work)
{
struct skhpb_lu *hpb = container_of(work, struct skhpb_lu, skhpb_rsp_work);
struct skhpb_rsp_info *rsp_info;
unsigned long flags;
int ret = 0;
SKHPB_DRIVER_D("rsp_work enter");
while (!hpb->hba->clk_gating.is_suspended) {
spin_lock_irqsave(&hpb->rsp_list_lock, flags);
rsp_info = list_first_entry_or_null(&hpb->lh_rsp_info,
struct skhpb_rsp_info, list_rsp_info);
spin_unlock_irqrestore(&hpb->rsp_list_lock, flags);
if (!rsp_info) {
break;
}
SKHPB_RSP_TIME(rsp_info->RSP_start);
switch (rsp_info->type) {
case SKHPB_RSP_REQ_REGION_UPDATE:
ret = skhpb_rsp_map_cmd_req(hpb, rsp_info);
if (ret)
return;
break;
case SKHPB_RSP_HPB_RESET:
skhpb_purge_active_block(hpb);
skhpb_map_loading_trigger(hpb, true, false);
break;
default:
break;
}
spin_lock_irqsave(&hpb->rsp_list_lock, flags);
list_del_init(&rsp_info->list_rsp_info);
list_add_tail(&rsp_info->list_rsp_info, &hpb->lh_rsp_info_free);
spin_unlock_irqrestore(&hpb->rsp_list_lock, flags);
}
SKHPB_DRIVER_D("rsp_work end");
}
static void skhpb_init_constant(void)
{
skhpb_sects_per_blk_shift = ffs(SKHPB_BLOCK) - ffs(SKHPB_SECTOR);
SKHPB_DRIVER_D("skhpb_sects_per_blk_shift: %u %u\n",
skhpb_sects_per_blk_shift, ffs(SKHPB_SECTORS_PER_BLOCK) - 1);
skhpb_bits_per_dword_shift = ffs(SKHPB_BITS_PER_DWORD) - 1;
skhpb_bits_per_dword_mask = SKHPB_BITS_PER_DWORD - 1;
SKHPB_DRIVER_D("bits_per_dword %u shift %u mask 0x%X\n",
SKHPB_BITS_PER_DWORD, skhpb_bits_per_dword_shift, skhpb_bits_per_dword_mask);
}
static void skhpb_table_mempool_remove(struct skhpb_lu *hpb)
{
struct skhpb_map_ctx *mctx, *next;
int i;
/*
* the mctx in the lh_map_ctx has been allocated completely.
*/
list_for_each_entry_safe(mctx, next, &hpb->lh_map_ctx, list_table) {
list_del(&mctx->list_table);
for (i = 0; i < hpb->mpages_per_subregion; i++)
__free_page(mctx->m_page[i]);
kvfree(mctx->ppn_dirty);
kfree(mctx->m_page);
kfree(mctx);
skhpb_alloc_mctx--;
}
}
static int skhpb_init_pinned_active_block(struct skhpb_lu *hpb,
struct skhpb_region *cb)
{
struct skhpb_subregion *cp;
int subregion, j;
int err = 0;
unsigned long flags;
for (subregion = 0 ; subregion < cb->subregion_count ; subregion++) {
cp = cb->subregion_tbl + subregion;
cp->mctx = skhpb_get_map_ctx(hpb);
if (IS_ERR(cp->mctx)) {
err = PTR_ERR(cp->mctx);
goto release;
}
spin_lock_irqsave(&hpb->hpb_lock, flags);
skhpb_add_subregion_to_req_list(hpb, cp);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
}
return 0;
release:
for (j = 0 ; j < subregion ; j++) {
cp = cb->subregion_tbl + j;
skhpb_put_map_ctx(hpb, cp->mctx);
}
return err;
}
static inline bool skhpb_is_pinned(
struct skhpb_lu_desc *lu_desc, int region)
{
if (lu_desc->lu_hpb_pinned_end_offset != -1 &&
region >= lu_desc->hpb_pinned_region_startidx &&
region <= lu_desc->lu_hpb_pinned_end_offset)
return true;
return false;
}
static inline void skhpb_init_jobs(struct skhpb_lu *hpb)
{
INIT_DELAYED_WORK(&hpb->skhpb_pinned_work, skhpb_pinned_work_handler);
INIT_DELAYED_WORK(&hpb->skhpb_map_req_retry_work, skhpb_map_req_retry_work_handler);
INIT_WORK(&hpb->skhpb_rsp_work, skhpb_delayed_rsp_work_handler);
}
static inline void skhpb_cancel_jobs(struct skhpb_lu *hpb)
{
cancel_delayed_work_sync(&hpb->skhpb_pinned_work);
cancel_work_sync(&hpb->skhpb_rsp_work);
cancel_delayed_work_sync(&hpb->skhpb_map_req_retry_work);
}
static void skhpb_init_subregion_tbl(struct skhpb_lu *hpb,
struct skhpb_region *cb, bool last_region)
{
int subregion;
for (subregion = 0 ; subregion < cb->subregion_count ; subregion++) {
struct skhpb_subregion *cp = cb->subregion_tbl + subregion;
cp->region = cb->region;
cp->subregion = subregion;
cp->subregion_state = SKHPB_SUBREGION_UNUSED;
cp->last = (last_region && subregion == cb->subregion_count - 1);
}
}
static inline int skhpb_alloc_subregion_tbl(struct skhpb_lu *hpb,
struct skhpb_region *cb, int subregion_count)
{
cb->subregion_tbl = kzalloc(sizeof(struct skhpb_subregion) * subregion_count,
GFP_KERNEL);
if (!cb->subregion_tbl)
return -ENOMEM;
cb->subregion_count = subregion_count;
return 0;
}
static int skhpb_table_mempool_init(struct skhpb_lu *hpb,
int num_regions, int subregions_per_region,
int entry_count, int entry_byte)
{
int i, j;
struct skhpb_map_ctx *mctx = NULL;
for (i = 0 ; i < num_regions * subregions_per_region ; i++) {
mctx = kzalloc(sizeof(struct skhpb_map_ctx), GFP_KERNEL);
if (!mctx)
goto release_mem;
mctx->m_page = kzalloc(sizeof(struct page *) *
hpb->mpages_per_subregion, GFP_KERNEL);
if (!mctx->m_page)
goto release_mem;
mctx->ppn_dirty = kvzalloc(entry_count / BITS_PER_BYTE, GFP_KERNEL);
if (!mctx->ppn_dirty)
goto release_mem;
for (j = 0; j < hpb->mpages_per_subregion; j++) {
mctx->m_page[j] = alloc_page(GFP_KERNEL | __GFP_ZERO);
if (!mctx->m_page[j])
goto release_mem;
}
/* SKSKHPB_DRIVER_D("[%d] mctx->m_page %p get_order %d\n", i,
mctx->m_page, get_order(hpb->mpages_per_subregion)); */
INIT_LIST_HEAD(&mctx->list_table);
list_add(&mctx->list_table, &hpb->lh_map_ctx);
hpb->debug_free_table++;
}
skhpb_alloc_mctx = num_regions * subregions_per_region;
/* SKSKHPB_DRIVER_D("number of mctx %d %d %d. debug_free_table %d\n",
num_regions * subregions_per_region, num_regions,
subregions_per_region, hpb->debug_free_table); */
return 0;
release_mem:
/*
* mctxs already added in lh_map_ctx will be removed
* in the caller function.
*/
if (!mctx)
goto out;
if (mctx->m_page) {
for (j = 0; j < hpb->mpages_per_subregion; j++)
if (mctx->m_page[j])
__free_page(mctx->m_page[j]);
kfree(mctx->m_page);
}
kvfree(mctx->ppn_dirty);
kfree(mctx);
out:
return -ENOMEM;
}
static int skhpb_req_mempool_init(struct ufs_hba *hba,
struct skhpb_lu *hpb, int qd)
{
struct skhpb_rsp_info *rsp_info = NULL;
struct skhpb_map_req *map_req = NULL;
int i, rec_qd;
if (!qd) {
qd = hba->nutrs;
SKHPB_DRIVER_D("hba->nutrs = %d\n", hba->nutrs);
}
if (hpb->hpb_ver >= 0x0200)
rec_qd = hpb->lu_max_active_regions;
else
rec_qd = qd;
INIT_LIST_HEAD(&hpb->lh_rsp_info_free);
INIT_LIST_HEAD(&hpb->lh_map_req_free);
INIT_LIST_HEAD(&hpb->lh_map_req_retry);
hpb->rsp_info = vzalloc(rec_qd * sizeof(struct skhpb_rsp_info));
if (!hpb->rsp_info)
goto release_mem;
hpb->map_req = vzalloc(qd * sizeof(struct skhpb_map_req));
if (!hpb->map_req)
goto release_mem;
for (i = 0; i < rec_qd; i++) {
rsp_info = hpb->rsp_info + i;
INIT_LIST_HEAD(&rsp_info->list_rsp_info);
list_add_tail(&rsp_info->list_rsp_info, &hpb->lh_rsp_info_free);
}
for (i = 0; i < qd; i++) {
map_req = hpb->map_req + i;
INIT_LIST_HEAD(&map_req->list_map_req);
list_add_tail(&map_req->list_map_req, &hpb->lh_map_req_free);
}
return 0;
release_mem:
return -ENOMEM;
}
static void skhpb_init_lu_constant(struct skhpb_lu *hpb,
struct skhpb_lu_desc *lu_desc,
struct skhpb_func_desc *func_desc)
{
unsigned long long region_unit_size, region_mem_size;
unsigned long long last_subregion_unit_size;
int entries_per_region;
/* From descriptors */
region_unit_size = SKHPB_SECTOR * (1ULL << func_desc->hpb_region_size);
region_mem_size = region_unit_size / SKHPB_BLOCK * SKHPB_ENTRY_SIZE;
hpb->subregion_unit_size = SKHPB_SECTOR * (1ULL << func_desc->hpb_subregion_size);
hpb->subregion_mem_size = hpb->subregion_unit_size /
SKHPB_BLOCK * SKHPB_ENTRY_SIZE;
if (func_desc->hpb_region_size == func_desc->hpb_subregion_size)
hpb->identical_size = true;
else
hpb->identical_size = false;
last_subregion_unit_size =
lu_desc->lu_logblk_cnt *
(1ULL << lu_desc->lu_logblk_size) % hpb->subregion_unit_size;
hpb->last_subregion_mem_size =
last_subregion_unit_size / SKHPB_BLOCK * SKHPB_ENTRY_SIZE;
if (hpb->last_subregion_mem_size == 0)
hpb->last_subregion_mem_size = hpb->subregion_mem_size;
hpb->hpb_ver = func_desc->hpb_ver;
hpb->lu_max_active_regions = lu_desc->lu_max_active_hpb_regions;
hpb->lru_info.max_lru_active_count =
lu_desc->lu_max_active_hpb_regions
- lu_desc->lu_num_hpb_pinned_regions;
/* relation : lu <-> region <-> sub region <-> entry */
hpb->lu_num_blocks = lu_desc->lu_logblk_cnt;
entries_per_region = region_mem_size / SKHPB_ENTRY_SIZE;
hpb->entries_per_subregion = hpb->subregion_mem_size / SKHPB_ENTRY_SIZE;
hpb->subregions_per_region = region_mem_size / hpb->subregion_mem_size;
hpb->hpb_control_mode = func_desc->hpb_control_mode;
#if defined(SKHPB_READ_LARGE_CHUNK_SUPPORT)
hpb->ppn_dirties_per_subregion =
hpb->entries_per_subregion/BITS_PER_PPN_DIRTY;
/* SKSKHPB_DRIVER_D("ppn_dirties_per_subregion:%d, %d \n",
hpb->ppn_dirties_per_subregion, BITS_PER_PPN_DIRTY); */
#endif
/*
* 1. regions_per_lu
* = (lu_num_blocks * 4096) / region_unit_size
* = (lu_num_blocks * SKHPB_ENTRY_SIZE) / region_mem_size
* = lu_num_blocks / (region_mem_size / SKHPB_ENTRY_SIZE)
*
* 2. regions_per_lu = lu_num_blocks / subregion_mem_size (is trik...)
* if SKHPB_ENTRY_SIZE != subregions_per_region, it is error.
*/
hpb->regions_per_lu = (hpb->lu_num_blocks
+ (region_mem_size / SKHPB_ENTRY_SIZE) - 1)
/ (region_mem_size / SKHPB_ENTRY_SIZE);
hpb->subregions_per_lu = (hpb->lu_num_blocks
+ (hpb->subregion_mem_size / SKHPB_ENTRY_SIZE) - 1)
/ (hpb->subregion_mem_size / SKHPB_ENTRY_SIZE);
/* mempool info */
hpb->mpage_bytes = PAGE_SIZE;
hpb->mpages_per_subregion = hpb->subregion_mem_size / hpb->mpage_bytes;
/* Bitmask Info. */
hpb->dwords_per_subregion = hpb->entries_per_subregion / SKHPB_BITS_PER_DWORD;
hpb->entries_per_region_shift = ffs(entries_per_region) - 1;
hpb->entries_per_region_mask = entries_per_region - 1;
hpb->entries_per_subregion_shift = ffs(hpb->entries_per_subregion) - 1;
hpb->entries_per_subregion_mask = hpb->entries_per_subregion - 1;
SKHPB_DRIVER_I("===== Device Descriptor =====\n");
SKHPB_DRIVER_I("hpb_region_size = %d, hpb_subregion_size = %d\n",
func_desc->hpb_region_size,
func_desc->hpb_subregion_size);
SKHPB_DRIVER_I("===== Constant Values =====\n");
SKHPB_DRIVER_I("region_unit_size = %llu, region_mem_size %llu\n",
region_unit_size, region_mem_size);
SKHPB_DRIVER_I("subregion_unit_size = %llu, subregion_mem_size %d\n",
hpb->subregion_unit_size, hpb->subregion_mem_size);
SKHPB_DRIVER_I("last_subregion_mem_size = %d\n", hpb->last_subregion_mem_size);
SKHPB_DRIVER_I("lu_num_blks = %llu, reg_per_lu = %d, subreg_per_lu = %d\n",
hpb->lu_num_blocks, hpb->regions_per_lu,
hpb->subregions_per_lu);
SKHPB_DRIVER_I("subregions_per_region = %d\n",
hpb->subregions_per_region);
SKHPB_DRIVER_I("entries_per_region %u shift %u mask 0x%X\n",
entries_per_region, hpb->entries_per_region_shift,
hpb->entries_per_region_mask);
SKHPB_DRIVER_I("entries_per_subregion %u shift %u mask 0x%X\n",
hpb->entries_per_subregion,
hpb->entries_per_subregion_shift,
hpb->entries_per_subregion_mask);
SKHPB_DRIVER_I("mpages_per_subregion : %d\n",
hpb->mpages_per_subregion);
SKHPB_DRIVER_I("===================================\n");
}
static int skhpb_lu_hpb_init(struct ufs_hba *hba, struct skhpb_lu *hpb,
struct skhpb_func_desc *func_desc,
struct skhpb_lu_desc *lu_desc, int lun,
bool *do_work_lun)
{
struct skhpb_region *cb;
struct skhpb_subregion *cp;
int region, subregion;
int total_subregion_count, subregion_count;
int ret, j;
*do_work_lun = false;
spin_lock_init(&hpb->hpb_lock);
spin_lock_init(&hpb->rsp_list_lock);
spin_lock_init(&hpb->map_list_lock);
/* init lru information */
INIT_LIST_HEAD(&hpb->lru_info.lru);
hpb->lru_info.selection_type = TYPE_LRU;
INIT_LIST_HEAD(&hpb->lh_subregion_req);
INIT_LIST_HEAD(&hpb->lh_rsp_info);
INIT_LIST_HEAD(&hpb->lh_map_ctx);
hpb->lu_hpb_enable = true;
skhpb_init_lu_constant(hpb, lu_desc, func_desc);
hpb->region_tbl = vzalloc(sizeof(struct skhpb_region) * hpb->regions_per_lu);
if (!hpb->region_tbl)
return -ENOMEM;
SKHPB_DRIVER_D("active_block_table bytes: %lu\n",
(sizeof(struct skhpb_region) * hpb->regions_per_lu));
ret = skhpb_table_mempool_init(hpb,
lu_desc->lu_max_active_hpb_regions,
hpb->subregions_per_region,
hpb->entries_per_subregion, SKHPB_ENTRY_SIZE);
if (ret) {
SKHPB_DRIVER_E("ppn table mempool init fail!\n");
goto release_mempool;
}
ret = skhpb_req_mempool_init(hba, hpb, lu_desc->lu_queue_depth);
if (ret) {
SKHPB_DRIVER_E("rsp_info_mempool init fail!\n");
goto release_mempool;
}
total_subregion_count = hpb->subregions_per_lu;
skhpb_init_jobs(hpb);
SKHPB_DRIVER_D("total_subregion_count: %d\n", total_subregion_count);
for (region = 0, subregion_count = 0,
total_subregion_count = hpb->subregions_per_lu;
region < hpb->regions_per_lu;
region++, total_subregion_count -= subregion_count) {
cb = hpb->region_tbl + region;
cb->region = region;
/* init lru region information*/
INIT_LIST_HEAD(&cb->list_region);
cb->hit_count = 0;
subregion_count = min(total_subregion_count,
hpb->subregions_per_region);
/* SKSKHPB_DRIVER_D("total: %d subregion_count: %d\n",
total_subregion_count, subregion_count); */
ret = skhpb_alloc_subregion_tbl(hpb, cb, subregion_count);
if (ret)
goto release_region_cp;
skhpb_init_subregion_tbl(hpb, cb, region == hpb->regions_per_lu - 1);
if (skhpb_is_pinned(lu_desc, region)) {
SKHPB_DRIVER_D("region: %d PINNED %d ~ %d\n",
region, lu_desc->hpb_pinned_region_startidx,
lu_desc->lu_hpb_pinned_end_offset);
ret = skhpb_init_pinned_active_block(hpb, cb);
if (ret)
goto release_region_cp;
*do_work_lun = true;
cb->is_pinned = true;
cb->region_state = SKHPB_REGION_ACTIVE;
} else {
/* SKSKHPB_DRIVER_D("region: %d inactive\n", cb->region); */
cb->is_pinned = false;
cb->region_state = SKHPB_REGION_INACTIVE;
}
}
if (total_subregion_count != 0) {
SKHPB_DRIVER_E("error total_subregion_count: %d\n",
total_subregion_count);
goto release_region_cp;
}
hpb->hba = hba;
hpb->lun = lun;
/*
* even if creating sysfs failed, skhpb could run normally.
* so we don't deal with error handling
*/
skhpb_create_sysfs(hba, hpb);
return 0;
release_region_cp:
for (j = 0 ; j < region ; j++) {
cb = hpb->region_tbl + j;
if (cb->subregion_tbl) {
for (subregion = 0; subregion < cb->subregion_count;
subregion++) {
cp = cb->subregion_tbl + subregion;
if (cp->mctx)
skhpb_put_map_ctx(hpb, cp->mctx);
}
kfree(cb->subregion_tbl);
}
}
release_mempool:
skhpb_table_mempool_remove(hpb);
*do_work_lun = false;
return ret;
}
static int skhpb_get_hpb_lu_desc(struct ufs_hba *hba,
struct skhpb_lu_desc *lu_desc, int lun)
{
int ret;
u8 logical_buf[SKHPB_QUERY_DESC_UNIT_MAX_SIZE] = { 0 };
ret = skhpb_read_unit_desc(hba, lun, logical_buf,
SKHPB_QUERY_DESC_UNIT_MAX_SIZE);
if (ret) {
SKHPB_DRIVER_E("read unit desc failed. ret %d\n", ret);
return ret;
}
lu_desc->lu_queue_depth = logical_buf[UNIT_DESC_PARAM_LU_Q_DEPTH];
// 2^log, ex) 0x0C = 4KB
lu_desc->lu_logblk_size = logical_buf[UNIT_DESC_PARAM_LOGICAL_BLK_SIZE];
lu_desc->lu_logblk_cnt =
get_unaligned_be64(&logical_buf[UNIT_DESC_PARAM_LOGICAL_BLK_COUNT]);
if (logical_buf[UNIT_DESC_PARAM_LU_ENABLE] == LU_HPB_ENABLE)
lu_desc->lu_hpb_enable = true;
else
lu_desc->lu_hpb_enable = false;
lu_desc->lu_max_active_hpb_regions =
get_unaligned_be16(logical_buf +
UNIT_DESC_HPB_LU_MAX_ACTIVE_REGIONS);
lu_desc->hpb_pinned_region_startidx =
get_unaligned_be16(logical_buf +
UNIT_DESC_HPB_LU_PIN_REGION_START_OFFSET);
lu_desc->lu_num_hpb_pinned_regions =
get_unaligned_be16(logical_buf +
UNIT_DESC_HPB_LU_NUM_PIN_REGIONS);
if (lu_desc->lu_hpb_enable) {
SKHPB_DRIVER_D("LUN(%d) [0A] bLogicalBlockSize %d\n",
lun, lu_desc->lu_logblk_size);
SKHPB_DRIVER_D("LUN(%d) [0B] qLogicalBlockCount %llu\n",
lun, lu_desc->lu_logblk_cnt);
SKHPB_DRIVER_D("LUN(%d) [03] bLuEnable %d\n",
lun, logical_buf[UNIT_DESC_PARAM_LU_ENABLE]);
SKHPB_DRIVER_D("LUN(%d) [06] bLuQueueDepth %d\n",
lun, lu_desc->lu_queue_depth);
SKHPB_DRIVER_D("LUN(%d) [23:24] wLUMaxActiveHPBRegions %d\n",
lun, lu_desc->lu_max_active_hpb_regions);
SKHPB_DRIVER_D("LUN(%d) [25:26] wHPBPinnedRegionStartIdx %d\n",
lun, lu_desc->hpb_pinned_region_startidx);
SKHPB_DRIVER_D("LUN(%d) [27:28] wNumHPBPinnedRegions %d\n",
lun, lu_desc->lu_num_hpb_pinned_regions);
}
if (lu_desc->lu_num_hpb_pinned_regions > 0) {
lu_desc->lu_hpb_pinned_end_offset =
lu_desc->hpb_pinned_region_startidx +
lu_desc->lu_num_hpb_pinned_regions - 1;
} else
lu_desc->lu_hpb_pinned_end_offset = -1;
if (lu_desc->lu_hpb_enable)
SKHPB_DRIVER_I("Enable, LU: %d, MAX_REGION: %d, PIN: %d - %d\n",
lun,
lu_desc->lu_max_active_hpb_regions,
lu_desc->hpb_pinned_region_startidx,
lu_desc->lu_num_hpb_pinned_regions);
return 0;
}
static void skhpb_quirk_setup(struct ufs_hba *hba,
struct skhpb_func_desc *desc)
{
if (hba->dev_quirks & SKHPB_QUIRK_PURGE_HINT_INFO_WHEN_SLEEP) {
hba->skhpb_quirk |= SKHPB_QUIRK_PURGE_HINT_INFO_WHEN_SLEEP;
SKHPB_DRIVER_I("QUIRK set PURGE_HINT_INFO_WHEN_SLEEP\n");
}
}
static int skhpb_read_dev_desc_support(struct ufs_hba *hba,
struct skhpb_func_desc *desc)
{
u8 desc_buf[SKHPB_QUERY_DESC_DEVICE_MAX_SIZE];
int err;
err = skhpb_read_device_desc(hba, desc_buf,
SKHPB_QUERY_DESC_DEVICE_MAX_SIZE);
if (err)
return err;
if (desc_buf[DEVICE_DESC_PARAM_FEAT_SUP] &
SKHPB_UFS_FEATURE_SUPPORT_HPB_BIT) {
hba->skhpb_feat |= SKHPB_UFS_FEATURE_SUPPORT_HPB_BIT;
SKHPB_DRIVER_I("FeaturesSupport= support\n");
} else {
SKHPB_DRIVER_I("FeaturesSupport= not support\n");
return -ENODEV;
}
desc->lu_cnt = desc_buf[DEVICE_DESC_PARAM_NUM_LU];
SKHPB_DRIVER_D("Dev LU count= %d\n", desc->lu_cnt);
desc->spec_ver =
(u16)SKHPB_SHIFT_BYTE_1(desc_buf[DEVICE_DESC_PARAM_SPEC_VER]) |
(u16)SKHPB_SHIFT_BYTE_0(desc_buf[DEVICE_DESC_PARAM_SPEC_VER + 1]);
SKHPB_DRIVER_I("Dev Spec Ver= %x.%x\n",
SKHPB_GET_BYTE_1(desc->spec_ver),
SKHPB_GET_BYTE_0(desc->spec_ver));
desc->hpb_ver =
(u16)SKHPB_SHIFT_BYTE_1(desc_buf[DEVICE_DESC_PARAM_HPB_VER]) |
(u16)SKHPB_SHIFT_BYTE_0(desc_buf[DEVICE_DESC_PARAM_HPB_VER + 1]);
SKHPB_DRIVER_I("Dev Ver= %x.%x.%x, DD Ver= %x.%x.%x\n",
(desc->hpb_ver >> 8) & 0xf,
(desc->hpb_ver >> 4) & 0xf,
(desc->hpb_ver >> 0) & 0xf,
SKHPB_GET_BYTE_2(SKHPB_DD_VER),
SKHPB_GET_BYTE_1(SKHPB_DD_VER),
SKHPB_GET_BYTE_0(SKHPB_DD_VER));
skhpb_quirk_setup(hba, desc);
if (hba->skhpb_quirk & SKHPB_QUIRK_ALWAYS_DEVICE_CONTROL_MODE)
desc->hpb_control_mode = DEV_CTRL_MODE;
else
desc->hpb_control_mode = (u8)desc_buf[DEVICE_DESC_PARAM_HPB_CONTROL];
SKHPB_DRIVER_I("HPB Control Mode = %s",
(desc->hpb_control_mode)?"DEV MODE":"HOST MODE");
if (desc->hpb_control_mode == HOST_CTRL_MODE) {
SKHPB_DRIVER_E("Driver does not support Host Control Mode");
return -ENODEV;
}
hba->hpb_control_mode = desc->hpb_control_mode;
return 0;
}
static int skhpb_read_geo_desc_support(struct ufs_hba *hba,
struct skhpb_func_desc *desc)
{
int err;
u8 geometry_buf[SKHPB_QUERY_DESC_GEOMETRY_MAX_SIZE];
err = skhpb_read_geo_desc(hba, geometry_buf,
SKHPB_QUERY_DESC_GEOMETRY_MAX_SIZE);
if (err)
return err;
desc->hpb_region_size = geometry_buf[GEOMETRY_DESC_HPB_REGION_SIZE];
desc->hpb_number_lu = geometry_buf[GEOMETRY_DESC_HPB_NUMBER_LU];
desc->hpb_subregion_size =
geometry_buf[GEOMETRY_DESC_HPB_SUBREGION_SIZE];
desc->hpb_device_max_active_regions =
get_unaligned_be16(geometry_buf
+ GEOMETRY_DESC_HPB_DEVICE_MAX_ACTIVE_REGIONS);
SKHPB_DRIVER_D("[48] bHPBRegionSize %u\n", desc->hpb_region_size);
SKHPB_DRIVER_D("[49] bHPBNumberLU %u\n", desc->hpb_number_lu);
SKHPB_DRIVER_D("[4A] bHPBSubRegionSize %u\n", desc->hpb_subregion_size);
SKHPB_DRIVER_D("[4B:4C] wDeviceMaxActiveHPBRegions %u\n",
desc->hpb_device_max_active_regions);
if (desc->hpb_number_lu == 0) {
SKHPB_DRIVER_E("HPB is not supported\n");
return -ENODEV;
}
/* for activation */
hba->skhpb_max_regions = desc->hpb_device_max_active_regions;
return 0;
}
int skhpb_control_validation(struct ufs_hba *hba,
struct skhpb_config_desc *config)
{
unsigned int num_regions = 0;
int lun;
if (!(hba->skhpb_feat & SKHPB_UFS_FEATURE_SUPPORT_HPB_BIT))
return -ENOTSUPP;
for (lun = 0 ; lun < UFS_UPIU_MAX_GENERAL_LUN ; lun++) {
unsigned char *unit = config->unit[lun];
if (unit[SKHPB_CONF_LU_ENABLE] >= LU_SET_MAX)
return -EINVAL;
/* total should not exceed max_active_regions */
num_regions += unit[SKHPB_CONF_ACTIVE_REGIONS] << 8;
num_regions += unit[SKHPB_CONF_ACTIVE_REGIONS + 1];
if (num_regions > hba->skhpb_max_regions)
return -EINVAL;
}
return 0;
}
static int skhpb_init(struct ufs_hba *hba)
{
struct skhpb_func_desc func_desc;
int lun, ret, retries;
int hpb_dev = 0;
bool do_work;
pm_runtime_get_sync(hba->dev);
ret = skhpb_read_dev_desc_support(hba, &func_desc);
if (ret)
goto out_state;
ret = skhpb_read_geo_desc_support(hba, &func_desc);
if (ret)
goto out_state;
for (retries = 0; retries < 20; retries++) {
if (!hba->lrb_in_use) {
ret = ufshcd_query_flag(hba, UPIU_QUERY_OPCODE_SET_FLAG,
QUERY_FLAG_IDN_HPB_RESET, NULL);
if (!ret) {
SKHPB_DRIVER_I("Query fHPBReset is successfully sent retries = %d\n", retries);
break;
}
} else
msleep(200);
}
if (ret == 0) {
bool fHPBReset = true;
ufshcd_query_flag_retry(hba,
UPIU_QUERY_OPCODE_READ_FLAG,
QUERY_FLAG_IDN_HPB_RESET,
&fHPBReset);
if (fHPBReset)
SKHPB_DRIVER_I("fHPBReset still set\n");
}
skhpb_init_constant();
do_work = false;
for (lun = 0 ; lun < UFS_UPIU_MAX_GENERAL_LUN ; lun++) {
struct skhpb_lu_desc lu_desc;
bool do_work_lun;
ret = skhpb_get_hpb_lu_desc(hba, &lu_desc, lun);
if (ret)
goto out_state;
if (lu_desc.lu_hpb_enable == false)
continue;
hba->skhpb_lup[lun] = kzalloc(sizeof(struct skhpb_lu),
GFP_KERNEL);
if (!hba->skhpb_lup[lun]) {
ret = -ENOMEM;
goto out_free_mem;
}
ret = skhpb_lu_hpb_init(hba, hba->skhpb_lup[lun],
&func_desc, &lu_desc, lun, &do_work_lun);
if (ret) {
if (ret == -ENODEV)
continue;
else
goto out_free_mem;
}
do_work |= do_work_lun;
hba->skhpb_quicklist_lu_enable[hpb_dev] = lun;
SKHPB_DRIVER_D("skhpb_quicklist_lu_enable[%d] = %d\n", hpb_dev, lun);
hpb_dev++;
}
if (hpb_dev)
goto done;
goto out_free_mem;
done:
INIT_WORK(&hba->skhpb_eh_work, skhpb_error_handler);
hba->skhpb_state = SKHPB_PRESENT;
hba->issue_ioctl = false;
pm_runtime_mark_last_busy(hba->dev);
pm_runtime_put_noidle(hba->dev);
if (do_work) {
for (lun = 0; lun < UFS_UPIU_MAX_GENERAL_LUN; lun++) {
struct skhpb_lu *hpb = hba->skhpb_lup[lun];
if (hpb)
schedule_delayed_work(&hpb->skhpb_pinned_work, 0);
}
}
return 0;
out_free_mem:
skhpb_release(hba, SKHPB_NOT_SUPPORTED);
out_state:
hba->skhpb_state = SKHPB_NOT_SUPPORTED;
pm_runtime_mark_last_busy(hba->dev);
pm_runtime_put_noidle(hba->dev);
return ret;
}
static void skhpb_map_loading_trigger(struct skhpb_lu *hpb,
bool only_pinned, bool do_work_handler)
{
int region, subregion;
unsigned long flags;
if (do_work_handler)
goto work_out;
flush_delayed_work(&hpb->skhpb_pinned_work);
for (region = 0 ; region < hpb->regions_per_lu ; region++) {
struct skhpb_region *cb;
cb = hpb->region_tbl + region;
if (cb->region_state != SKHPB_REGION_ACTIVE &&
!cb->is_pinned)
continue;
if ((only_pinned && cb->is_pinned) ||
!only_pinned) {
spin_lock_irqsave(&hpb->hpb_lock, flags);
for (subregion = 0; subregion < cb->subregion_count;
subregion++)
skhpb_add_subregion_to_req_list(hpb,
cb->subregion_tbl + subregion);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
do_work_handler = true;
}
}
work_out:
if (do_work_handler)
schedule_delayed_work(&hpb->skhpb_pinned_work, 0);
}
static void skhpb_purge_active_block(struct skhpb_lu *hpb)
{
int region, subregion;
int state;
struct skhpb_region *cb;
struct skhpb_subregion *cp;
unsigned long flags;
spin_lock_irqsave(&hpb->hpb_lock, flags);
for (region = 0 ; region < hpb->regions_per_lu ; region++) {
cb = hpb->region_tbl + region;
if (cb->region_state == SKHPB_REGION_INACTIVE) {
continue;
}
if (cb->is_pinned) {
state = SKHPB_SUBREGION_DIRTY;
} else if (cb->region_state == SKHPB_REGION_ACTIVE) {
state = SKHPB_SUBREGION_UNUSED;
skhpb_cleanup_lru_info(&hpb->lru_info, cb);
} else {
SKHPB_DRIVER_E("Unsupported state of region\n");
continue;
}
SKHPB_DRIVER_D("region %d state %d dft %d\n",
region, state,
hpb->debug_free_table);
for (subregion = 0 ; subregion < cb->subregion_count;
subregion++) {
cp = cb->subregion_tbl + subregion;
skhpb_purge_active_page(hpb, cp, state);
}
SKHPB_DRIVER_D("region %d state %d dft %d\n",
region, state, hpb->debug_free_table);
}
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
}
static void skhpb_retrieve_rsp_info(struct skhpb_lu *hpb)
{
struct skhpb_rsp_info *rsp_info;
unsigned long flags;
while (1) {
spin_lock_irqsave(&hpb->rsp_list_lock, flags);
rsp_info = list_first_entry_or_null(&hpb->lh_rsp_info,
struct skhpb_rsp_info, list_rsp_info);
if (!rsp_info) {
spin_unlock_irqrestore(&hpb->rsp_list_lock, flags);
break;
}
list_move_tail(&rsp_info->list_rsp_info,
&hpb->lh_rsp_info_free);
spin_unlock_irqrestore(&hpb->rsp_list_lock, flags);
SKHPB_DRIVER_D("add_list %p -> %p",
&hpb->lh_rsp_info_free, rsp_info);
}
}
static void skhpb_probe(struct ufs_hba *hba)
{
struct skhpb_lu *hpb;
int lu;
for (lu = 0 ; lu < UFS_UPIU_MAX_GENERAL_LUN ; lu++) {
hpb = hba->skhpb_lup[lu];
if (hpb && hpb->lu_hpb_enable) {
skhpb_cancel_jobs(hpb);
skhpb_retrieve_rsp_info(hpb);
skhpb_purge_active_block(hpb);
SKHPB_DRIVER_I("SKHPB lun %d reset\n", lu);
// tasklet_init(&hpb->hpb_work_handler,
// skhpb_work_handler_fn, (unsigned long)hpb);
}
}
hba->skhpb_state = SKHPB_PRESENT;
}
static void skhpb_destroy_subregion_tbl(struct skhpb_lu *hpb,
struct skhpb_region *cb)
{
int subregion;
for (subregion = 0 ; subregion < cb->subregion_count ; subregion++) {
struct skhpb_subregion *cp;
cp = cb->subregion_tbl + subregion;
cp->subregion_state = SKHPB_SUBREGION_UNUSED;
skhpb_put_map_ctx(hpb, cp->mctx);
}
kfree(cb->subregion_tbl);
}
static void skhpb_destroy_region_tbl(struct skhpb_lu *hpb)
{
int region;
if (!hpb->region_tbl)
return;
for (region = 0 ; region < hpb->regions_per_lu ; region++) {
struct skhpb_region *cb;
cb = hpb->region_tbl + region;
if (cb->region_state == SKHPB_REGION_ACTIVE) {
cb->region_state = SKHPB_REGION_INACTIVE;
skhpb_destroy_subregion_tbl(hpb, cb);
}
}
vfree(hpb->region_tbl);
}
void skhpb_suspend(struct ufs_hba *hba)
{
int lun;
unsigned long flags;
struct skhpb_lu *hpb;
struct skhpb_rsp_info *rsp_info;
struct skhpb_map_req *map_req;
if (hba->skhpb_quirk & SKHPB_QUIRK_PURGE_HINT_INFO_WHEN_SLEEP) {
for (lun = 0 ; lun < UFS_UPIU_MAX_GENERAL_LUN ; lun++) {
hpb = hba->skhpb_lup[lun];
if (!hpb)
continue;
while (1) {
spin_lock_irqsave(&hpb->rsp_list_lock, flags);
rsp_info = list_first_entry_or_null(&hpb->lh_rsp_info,
struct skhpb_rsp_info, list_rsp_info);
if (!rsp_info) {
spin_unlock_irqrestore(&hpb->rsp_list_lock, flags);
break;
}
list_del_init(&rsp_info->list_rsp_info);
list_add_tail(&rsp_info->list_rsp_info, &hpb->lh_rsp_info_free);
atomic64_inc(&hpb->canceled_resp);
spin_unlock_irqrestore(&hpb->rsp_list_lock, flags);
}
while (1) {
spin_lock_irqsave(&hpb->map_list_lock, flags);
map_req = list_first_entry_or_null(&hpb->lh_map_req_retry,
struct skhpb_map_req, list_map_req);
if (!map_req) {
spin_unlock_irqrestore(&hpb->map_list_lock, flags);
break;
}
list_del_init(&map_req->list_map_req);
list_add_tail(&map_req->list_map_req, &hpb->lh_map_req_free);
atomic64_inc(&hpb->canceled_map_req);
spin_unlock_irqrestore(&hpb->map_list_lock, flags);
}
}
}
for (lun = 0 ; lun < UFS_UPIU_MAX_GENERAL_LUN ; lun++) {
hpb = hba->skhpb_lup[lun];
if (!hpb)
continue;
skhpb_cancel_jobs(hpb);
}
}
void skhpb_resume(struct ufs_hba *hba)
{
int lun;
struct skhpb_lu *hpb;
unsigned long flags;
for (lun = 0 ; lun < UFS_UPIU_MAX_GENERAL_LUN ; lun++) {
hpb = hba->skhpb_lup[lun];
if (!hpb)
continue;
spin_lock_irqsave(&hpb->map_list_lock, flags);
if (!list_empty(&hpb->lh_map_req_retry))
schedule_delayed_work(&hpb->skhpb_map_req_retry_work, msecs_to_jiffies(1000));
spin_unlock_irqrestore(&hpb->map_list_lock, flags);
}
}
void skhpb_release(struct ufs_hba *hba, int state)
{
int lun;
hba->skhpb_state = SKHPB_FAILED;
for (lun = 0 ; lun < UFS_UPIU_MAX_GENERAL_LUN ; lun++) {
struct skhpb_lu *hpb = hba->skhpb_lup[lun];
if (!hpb)
continue;
hba->skhpb_lup[lun] = NULL;
if (!hpb->lu_hpb_enable)
continue;
hpb->lu_hpb_enable = false;
skhpb_cancel_jobs(hpb);
skhpb_destroy_region_tbl(hpb);
skhpb_table_mempool_remove(hpb);
vfree(hpb->rsp_info);
vfree(hpb->map_req);
kobject_uevent(&hpb->kobj, KOBJ_REMOVE);
kobject_del(&hpb->kobj); // TODO --count & del?
kfree(hpb);
}
if (skhpb_alloc_mctx != 0)
SKHPB_DRIVER_E("warning: skhpb_alloc_mctx %d", skhpb_alloc_mctx);
hba->skhpb_state = state;
}
void skhpb_init_handler(struct work_struct *work)
{
struct delayed_work *dwork = to_delayed_work(work);
struct ufs_hba *hba =
container_of(dwork, struct ufs_hba, skhpb_init_work);
struct Scsi_Host *shost = hba->host;
bool async_scan;
if (hba->skhpb_state == SKHPB_NOT_SUPPORTED)
return;
spin_lock_irq(shost->host_lock);
async_scan = shost->async_scan;
spin_unlock_irq(shost->host_lock);
if (async_scan) {
schedule_delayed_work(dwork, msecs_to_jiffies(100));
return;
}
if (hba->skhpb_state == SKHPB_NEED_INIT) {
int err = skhpb_init(hba);
if (hba->skhpb_state == SKHPB_NOT_SUPPORTED)
SKHPB_DRIVER_E("Run without HPB - err=%d\n", err);
} else if (hba->skhpb_state == SKHPB_RESET) {
skhpb_probe(hba);
}
}
void ufshcd_init_hpb(struct ufs_hba *hba)
{
int lun;
hba->skhpb_feat = 0;
hba->skhpb_quirk = 0;
hba->skhpb_state = SKHPB_NEED_INIT;
for (lun = 0 ; lun < UFS_UPIU_MAX_GENERAL_LUN ; lun++) {
hba->skhpb_lup[lun] = NULL;
hba->sdev_ufs_lu[lun] = NULL;
hba->skhpb_quicklist_lu_enable[lun] = SKHPB_U8_MAX;
}
INIT_DELAYED_WORK(&hba->skhpb_init_work, skhpb_init_handler);
}
static void skhpb_error_handler(struct work_struct *work)
{
struct ufs_hba *hba;
hba = container_of(work, struct ufs_hba, skhpb_eh_work);
SKHPB_DRIVER_E("SKHPB driver runs without SKHPB\n");
SKHPB_DRIVER_E("SKHPB will be removed from the kernel\n");
skhpb_release(hba, SKHPB_FAILED);
}
static void skhpb_stat_init(struct skhpb_lu *hpb)
{
atomic64_set(&hpb->hit, 0);
atomic64_set(&hpb->size_miss, 0);
atomic64_set(&hpb->region_miss, 0);
atomic64_set(&hpb->subregion_miss, 0);
atomic64_set(&hpb->entry_dirty_miss, 0);
atomic64_set(&hpb->w_map_req_cnt, 0);
#if defined(SKHPB_READ_LARGE_CHUNK_SUPPORT)
atomic64_set(&hpb->lc_entry_dirty_miss, 0);
atomic64_set(&hpb->lc_reg_subreg_miss, 0);
atomic64_set(&hpb->lc_hit, 0);
#endif
atomic64_set(&hpb->rb_noti_cnt, 0);
atomic64_set(&hpb->reset_noti_cnt, 0);
atomic64_set(&hpb->map_req_cnt, 0);
atomic64_set(&hpb->region_evict, 0);
atomic64_set(&hpb->region_add, 0);
atomic64_set(&hpb->rb_fail, 0);
atomic64_set(&hpb->canceled_resp, 0);
atomic64_set(&hpb->canceled_map_req, 0);
atomic64_set(&hpb->alloc_map_req_cnt, 0);
}
static ssize_t skhpb_sysfs_info_from_region_store(struct skhpb_lu *hpb,
const char *buf, size_t count)
{
unsigned long long value;
int region, subregion;
struct skhpb_region *cb;
struct skhpb_subregion *cp;
if (kstrtoull(buf, 0, &value)) {
SKHPB_DRIVER_E("kstrtoul error\n");
return -EINVAL;
}
if (value >= hpb->regions_per_lu) {
SKHPB_DRIVER_E("value %llu >= regions_per_lu %d error\n",
value, hpb->regions_per_lu);
return -EINVAL;
}
region = (int)value;
cb = hpb->region_tbl + region;
SKHPB_DRIVER_I("get_info_from_region[%d]=", region);
SKHPB_DRIVER_I("region %u state %s", region,
((cb->region_state == SKHPB_REGION_INACTIVE) ? "INACTIVE" :
((cb->region_state == SKHPB_REGION_ACTIVE) ? "ACTIVE" : "INVALID"))
);
for (subregion = 0; subregion < cb->subregion_count; subregion++) {
cp = cb->subregion_tbl + subregion;
SKHPB_DRIVER_I("subregion %u state %s", subregion,
((cp->subregion_state == SKHPB_SUBREGION_UNUSED) ? "UNUSED" :
((cp->subregion_state == SKHPB_SUBREGION_DIRTY) ? "DIRTY" :
((cp->subregion_state == SKHPB_SUBREGION_CLEAN) ? "CLEAN" :
(cp->subregion_state == SKHPB_SUBREGION_ISSUED) ?
"ISSUED" : "INVALID")))
);
}
return count;
}
static ssize_t skhpb_sysfs_info_from_lba_store(struct skhpb_lu *hpb,
const char *buf, size_t count)
{
skhpb_t ppn;
unsigned long long value;
unsigned int lpn;
int region, subregion, subregion_offset;
struct skhpb_region *cb;
struct skhpb_subregion *cp;
unsigned long flags;
int dirty;
if (kstrtoull(buf, 0, &value)) {
SKHPB_DRIVER_E("kstrtoul error\n");
return -EINVAL;
}
if (value > hpb->lu_num_blocks * SKHPB_SECTORS_PER_BLOCK) {
SKHPB_DRIVER_E("value %llu > lu_num_blocks %llu error\n",
value, hpb->lu_num_blocks);
return -EINVAL;
}
lpn = value / SKHPB_SECTORS_PER_BLOCK;
skhpb_get_pos_from_lpn(hpb, lpn, &region, &subregion,
&subregion_offset);
if (!skhpb_check_region_subregion_validity(hpb, region, subregion))
return -EINVAL;
cb = hpb->region_tbl + region;
cp = cb->subregion_tbl + subregion;
if (cb->region_state != SKHPB_REGION_INACTIVE) {
ppn = skhpb_get_ppn(cp->mctx, subregion_offset);
spin_lock_irqsave(&hpb->hpb_lock, flags);
dirty = skhpb_ppn_dirty_check(hpb, cp, subregion_offset);
spin_unlock_irqrestore(&hpb->hpb_lock, flags);
} else {
ppn = 0;
dirty = -1;
}
SKHPB_DRIVER_I("get_info_from_lba[%llu]=", value);
SKHPB_DRIVER_I("sector %llu region %d state %s subregion %d state %s",
value, region,
((cb->region_state == SKHPB_REGION_INACTIVE) ? "INACTIVE" :
((cb->region_state == SKHPB_REGION_ACTIVE) ? "ACTIVE" : "INVALID")),
subregion,
((cp->subregion_state == SKHPB_SUBREGION_UNUSED) ? "UNUSED" :
((cp->subregion_state == SKHPB_SUBREGION_DIRTY) ? "DIRTY" :
((cp->subregion_state == SKHPB_SUBREGION_CLEAN) ? "CLEAN" :
(cp->subregion_state == SKHPB_SUBREGION_ISSUED) ?
"ISSUED":"INVALID")))
);
SKHPB_DRIVER_I("sector %llu lpn %u ppn %llx dirty %d",
value, lpn, ppn, dirty);
return count;
}
static ssize_t skhpb_sysfs_map_req_show(struct skhpb_lu *hpb, char *buf)
{
return snprintf(buf, PAGE_SIZE,
"map_req_count[RB_NOTI RESET_NOTI MAP_REQ]= %lld %lld %lld\n",
(long long)atomic64_read(&hpb->rb_noti_cnt),
(long long)atomic64_read(&hpb->reset_noti_cnt),
(long long)atomic64_read(&hpb->map_req_cnt));
}
static ssize_t skhpb_sysfs_count_reset_store(struct skhpb_lu *hpb,
const char *buf, size_t count)
{
unsigned long debug;
if (kstrtoul(buf, 0, &debug))
return -EINVAL;
skhpb_stat_init(hpb);
return count;
}
static ssize_t skhpb_sysfs_add_evict_show(struct skhpb_lu *hpb, char *buf)
{
return snprintf(buf, PAGE_SIZE, "add_evict_count[ADD EVICT]= %lld %lld\n",
(long long)atomic64_read(&hpb->region_add),
(long long)atomic64_read(&hpb->region_evict));
}
static ssize_t skhpb_sysfs_statistics_show(struct skhpb_lu *hpb, char *buf)
{
long long size_miss, region_miss, subregion_miss, entry_dirty_miss,
hit, miss_all, rb_fail, canceled_resp, canceled_map_req;
#if defined(SKHPB_READ_LARGE_CHUNK_SUPPORT)
long long lc_dirty_miss = 0, lc_state_miss = 0, lc_hit = 0;
#endif
int count = 0;
hit = atomic64_read(&hpb->hit);
size_miss = atomic64_read(&hpb->size_miss);
region_miss = atomic64_read(&hpb->region_miss);
subregion_miss = atomic64_read(&hpb->subregion_miss);
entry_dirty_miss = atomic64_read(&hpb->entry_dirty_miss);
rb_fail = atomic64_read(&hpb->rb_fail);
canceled_resp = atomic64_read(&hpb->canceled_resp);
canceled_map_req = atomic64_read(&hpb->canceled_map_req);
#if defined(SKHPB_READ_LARGE_CHUNK_SUPPORT)
lc_dirty_miss = atomic64_read(&hpb->lc_entry_dirty_miss);
lc_state_miss = atomic64_read(&hpb->lc_reg_subreg_miss);
lc_hit = atomic64_read(&hpb->lc_hit);
entry_dirty_miss += lc_dirty_miss;
subregion_miss += lc_state_miss;
#endif
miss_all = size_miss + region_miss + subregion_miss + entry_dirty_miss;
count += snprintf(buf + count, PAGE_SIZE,
"Total: %lld\nHit_Counts: %lld\n",
hit + miss_all, hit);
count += snprintf(buf + count, PAGE_SIZE,
"Miss_Counts[ALL SIZE REG SUBREG DIRTY]= %lld %lld %lld %lld %lld\n",
miss_all, size_miss, region_miss, subregion_miss, entry_dirty_miss);
#if defined(SKHPB_READ_LARGE_CHUNK_SUPPORT)
count += snprintf(buf + count, PAGE_SIZE,
"LARG_CHNK Hit_Count: %lld, miss_count[ALL DIRTY REG_SUBREG]= %lld %lld %lld\n",
lc_hit, lc_dirty_miss + lc_state_miss, lc_dirty_miss, lc_state_miss);
#endif
if (hpb->hba->skhpb_quirk & SKHPB_QUIRK_PURGE_HINT_INFO_WHEN_SLEEP)
count += snprintf(buf + count, PAGE_SIZE,
"READ_BUFFER miss_count[RB_FAIL CANCEL_RESP CANCEL_MAP]= %lld %lld %lld\n",
rb_fail, canceled_resp, canceled_map_req);
else
count += snprintf(buf + count, PAGE_SIZE,
"READ_BUFFER miss_count[RB_FAIL]= %lld\n", rb_fail);
return count;
}
static ssize_t skhpb_sysfs_version_show(struct skhpb_lu *hpb, char *buf)
{
return snprintf(buf, PAGE_SIZE,
"HPBversion[HPB DD]= %x.%x.%x %x.%x.%x\n",
(hpb->hpb_ver >> 8) & 0xf,
(hpb->hpb_ver >> 4) & 0xf,
(hpb->hpb_ver >> 0) & 0xf,
SKHPB_GET_BYTE_2(SKHPB_DD_VER),
SKHPB_GET_BYTE_1(SKHPB_DD_VER),
SKHPB_GET_BYTE_0(SKHPB_DD_VER));
}
static ssize_t skhpb_sysfs_active_count_show(
struct skhpb_lu *hpb, char *buf)
{
struct skhpb_region *cb;
int region;
int pinned_cnt = 0;
for (region = 0 ; region < hpb->regions_per_lu ; region++) {
cb = hpb->region_tbl + region;
if (cb->is_pinned && cb->region_state == SKHPB_REGION_ACTIVE)
pinned_cnt++;
}
return snprintf(buf, PAGE_SIZE,
"active_count[ACTIVE_CNT PINNED_CNT]= %ld %d\n",
atomic64_read(&hpb->lru_info.active_count), pinned_cnt);
}
static ssize_t skhpb_sysfs_hpb_disable_show(
struct skhpb_lu *hpb, char *buf)
{
return snprintf(buf, PAGE_SIZE,
"[0] BOTH_enabled= %d\
\n[1] ONLY_HPB_BUFFER_disabled= %d\
\n[2] ONLY_HPB_READ_disabled= %d\
\n[3] BOTH_disabled= %d\n",
(hpb->force_map_req_disable == 0 &&
hpb->force_hpb_read_disable == 0)?1:0,
hpb->force_map_req_disable,
hpb->force_hpb_read_disable,
(hpb->force_map_req_disable == 1 &&
hpb->force_hpb_read_disable == 1)?1:0);
}
static ssize_t skhpb_sysfs_hpb_disable_store(struct skhpb_lu *hpb,
const char *buf, size_t count)
{
unsigned long value;
if (kstrtoul(buf, 0, &value))
return -EINVAL;
if (value > 3) {
SKHPB_DRIVER_E("Error, Only [0-3] is valid:\
\n[0] BOTH_enable\
\n[1] ONLY_HPB_BUFFER_disable\
\n[2] ONLY_HPB_READ_disable\
\n[3] BOTH_disabie\n");
return -EINVAL;
}
hpb->force_map_req_disable = value & 0x1;
hpb->force_hpb_read_disable = value & 0x2;
return count;
}
static ssize_t skhpb_sysfs_hpb_reset_store(struct skhpb_lu *hpb,
const char *buf, size_t count)
{
unsigned long value;
unsigned int doorbel;
struct skhpb_rsp_info *rsp_info;
unsigned long flags;
int ret, retries, lun;
struct skhpb_lu *hpb_lu;
if (kstrtoul(buf, 0, &value))
return -EINVAL;
if (!value)
return count;
for (retries = 1; retries <= 20; retries++) {
pm_runtime_get_sync(SKHPB_DEV(hpb));
ufshcd_hold(hpb->hba, false);
doorbel = ufshcd_readl(hpb->hba, REG_UTP_TRANSFER_REQ_DOOR_BELL);
ufshcd_release(hpb->hba);
pm_runtime_mark_last_busy(SKHPB_DEV(hpb));
pm_runtime_put_noidle(SKHPB_DEV(hpb));
if (!doorbel) {
pm_runtime_get_sync(SKHPB_DEV(hpb));
ret = ufshcd_query_flag_retry(hpb->hba,
UPIU_QUERY_OPCODE_SET_FLAG,
QUERY_FLAG_IDN_HPB_RESET,
NULL);
pm_runtime_mark_last_busy(SKHPB_DEV(hpb));
pm_runtime_put_noidle(SKHPB_DEV(hpb));
if (!ret) {
SKHPB_DRIVER_I("Query fHPBReset is successfully sent\n");
break;
}
}
SKHPB_DRIVER_I("fHPBReset failed and will retry[%d] after some time, DOORBELL: 0x%x\n",
retries, doorbel);
msleep(200);
}
if (ret == 0) {
bool fHPBReset = true;
pm_runtime_get_sync(SKHPB_DEV(hpb));
ufshcd_query_flag_retry(hpb->hba,
UPIU_QUERY_OPCODE_READ_FLAG,
QUERY_FLAG_IDN_HPB_RESET,
&fHPBReset);
pm_runtime_mark_last_busy(SKHPB_DEV(hpb));
pm_runtime_put_noidle(SKHPB_DEV(hpb));
if (fHPBReset)
SKHPB_DRIVER_E("fHPBReset is still in progress at device, but keep going\n");
} else {
SKHPB_DRIVER_E("Fail to set fHPBReset flag\n");
goto out;
}
flush_delayed_work(&hpb->skhpb_pinned_work);
for (lun = 0 ; lun < UFS_UPIU_MAX_GENERAL_LUN ; lun++) {
hpb_lu = hpb->hba->skhpb_lup[lun];
if (!hpb_lu || !hpb_lu->lu_hpb_enable)
continue;
skhpb_stat_init(hpb_lu);
spin_lock_irqsave(&hpb_lu->rsp_list_lock, flags);
rsp_info = skhpb_get_req_info(hpb_lu);
spin_unlock_irqrestore(&hpb_lu->rsp_list_lock, flags);
if (!rsp_info)
goto out;
rsp_info->type = SKHPB_RSP_HPB_RESET;
SKHPB_RSP_TIME(rsp_info->RSP_start);
spin_lock_irqsave(&hpb_lu->rsp_list_lock, flags);
list_add_tail(&rsp_info->list_rsp_info, &hpb_lu->lh_rsp_info);
spin_unlock_irqrestore(&hpb_lu->rsp_list_lock, flags);
SKHPB_DRIVER_I("Host HPB reset start LU%d - fHPBReset\n", lun);
schedule_work(&hpb_lu->skhpb_rsp_work);
}
out:
return count;
}
static ssize_t skhpb_sysfs_debug_log_show(
struct skhpb_lu *hpb, char *buf)
{
int value = skhpb_debug_mask;
if (value == (SKHPB_LOG_OFF)) {
value = SKHPB_LOG_LEVEL_OFF;
} else if (value == (SKHPB_LOG_ERR)) {
value = SKHPB_LOG_LEVEL_ERR;
} else if (value == (SKHPB_LOG_ERR | SKHPB_LOG_INFO)) {
value = SKHPB_LOG_LEVEL_INFO;
} else if (value == (SKHPB_LOG_ERR | SKHPB_LOG_INFO | SKHPB_LOG_DEBUG)) {
value = SKHPB_LOG_LEVEL_DEBUG;
} else if (value == (SKHPB_LOG_ERR | SKHPB_LOG_INFO | SKHPB_LOG_DEBUG | SKHPB_LOG_HEX)) {
value = SKHPB_LOG_LEVEL_HEX;
}
return snprintf(buf, PAGE_SIZE, "[0] : LOG_LEVEL_OFF\
\n[1] : LOG_LEVEL_ERR\
\n[2] : LOG_LEVEL_INFO\
\n[3] : LOG_LEVEL_DEBUG\
\n[4] : LOG_LEVEL_HEX\
\n-----------------------\
\nLog-Level = %d\n", value);
}
static ssize_t skhpb_sysfs_debug_log_store(struct skhpb_lu *hpb,
const char *buf, size_t count)
{
unsigned long value;
if (kstrtoul(buf, 0, &value))
return -EINVAL;
if (value > SKHPB_LOG_LEVEL_HEX) {
SKHPB_DRIVER_E("Error, Only [0-4] is valid:\
\n[0] : LOG_LEVEL_OFF\
\n[1] : LOG_LEVEL_ERR\
\n[2] : LOG_LEVEL_INFO\
\n[3] : LOG_LEVEL_DEBUG\
\n[4] : LOG_LEVEL_HEX\n");
return -EINVAL;
}
if (value == SKHPB_LOG_LEVEL_OFF) {
skhpb_debug_mask = SKHPB_LOG_OFF;
} else if (value == SKHPB_LOG_LEVEL_ERR) {
skhpb_debug_mask = SKHPB_LOG_ERR;
} else if (value == SKHPB_LOG_LEVEL_INFO) {
skhpb_debug_mask = SKHPB_LOG_ERR | SKHPB_LOG_INFO;
} else if (value == SKHPB_LOG_LEVEL_DEBUG) {
skhpb_debug_mask = SKHPB_LOG_ERR | SKHPB_LOG_INFO | SKHPB_LOG_DEBUG;
} else if (value == SKHPB_LOG_LEVEL_HEX) {
skhpb_debug_mask = SKHPB_LOG_ERR | SKHPB_LOG_INFO | SKHPB_LOG_DEBUG | SKHPB_LOG_HEX;
}
return count;
}
static ssize_t skhpb_sysfs_rsp_time_show(struct skhpb_lu *hpb, char *buf)
{
return snprintf(buf, PAGE_SIZE,
"map_req_time: %s\n", (debug_map_req ? "Enable":"Disable"));
}
static ssize_t skhpb_sysfs_rsp_time_store(struct skhpb_lu *hpb,
const char *buf, size_t count)
{
unsigned long value;
if (kstrtoul(buf, 0, &value))
return -EINVAL;
if (value > 1)
return count;
debug_map_req = value;
return count;
}
static struct skhpb_sysfs_entry skhpb_sysfs_entries[] = {
__ATTR(hpb_disable, 0644,
skhpb_sysfs_hpb_disable_show,
skhpb_sysfs_hpb_disable_store),
__ATTR(HPBVersion, 0444, skhpb_sysfs_version_show, NULL),
__ATTR(statistics, 0444, skhpb_sysfs_statistics_show, NULL),
__ATTR(active_count, 0444, skhpb_sysfs_active_count_show, NULL),
__ATTR(add_evict_count, 0444, skhpb_sysfs_add_evict_show, NULL),
__ATTR(count_reset, 0200, NULL, skhpb_sysfs_count_reset_store),
__ATTR(map_req_count, 0444, skhpb_sysfs_map_req_show, NULL),
__ATTR(get_info_from_region, 0200, NULL,
skhpb_sysfs_info_from_region_store),
__ATTR(get_info_from_lba, 0200, NULL, skhpb_sysfs_info_from_lba_store),
__ATTR(hpb_reset, 0200, NULL, skhpb_sysfs_hpb_reset_store),
__ATTR(debug_log, 0644,
skhpb_sysfs_debug_log_show,
skhpb_sysfs_debug_log_store),
__ATTR(response_time, 0644,
skhpb_sysfs_rsp_time_show,
skhpb_sysfs_rsp_time_store),
__ATTR_NULL
};
static ssize_t skhpb_attr_show(struct kobject *kobj,
struct attribute *attr, char *page)
{
struct skhpb_sysfs_entry *entry;
struct skhpb_lu *hpb;
ssize_t error;
entry = container_of(attr,
struct skhpb_sysfs_entry, attr);
hpb = container_of(kobj, struct skhpb_lu, kobj);
if (!entry->show)
return -EIO;
mutex_lock(&hpb->sysfs_lock);
error = entry->show(hpb, page);
mutex_unlock(&hpb->sysfs_lock);
return error;
}
static ssize_t skhpb_attr_store(struct kobject *kobj,
struct attribute *attr,
const char *page, size_t length)
{
struct skhpb_sysfs_entry *entry;
struct skhpb_lu *hpb;
ssize_t error;
entry = container_of(attr, struct skhpb_sysfs_entry, attr);
hpb = container_of(kobj, struct skhpb_lu, kobj);
if (!entry->store)
return -EIO;
mutex_lock(&hpb->sysfs_lock);
error = entry->store(hpb, page, length);
mutex_unlock(&hpb->sysfs_lock);
return error;
}
static const struct sysfs_ops skhpb_sysfs_ops = {
.show = skhpb_attr_show,
.store = skhpb_attr_store,
};
static struct kobj_type skhpb_ktype = {
.sysfs_ops = &skhpb_sysfs_ops,
.release = NULL,
};
static int skhpb_create_sysfs(struct ufs_hba *hba,
struct skhpb_lu *hpb)
{
struct device *dev = hba->dev;
struct skhpb_sysfs_entry *entry;
int err;
hpb->sysfs_entries = skhpb_sysfs_entries;
skhpb_stat_init(hpb);
kobject_init(&hpb->kobj, &skhpb_ktype);
mutex_init(&hpb->sysfs_lock);
err = kobject_add(&hpb->kobj, kobject_get(&dev->kobj),
"ufshpb_lu%d", hpb->lun);
if (!err) {
for (entry = hpb->sysfs_entries;
entry->attr.name != NULL ; entry++) {
if (sysfs_create_file(&hpb->kobj, &entry->attr))
break;
}
kobject_uevent(&hpb->kobj, KOBJ_ADD);
}
return err;
}