Commit ae2294b1 authored by Boris Brezillon's avatar Boris Brezillon Committed by Miquel Raynal

mtd: rawnand: Pass the CS line to be selected in struct nand_operation

In order to deprecate the ->select_chip hook we need to pass the CS
line a NAND operations are targeting. This is done through the
addition of a cs field to the nand_operation struct.

We also need to keep track of the currently selected target to
properly initialize op->cs, hence the ->cur_cs field addition to the
nand_chip struct.

Note that op->cs is not assigned in nand_exec_op() because we might
rework the way we execute NAND operations in the future (adopt a
queuing mechanism instead of the serialization we have right now).
Signed-off-by: default avatarBoris Brezillon <boris.brezillon@bootlin.com>
Tested-by: default avatarJanusz Krzysztofik <jmkrzyszt@gmail.com>
Signed-off-by: default avatarMiquel Raynal <miquel.raynal@bootlin.com>
parent 1d017859
...@@ -101,6 +101,9 @@ static inline int nand_exec_op(struct nand_chip *chip, ...@@ -101,6 +101,9 @@ static inline int nand_exec_op(struct nand_chip *chip,
if (!chip->exec_op) if (!chip->exec_op)
return -ENOTSUPP; return -ENOTSUPP;
if (WARN_ON(op->cs >= chip->numchips))
return -EINVAL;
return chip->exec_op(chip, op, false); return chip->exec_op(chip, op, false);
} }
......
...@@ -246,6 +246,7 @@ void nand_select_target(struct nand_chip *chip, unsigned int cs) ...@@ -246,6 +246,7 @@ void nand_select_target(struct nand_chip *chip, unsigned int cs)
if (WARN_ON(cs > chip->numchips)) if (WARN_ON(cs > chip->numchips))
return; return;
chip->cur_cs = cs;
chip->select_chip(chip, cs); chip->select_chip(chip, cs);
} }
EXPORT_SYMBOL_GPL(nand_select_target); EXPORT_SYMBOL_GPL(nand_select_target);
...@@ -260,6 +261,7 @@ EXPORT_SYMBOL_GPL(nand_select_target); ...@@ -260,6 +261,7 @@ EXPORT_SYMBOL_GPL(nand_select_target);
void nand_deselect_target(struct nand_chip *chip) void nand_deselect_target(struct nand_chip *chip)
{ {
chip->select_chip(chip, -1); chip->select_chip(chip, -1);
chip->cur_cs = -1;
} }
EXPORT_SYMBOL_GPL(nand_deselect_target); EXPORT_SYMBOL_GPL(nand_deselect_target);
...@@ -1022,7 +1024,7 @@ static int nand_sp_exec_read_page_op(struct nand_chip *chip, unsigned int page, ...@@ -1022,7 +1024,7 @@ static int nand_sp_exec_read_page_op(struct nand_chip *chip, unsigned int page,
PSEC_TO_NSEC(sdr->tRR_min)), PSEC_TO_NSEC(sdr->tRR_min)),
NAND_OP_DATA_IN(len, buf, 0), NAND_OP_DATA_IN(len, buf, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
int ret; int ret;
/* Drop the DATA_IN instruction if len is set to 0. */ /* Drop the DATA_IN instruction if len is set to 0. */
...@@ -1065,7 +1067,7 @@ static int nand_lp_exec_read_page_op(struct nand_chip *chip, unsigned int page, ...@@ -1065,7 +1067,7 @@ static int nand_lp_exec_read_page_op(struct nand_chip *chip, unsigned int page,
PSEC_TO_NSEC(sdr->tRR_min)), PSEC_TO_NSEC(sdr->tRR_min)),
NAND_OP_DATA_IN(len, buf, 0), NAND_OP_DATA_IN(len, buf, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
int ret; int ret;
/* Drop the DATA_IN instruction if len is set to 0. */ /* Drop the DATA_IN instruction if len is set to 0. */
...@@ -1160,7 +1162,7 @@ int nand_read_param_page_op(struct nand_chip *chip, u8 page, void *buf, ...@@ -1160,7 +1162,7 @@ int nand_read_param_page_op(struct nand_chip *chip, u8 page, void *buf,
PSEC_TO_NSEC(sdr->tRR_min)), PSEC_TO_NSEC(sdr->tRR_min)),
NAND_OP_8BIT_DATA_IN(len, buf, 0), NAND_OP_8BIT_DATA_IN(len, buf, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
/* Drop the DATA_IN instruction if len is set to 0. */ /* Drop the DATA_IN instruction if len is set to 0. */
if (!len) if (!len)
...@@ -1216,7 +1218,7 @@ int nand_change_read_column_op(struct nand_chip *chip, ...@@ -1216,7 +1218,7 @@ int nand_change_read_column_op(struct nand_chip *chip,
PSEC_TO_NSEC(sdr->tCCS_min)), PSEC_TO_NSEC(sdr->tCCS_min)),
NAND_OP_DATA_IN(len, buf, 0), NAND_OP_DATA_IN(len, buf, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
int ret; int ret;
ret = nand_fill_column_cycles(chip, addrs, offset_in_page); ret = nand_fill_column_cycles(chip, addrs, offset_in_page);
...@@ -1298,7 +1300,7 @@ static int nand_exec_prog_page_op(struct nand_chip *chip, unsigned int page, ...@@ -1298,7 +1300,7 @@ static int nand_exec_prog_page_op(struct nand_chip *chip, unsigned int page,
NAND_OP_CMD(NAND_CMD_PAGEPROG, PSEC_TO_NSEC(sdr->tWB_max)), NAND_OP_CMD(NAND_CMD_PAGEPROG, PSEC_TO_NSEC(sdr->tWB_max)),
NAND_OP_WAIT_RDY(PSEC_TO_MSEC(sdr->tPROG_max), 0), NAND_OP_WAIT_RDY(PSEC_TO_MSEC(sdr->tPROG_max), 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
int naddrs = nand_fill_column_cycles(chip, addrs, offset_in_page); int naddrs = nand_fill_column_cycles(chip, addrs, offset_in_page);
int ret; int ret;
u8 status; u8 status;
...@@ -1412,7 +1414,7 @@ int nand_prog_page_end_op(struct nand_chip *chip) ...@@ -1412,7 +1414,7 @@ int nand_prog_page_end_op(struct nand_chip *chip)
PSEC_TO_NSEC(sdr->tWB_max)), PSEC_TO_NSEC(sdr->tWB_max)),
NAND_OP_WAIT_RDY(PSEC_TO_MSEC(sdr->tPROG_max), 0), NAND_OP_WAIT_RDY(PSEC_TO_MSEC(sdr->tPROG_max), 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
ret = nand_exec_op(chip, &op); ret = nand_exec_op(chip, &op);
if (ret) if (ret)
...@@ -1520,7 +1522,7 @@ int nand_change_write_column_op(struct nand_chip *chip, ...@@ -1520,7 +1522,7 @@ int nand_change_write_column_op(struct nand_chip *chip,
NAND_OP_ADDR(2, addrs, PSEC_TO_NSEC(sdr->tCCS_min)), NAND_OP_ADDR(2, addrs, PSEC_TO_NSEC(sdr->tCCS_min)),
NAND_OP_DATA_OUT(len, buf, 0), NAND_OP_DATA_OUT(len, buf, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
int ret; int ret;
ret = nand_fill_column_cycles(chip, addrs, offset_in_page); ret = nand_fill_column_cycles(chip, addrs, offset_in_page);
...@@ -1574,7 +1576,7 @@ int nand_readid_op(struct nand_chip *chip, u8 addr, void *buf, ...@@ -1574,7 +1576,7 @@ int nand_readid_op(struct nand_chip *chip, u8 addr, void *buf,
NAND_OP_ADDR(1, &addr, PSEC_TO_NSEC(sdr->tADL_min)), NAND_OP_ADDR(1, &addr, PSEC_TO_NSEC(sdr->tADL_min)),
NAND_OP_8BIT_DATA_IN(len, buf, 0), NAND_OP_8BIT_DATA_IN(len, buf, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
/* Drop the DATA_IN instruction if len is set to 0. */ /* Drop the DATA_IN instruction if len is set to 0. */
if (!len) if (!len)
...@@ -1613,7 +1615,7 @@ int nand_status_op(struct nand_chip *chip, u8 *status) ...@@ -1613,7 +1615,7 @@ int nand_status_op(struct nand_chip *chip, u8 *status)
PSEC_TO_NSEC(sdr->tADL_min)), PSEC_TO_NSEC(sdr->tADL_min)),
NAND_OP_8BIT_DATA_IN(1, status, 0), NAND_OP_8BIT_DATA_IN(1, status, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
if (!status) if (!status)
op.ninstrs--; op.ninstrs--;
...@@ -1646,7 +1648,7 @@ int nand_exit_status_op(struct nand_chip *chip) ...@@ -1646,7 +1648,7 @@ int nand_exit_status_op(struct nand_chip *chip)
struct nand_op_instr instrs[] = { struct nand_op_instr instrs[] = {
NAND_OP_CMD(NAND_CMD_READ0, 0), NAND_OP_CMD(NAND_CMD_READ0, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
return nand_exec_op(chip, &op); return nand_exec_op(chip, &op);
} }
...@@ -1685,7 +1687,7 @@ int nand_erase_op(struct nand_chip *chip, unsigned int eraseblock) ...@@ -1685,7 +1687,7 @@ int nand_erase_op(struct nand_chip *chip, unsigned int eraseblock)
PSEC_TO_MSEC(sdr->tWB_max)), PSEC_TO_MSEC(sdr->tWB_max)),
NAND_OP_WAIT_RDY(PSEC_TO_MSEC(sdr->tBERS_max), 0), NAND_OP_WAIT_RDY(PSEC_TO_MSEC(sdr->tBERS_max), 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
if (chip->options & NAND_ROW_ADDR_3) if (chip->options & NAND_ROW_ADDR_3)
instrs[1].ctx.addr.naddrs++; instrs[1].ctx.addr.naddrs++;
...@@ -1743,7 +1745,7 @@ static int nand_set_features_op(struct nand_chip *chip, u8 feature, ...@@ -1743,7 +1745,7 @@ static int nand_set_features_op(struct nand_chip *chip, u8 feature,
PSEC_TO_NSEC(sdr->tWB_max)), PSEC_TO_NSEC(sdr->tWB_max)),
NAND_OP_WAIT_RDY(PSEC_TO_MSEC(sdr->tFEAT_max), 0), NAND_OP_WAIT_RDY(PSEC_TO_MSEC(sdr->tFEAT_max), 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
return nand_exec_op(chip, &op); return nand_exec_op(chip, &op);
} }
...@@ -1791,7 +1793,7 @@ static int nand_get_features_op(struct nand_chip *chip, u8 feature, ...@@ -1791,7 +1793,7 @@ static int nand_get_features_op(struct nand_chip *chip, u8 feature,
NAND_OP_8BIT_DATA_IN(ONFI_SUBFEATURE_PARAM_LEN, NAND_OP_8BIT_DATA_IN(ONFI_SUBFEATURE_PARAM_LEN,
data, 0), data, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
return nand_exec_op(chip, &op); return nand_exec_op(chip, &op);
} }
...@@ -1811,7 +1813,7 @@ static int nand_wait_rdy_op(struct nand_chip *chip, unsigned int timeout_ms, ...@@ -1811,7 +1813,7 @@ static int nand_wait_rdy_op(struct nand_chip *chip, unsigned int timeout_ms,
NAND_OP_WAIT_RDY(PSEC_TO_MSEC(timeout_ms), NAND_OP_WAIT_RDY(PSEC_TO_MSEC(timeout_ms),
PSEC_TO_NSEC(delay_ns)), PSEC_TO_NSEC(delay_ns)),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
return nand_exec_op(chip, &op); return nand_exec_op(chip, &op);
} }
...@@ -1844,7 +1846,7 @@ int nand_reset_op(struct nand_chip *chip) ...@@ -1844,7 +1846,7 @@ int nand_reset_op(struct nand_chip *chip)
NAND_OP_CMD(NAND_CMD_RESET, PSEC_TO_NSEC(sdr->tWB_max)), NAND_OP_CMD(NAND_CMD_RESET, PSEC_TO_NSEC(sdr->tWB_max)),
NAND_OP_WAIT_RDY(PSEC_TO_MSEC(sdr->tRST_max), 0), NAND_OP_WAIT_RDY(PSEC_TO_MSEC(sdr->tRST_max), 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
return nand_exec_op(chip, &op); return nand_exec_op(chip, &op);
} }
...@@ -1878,7 +1880,7 @@ int nand_read_data_op(struct nand_chip *chip, void *buf, unsigned int len, ...@@ -1878,7 +1880,7 @@ int nand_read_data_op(struct nand_chip *chip, void *buf, unsigned int len,
struct nand_op_instr instrs[] = { struct nand_op_instr instrs[] = {
NAND_OP_DATA_IN(len, buf, 0), NAND_OP_DATA_IN(len, buf, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
instrs[0].ctx.data.force_8bit = force_8bit; instrs[0].ctx.data.force_8bit = force_8bit;
...@@ -1922,7 +1924,7 @@ int nand_write_data_op(struct nand_chip *chip, const void *buf, ...@@ -1922,7 +1924,7 @@ int nand_write_data_op(struct nand_chip *chip, const void *buf,
struct nand_op_instr instrs[] = { struct nand_op_instr instrs[] = {
NAND_OP_DATA_OUT(len, buf, 0), NAND_OP_DATA_OUT(len, buf, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
instrs[0].ctx.data.force_8bit = force_8bit; instrs[0].ctx.data.force_8bit = force_8bit;
...@@ -5006,6 +5008,9 @@ static int nand_scan_ident(struct nand_chip *chip, unsigned int maxchips, ...@@ -5006,6 +5008,9 @@ static int nand_scan_ident(struct nand_chip *chip, unsigned int maxchips,
unsigned int i; unsigned int i;
int ret; int ret;
/* Assume all dies are deselected when we enter nand_scan_ident(). */
chip->cur_cs = -1;
/* Enforce the right timings for reset/detection */ /* Enforce the right timings for reset/detection */
onfi_fill_data_interface(chip, NAND_SDR_IFACE, 0); onfi_fill_data_interface(chip, NAND_SDR_IFACE, 0);
......
...@@ -84,7 +84,7 @@ static int hynix_nand_cmd_op(struct nand_chip *chip, u8 cmd) ...@@ -84,7 +84,7 @@ static int hynix_nand_cmd_op(struct nand_chip *chip, u8 cmd)
struct nand_op_instr instrs[] = { struct nand_op_instr instrs[] = {
NAND_OP_CMD(cmd, 0), NAND_OP_CMD(cmd, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
return nand_exec_op(chip, &op); return nand_exec_op(chip, &op);
} }
...@@ -103,7 +103,7 @@ static int hynix_nand_reg_write_op(struct nand_chip *chip, u8 addr, u8 val) ...@@ -103,7 +103,7 @@ static int hynix_nand_reg_write_op(struct nand_chip *chip, u8 addr, u8 val)
NAND_OP_ADDR(1, &addr, 0), NAND_OP_ADDR(1, &addr, 0),
NAND_OP_8BIT_DATA_OUT(1, &val, 0), NAND_OP_8BIT_DATA_OUT(1, &val, 0),
}; };
struct nand_operation op = NAND_OPERATION(instrs); struct nand_operation op = NAND_OPERATION(chip->cur_cs, instrs);
return nand_exec_op(chip, &op); return nand_exec_op(chip, &op);
} }
......
...@@ -875,18 +875,21 @@ struct nand_op_parser { ...@@ -875,18 +875,21 @@ struct nand_op_parser {
/** /**
* struct nand_operation - NAND operation descriptor * struct nand_operation - NAND operation descriptor
* @cs: the CS line to select for this NAND operation
* @instrs: array of instructions to execute * @instrs: array of instructions to execute
* @ninstrs: length of the @instrs array * @ninstrs: length of the @instrs array
* *
* The actual operation structure that will be passed to chip->exec_op(). * The actual operation structure that will be passed to chip->exec_op().
*/ */
struct nand_operation { struct nand_operation {
unsigned int cs;
const struct nand_op_instr *instrs; const struct nand_op_instr *instrs;
unsigned int ninstrs; unsigned int ninstrs;
}; };
#define NAND_OPERATION(_instrs) \ #define NAND_OPERATION(_cs, _instrs) \
{ \ { \
.cs = _cs, \
.instrs = _instrs, \ .instrs = _instrs, \
.ninstrs = ARRAY_SIZE(_instrs), \ .ninstrs = ARRAY_SIZE(_instrs), \
} }
...@@ -1008,6 +1011,10 @@ struct nand_legacy { ...@@ -1008,6 +1011,10 @@ struct nand_legacy {
* this nand device will encounter their life times. * this nand device will encounter their life times.
* @blocks_per_die: [INTERN] The number of PEBs in a die * @blocks_per_die: [INTERN] The number of PEBs in a die
* @data_interface: [INTERN] NAND interface timing information * @data_interface: [INTERN] NAND interface timing information
* @cur_cs: currently selected target. -1 means no target selected,
* otherwise we should always have cur_cs >= 0 &&
* cur_cs < numchips. NAND Controller drivers should not
* modify this value, but they're allowed to read it.
* @read_retries: [INTERN] the number of read retry modes supported * @read_retries: [INTERN] the number of read retry modes supported
* @setup_data_interface: [OPTIONAL] setup the data interface and timing. If * @setup_data_interface: [OPTIONAL] setup the data interface and timing. If
* chipnr is set to %NAND_DATA_IFACE_CHECK_ONLY this * chipnr is set to %NAND_DATA_IFACE_CHECK_ONLY this
...@@ -1069,6 +1076,8 @@ struct nand_chip { ...@@ -1069,6 +1076,8 @@ struct nand_chip {
struct nand_data_interface data_interface; struct nand_data_interface data_interface;
int cur_cs;
int read_retries; int read_retries;
flstate_t state; flstate_t state;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment