Commit 7b1dd0f8 authored by Sakari Ailus's avatar Sakari Ailus Committed by Mauro Carvalho Chehab

media: ccs: Make hwcfg part of the device specific struct

There's no need to allocate the hardware configuration struct separately.
Put it in struct ccs_sensor.
Signed-off-by: default avatarSakari Ailus <sakari.ailus@linux.intel.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab+huawei@kernel.org>
parent 5323aaf1
...@@ -809,7 +809,7 @@ static int ccs_init_late_controls(struct ccs_sensor *sensor) ...@@ -809,7 +809,7 @@ static int ccs_init_late_controls(struct ccs_sensor *sensor)
sensor->link_freq = v4l2_ctrl_new_int_menu( sensor->link_freq = v4l2_ctrl_new_int_menu(
&sensor->src->ctrl_handler, &ccs_ctrl_ops, &sensor->src->ctrl_handler, &ccs_ctrl_ops,
V4L2_CID_LINK_FREQ, __fls(*valid_link_freqs), V4L2_CID_LINK_FREQ, __fls(*valid_link_freqs),
__ffs(*valid_link_freqs), sensor->hwcfg->op_sys_clock); __ffs(*valid_link_freqs), sensor->hwcfg.op_sys_clock);
return sensor->src->ctrl_handler.error; return sensor->src->ctrl_handler.error;
} }
...@@ -922,8 +922,8 @@ static int ccs_get_mbus_formats(struct ccs_sensor *sensor) ...@@ -922,8 +922,8 @@ static int ccs_get_mbus_formats(struct ccs_sensor *sensor)
pll->bits_per_pixel = f->compressed; pll->bits_per_pixel = f->compressed;
for (j = 0; sensor->hwcfg->op_sys_clock[j]; j++) { for (j = 0; sensor->hwcfg.op_sys_clock[j]; j++) {
pll->link_freq = sensor->hwcfg->op_sys_clock[j]; pll->link_freq = sensor->hwcfg.op_sys_clock[j];
rval = ccs_pll_try(sensor, pll); rval = ccs_pll_try(sensor, pll);
dev_dbg(&client->dev, "link freq %u Hz, bpp %u %s\n", dev_dbg(&client->dev, "link freq %u Hz, bpp %u %s\n",
...@@ -1123,21 +1123,21 @@ static int ccs_change_cci_addr(struct ccs_sensor *sensor) ...@@ -1123,21 +1123,21 @@ static int ccs_change_cci_addr(struct ccs_sensor *sensor)
int rval; int rval;
u32 val; u32 val;
client->addr = sensor->hwcfg->i2c_addr_dfl; client->addr = sensor->hwcfg.i2c_addr_dfl;
rval = ccs_write(sensor, CCI_ADDRESS_CTRL, rval = ccs_write(sensor, CCI_ADDRESS_CTRL,
sensor->hwcfg->i2c_addr_alt << 1); sensor->hwcfg.i2c_addr_alt << 1);
if (rval) if (rval)
return rval; return rval;
client->addr = sensor->hwcfg->i2c_addr_alt; client->addr = sensor->hwcfg.i2c_addr_alt;
/* verify addr change went ok */ /* verify addr change went ok */
rval = ccs_read(sensor, CCI_ADDRESS_CTRL, &val); rval = ccs_read(sensor, CCI_ADDRESS_CTRL, &val);
if (rval) if (rval)
return rval; return rval;
if (val != sensor->hwcfg->i2c_addr_alt << 1) if (val != sensor->hwcfg.i2c_addr_alt << 1)
return -ENODEV; return -ENODEV;
return 0; return 0;
...@@ -1151,13 +1151,13 @@ static int ccs_change_cci_addr(struct ccs_sensor *sensor) ...@@ -1151,13 +1151,13 @@ static int ccs_change_cci_addr(struct ccs_sensor *sensor)
static int ccs_setup_flash_strobe(struct ccs_sensor *sensor) static int ccs_setup_flash_strobe(struct ccs_sensor *sensor)
{ {
struct ccs_flash_strobe_parms *strobe_setup; struct ccs_flash_strobe_parms *strobe_setup;
unsigned int ext_freq = sensor->hwcfg->ext_clk; unsigned int ext_freq = sensor->hwcfg.ext_clk;
u32 tmp; u32 tmp;
u32 strobe_adjustment; u32 strobe_adjustment;
u32 strobe_width_high_rs; u32 strobe_width_high_rs;
int rval; int rval;
strobe_setup = sensor->hwcfg->strobe_setup; strobe_setup = sensor->hwcfg.strobe_setup;
/* /*
* How to calculate registers related to strobe length. Please * How to calculate registers related to strobe length. Please
...@@ -1265,7 +1265,7 @@ static int ccs_setup_flash_strobe(struct ccs_sensor *sensor) ...@@ -1265,7 +1265,7 @@ static int ccs_setup_flash_strobe(struct ccs_sensor *sensor)
rval = ccs_write(sensor, FLASH_TRIGGER_RS, strobe_setup->trigger); rval = ccs_write(sensor, FLASH_TRIGGER_RS, strobe_setup->trigger);
out: out:
sensor->hwcfg->strobe_setup->trigger = 0; sensor->hwcfg.strobe_setup->trigger = 0;
return rval; return rval;
} }
...@@ -1304,7 +1304,7 @@ static int ccs_power_on(struct device *dev) ...@@ -1304,7 +1304,7 @@ static int ccs_power_on(struct device *dev)
gpiod_set_value(sensor->reset, 0); gpiod_set_value(sensor->reset, 0);
gpiod_set_value(sensor->xshutdown, 1); gpiod_set_value(sensor->xshutdown, 1);
sleep = SMIAPP_RESET_DELAY(sensor->hwcfg->ext_clk); sleep = SMIAPP_RESET_DELAY(sensor->hwcfg.ext_clk);
usleep_range(sleep, sleep); usleep_range(sleep, sleep);
/* /*
...@@ -1318,7 +1318,7 @@ static int ccs_power_on(struct device *dev) ...@@ -1318,7 +1318,7 @@ static int ccs_power_on(struct device *dev)
* is found. * is found.
*/ */
if (sensor->hwcfg->i2c_addr_alt) { if (sensor->hwcfg.i2c_addr_alt) {
rval = ccs_change_cci_addr(sensor); rval = ccs_change_cci_addr(sensor);
if (rval) { if (rval) {
dev_err(dev, "cci address change error\n"); dev_err(dev, "cci address change error\n");
...@@ -1332,7 +1332,7 @@ static int ccs_power_on(struct device *dev) ...@@ -1332,7 +1332,7 @@ static int ccs_power_on(struct device *dev)
goto out_cci_addr_fail; goto out_cci_addr_fail;
} }
if (sensor->hwcfg->i2c_addr_alt) { if (sensor->hwcfg.i2c_addr_alt) {
rval = ccs_change_cci_addr(sensor); rval = ccs_change_cci_addr(sensor);
if (rval) { if (rval) {
dev_err(dev, "cci address change error\n"); dev_err(dev, "cci address change error\n");
...@@ -1348,13 +1348,13 @@ static int ccs_power_on(struct device *dev) ...@@ -1348,13 +1348,13 @@ static int ccs_power_on(struct device *dev)
} }
rval = ccs_write(sensor, EXTCLK_FREQUENCY_MHZ, rval = ccs_write(sensor, EXTCLK_FREQUENCY_MHZ,
sensor->hwcfg->ext_clk / (1000000 / (1 << 8))); sensor->hwcfg.ext_clk / (1000000 / (1 << 8)));
if (rval) { if (rval) {
dev_err(dev, "extclk frequency set failed\n"); dev_err(dev, "extclk frequency set failed\n");
goto out_cci_addr_fail; goto out_cci_addr_fail;
} }
rval = ccs_write(sensor, CSI_LANE_MODE, sensor->hwcfg->lanes - 1); rval = ccs_write(sensor, CSI_LANE_MODE, sensor->hwcfg.lanes - 1);
if (rval) { if (rval) {
dev_err(dev, "csi lane mode set failed\n"); dev_err(dev, "csi lane mode set failed\n");
goto out_cci_addr_fail; goto out_cci_addr_fail;
...@@ -1368,7 +1368,7 @@ static int ccs_power_on(struct device *dev) ...@@ -1368,7 +1368,7 @@ static int ccs_power_on(struct device *dev)
} }
rval = ccs_write(sensor, CSI_SIGNALING_MODE, rval = ccs_write(sensor, CSI_SIGNALING_MODE,
sensor->hwcfg->csi_signalling_mode); sensor->hwcfg.csi_signalling_mode);
if (rval) { if (rval) {
dev_err(dev, "csi signalling mode set failed\n"); dev_err(dev, "csi signalling mode set failed\n");
goto out_cci_addr_fail; goto out_cci_addr_fail;
...@@ -1412,7 +1412,7 @@ static int ccs_power_off(struct device *dev) ...@@ -1412,7 +1412,7 @@ static int ccs_power_off(struct device *dev)
* really see a power off and next time the cci address change * really see a power off and next time the cci address change
* will fail. So do a soft reset explicitly here. * will fail. So do a soft reset explicitly here.
*/ */
if (sensor->hwcfg->i2c_addr_alt) if (sensor->hwcfg.i2c_addr_alt)
ccs_write(sensor, SOFTWARE_RESET, CCS_SOFTWARE_RESET_ON); ccs_write(sensor, SOFTWARE_RESET, CCS_SOFTWARE_RESET_ON);
gpiod_set_value(sensor->reset, 1); gpiod_set_value(sensor->reset, 1);
...@@ -1551,8 +1551,8 @@ static int ccs_start_streaming(struct ccs_sensor *sensor) ...@@ -1551,8 +1551,8 @@ static int ccs_start_streaming(struct ccs_sensor *sensor)
if (CCS_LIM(sensor, FLASH_MODE_CAPABILITY) & if (CCS_LIM(sensor, FLASH_MODE_CAPABILITY) &
(CCS_FLASH_MODE_CAPABILITY_SINGLE_STROBE | (CCS_FLASH_MODE_CAPABILITY_SINGLE_STROBE |
SMIAPP_FLASH_MODE_CAPABILITY_MULTIPLE_STROBE) && SMIAPP_FLASH_MODE_CAPABILITY_MULTIPLE_STROBE) &&
sensor->hwcfg->strobe_setup != NULL && sensor->hwcfg.strobe_setup != NULL &&
sensor->hwcfg->strobe_setup->trigger != 0) { sensor->hwcfg.strobe_setup->trigger != 0) {
rval = ccs_setup_flash_strobe(sensor); rval = ccs_setup_flash_strobe(sensor);
if (rval) if (rval)
goto out; goto out;
...@@ -2850,9 +2850,9 @@ static int __maybe_unused ccs_resume(struct device *dev) ...@@ -2850,9 +2850,9 @@ static int __maybe_unused ccs_resume(struct device *dev)
return rval; return rval;
} }
static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev) static int ccs_get_hwconfig(struct ccs_sensor *sensor, struct device *dev)
{ {
struct ccs_hwconfig *hwcfg; struct ccs_hwconfig *hwcfg = &sensor->hwcfg;
struct v4l2_fwnode_endpoint bus_cfg = { .bus_type = 0 }; struct v4l2_fwnode_endpoint bus_cfg = { .bus_type = 0 };
struct fwnode_handle *ep; struct fwnode_handle *ep;
struct fwnode_handle *fwnode = dev_fwnode(dev); struct fwnode_handle *fwnode = dev_fwnode(dev);
...@@ -2862,7 +2862,7 @@ static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev) ...@@ -2862,7 +2862,7 @@ static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev)
ep = fwnode_graph_get_next_endpoint(fwnode, NULL); ep = fwnode_graph_get_next_endpoint(fwnode, NULL);
if (!ep) if (!ep)
return NULL; return -ENODEV;
bus_cfg.bus_type = V4L2_MBUS_CSI2_DPHY; bus_cfg.bus_type = V4L2_MBUS_CSI2_DPHY;
rval = v4l2_fwnode_endpoint_alloc_parse(ep, &bus_cfg); rval = v4l2_fwnode_endpoint_alloc_parse(ep, &bus_cfg);
...@@ -2874,10 +2874,6 @@ static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev) ...@@ -2874,10 +2874,6 @@ static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev)
if (rval) if (rval)
goto out_err; goto out_err;
hwcfg = devm_kzalloc(dev, sizeof(*hwcfg), GFP_KERNEL);
if (!hwcfg)
goto out_err;
switch (bus_cfg.bus_type) { switch (bus_cfg.bus_type) {
case V4L2_MBUS_CSI2_DPHY: case V4L2_MBUS_CSI2_DPHY:
hwcfg->csi_signalling_mode = CCS_CSI_SIGNALING_MODE_CSI_2_DPHY; hwcfg->csi_signalling_mode = CCS_CSI_SIGNALING_MODE_CSI_2_DPHY;
...@@ -2891,6 +2887,7 @@ static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev) ...@@ -2891,6 +2887,7 @@ static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev)
break; break;
default: default:
dev_err(dev, "unsupported bus %u\n", bus_cfg.bus_type); dev_err(dev, "unsupported bus %u\n", bus_cfg.bus_type);
rval = -EINVAL;
goto out_err; goto out_err;
} }
...@@ -2907,6 +2904,7 @@ static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev) ...@@ -2907,6 +2904,7 @@ static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev)
break; break;
default: default:
dev_err(dev, "invalid rotation %u\n", rotation); dev_err(dev, "invalid rotation %u\n", rotation);
rval = -EINVAL;
goto out_err; goto out_err;
} }
} }
...@@ -2921,14 +2919,17 @@ static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev) ...@@ -2921,14 +2919,17 @@ static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev)
if (!bus_cfg.nr_of_link_frequencies) { if (!bus_cfg.nr_of_link_frequencies) {
dev_warn(dev, "no link frequencies defined\n"); dev_warn(dev, "no link frequencies defined\n");
rval = -EINVAL;
goto out_err; goto out_err;
} }
hwcfg->op_sys_clock = devm_kcalloc( hwcfg->op_sys_clock = devm_kcalloc(
dev, bus_cfg.nr_of_link_frequencies + 1 /* guardian */, dev, bus_cfg.nr_of_link_frequencies + 1 /* guardian */,
sizeof(*hwcfg->op_sys_clock), GFP_KERNEL); sizeof(*hwcfg->op_sys_clock), GFP_KERNEL);
if (!hwcfg->op_sys_clock) if (!hwcfg->op_sys_clock) {
rval = -ENOMEM;
goto out_err; goto out_err;
}
for (i = 0; i < bus_cfg.nr_of_link_frequencies; i++) { for (i = 0; i < bus_cfg.nr_of_link_frequencies; i++) {
hwcfg->op_sys_clock[i] = bus_cfg.link_frequencies[i]; hwcfg->op_sys_clock[i] = bus_cfg.link_frequencies[i];
...@@ -2937,29 +2938,30 @@ static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev) ...@@ -2937,29 +2938,30 @@ static struct ccs_hwconfig *ccs_get_hwconfig(struct device *dev)
v4l2_fwnode_endpoint_free(&bus_cfg); v4l2_fwnode_endpoint_free(&bus_cfg);
fwnode_handle_put(ep); fwnode_handle_put(ep);
return hwcfg;
return 0;
out_err: out_err:
v4l2_fwnode_endpoint_free(&bus_cfg); v4l2_fwnode_endpoint_free(&bus_cfg);
fwnode_handle_put(ep); fwnode_handle_put(ep);
return NULL;
return rval;
} }
static int ccs_probe(struct i2c_client *client) static int ccs_probe(struct i2c_client *client)
{ {
struct ccs_sensor *sensor; struct ccs_sensor *sensor;
struct ccs_hwconfig *hwcfg = ccs_get_hwconfig(&client->dev);
unsigned int i; unsigned int i;
int rval; int rval;
if (hwcfg == NULL)
return -ENODEV;
sensor = devm_kzalloc(&client->dev, sizeof(*sensor), GFP_KERNEL); sensor = devm_kzalloc(&client->dev, sizeof(*sensor), GFP_KERNEL);
if (sensor == NULL) if (sensor == NULL)
return -ENOMEM; return -ENOMEM;
sensor->hwcfg = hwcfg; rval = ccs_get_hwconfig(sensor, &client->dev);
if (rval)
return rval;
sensor->src = &sensor->ssds[sensor->ssds_used]; sensor->src = &sensor->ssds[sensor->ssds_used];
v4l2_i2c_subdev_init(&sensor->src->sd, client, &ccs_ops); v4l2_i2c_subdev_init(&sensor->src->sd, client, &ccs_ops);
...@@ -2982,33 +2984,33 @@ static int ccs_probe(struct i2c_client *client) ...@@ -2982,33 +2984,33 @@ static int ccs_probe(struct i2c_client *client)
} }
if (sensor->ext_clk) { if (sensor->ext_clk) {
if (sensor->hwcfg->ext_clk) { if (sensor->hwcfg.ext_clk) {
unsigned long rate; unsigned long rate;
rval = clk_set_rate(sensor->ext_clk, rval = clk_set_rate(sensor->ext_clk,
sensor->hwcfg->ext_clk); sensor->hwcfg.ext_clk);
if (rval < 0) { if (rval < 0) {
dev_err(&client->dev, dev_err(&client->dev,
"unable to set clock freq to %u\n", "unable to set clock freq to %u\n",
sensor->hwcfg->ext_clk); sensor->hwcfg.ext_clk);
return rval; return rval;
} }
rate = clk_get_rate(sensor->ext_clk); rate = clk_get_rate(sensor->ext_clk);
if (rate != sensor->hwcfg->ext_clk) { if (rate != sensor->hwcfg.ext_clk) {
dev_err(&client->dev, dev_err(&client->dev,
"can't set clock freq, asked for %u but got %lu\n", "can't set clock freq, asked for %u but got %lu\n",
sensor->hwcfg->ext_clk, rate); sensor->hwcfg.ext_clk, rate);
return rval; return rval;
} }
} else { } else {
sensor->hwcfg->ext_clk = clk_get_rate(sensor->ext_clk); sensor->hwcfg.ext_clk = clk_get_rate(sensor->ext_clk);
dev_dbg(&client->dev, "obtained clock freq %u\n", dev_dbg(&client->dev, "obtained clock freq %u\n",
sensor->hwcfg->ext_clk); sensor->hwcfg.ext_clk);
} }
} else if (sensor->hwcfg->ext_clk) { } else if (sensor->hwcfg.ext_clk) {
dev_dbg(&client->dev, "assuming clock freq %u\n", dev_dbg(&client->dev, "assuming clock freq %u\n",
sensor->hwcfg->ext_clk); sensor->hwcfg.ext_clk);
} else { } else {
dev_err(&client->dev, "unable to obtain clock freq\n"); dev_err(&client->dev, "unable to obtain clock freq\n");
return -EINVAL; return -EINVAL;
...@@ -3061,7 +3063,7 @@ static int ccs_probe(struct i2c_client *client) ...@@ -3061,7 +3063,7 @@ static int ccs_probe(struct i2c_client *client)
* *
* Rotation also changes the bayer pattern. * Rotation also changes the bayer pattern.
*/ */
if (sensor->hwcfg->module_board_orient == if (sensor->hwcfg.module_board_orient ==
CCS_MODULE_BOARD_ORIENT_180) CCS_MODULE_BOARD_ORIENT_180)
sensor->hvflip_inv_mask = sensor->hvflip_inv_mask =
CCS_IMAGE_ORIENTATION_HORIZONTAL_MIRROR | CCS_IMAGE_ORIENTATION_HORIZONTAL_MIRROR |
...@@ -3133,8 +3135,8 @@ static int ccs_probe(struct i2c_client *client) ...@@ -3133,8 +3135,8 @@ static int ccs_probe(struct i2c_client *client)
/* prepare PLL configuration input values */ /* prepare PLL configuration input values */
sensor->pll.bus_type = SMIAPP_PLL_BUS_TYPE_CSI2; sensor->pll.bus_type = SMIAPP_PLL_BUS_TYPE_CSI2;
sensor->pll.csi2.lanes = sensor->hwcfg->lanes; sensor->pll.csi2.lanes = sensor->hwcfg.lanes;
sensor->pll.ext_clk_freq_hz = sensor->hwcfg->ext_clk; sensor->pll.ext_clk_freq_hz = sensor->hwcfg.ext_clk;
sensor->pll.scale_n = CCS_LIM(sensor, SCALER_N_MIN); sensor->pll.scale_n = CCS_LIM(sensor, SCALER_N_MIN);
ccs_create_subdev(sensor, sensor->scaler, " scaler", 2, ccs_create_subdev(sensor, sensor->scaler, " scaler", 2,
......
...@@ -152,13 +152,13 @@ static int jt8ev1_post_poweron(struct ccs_sensor *sensor) ...@@ -152,13 +152,13 @@ static int jt8ev1_post_poweron(struct ccs_sensor *sensor)
if (rval < 0) if (rval < 0)
return rval; return rval;
switch (sensor->hwcfg->ext_clk) { switch (sensor->hwcfg.ext_clk) {
case 9600000: case 9600000:
return ccs_write_addr_8s(sensor, regs_96, return ccs_write_addr_8s(sensor, regs_96,
ARRAY_SIZE(regs_96)); ARRAY_SIZE(regs_96));
default: default:
dev_warn(&client->dev, "no MSRs for %d Hz ext_clk\n", dev_warn(&client->dev, "no MSRs for %d Hz ext_clk\n",
sensor->hwcfg->ext_clk); sensor->hwcfg.ext_clk);
return 0; return 0;
} }
} }
......
...@@ -215,7 +215,7 @@ struct ccs_sensor { ...@@ -215,7 +215,7 @@ struct ccs_sensor {
struct ccs_subdev *binner; struct ccs_subdev *binner;
struct ccs_subdev *scaler; struct ccs_subdev *scaler;
struct ccs_subdev *pixel_array; struct ccs_subdev *pixel_array;
struct ccs_hwconfig *hwcfg; struct ccs_hwconfig hwcfg;
struct regulator *vana; struct regulator *vana;
struct clk *ext_clk; struct clk *ext_clk;
struct gpio_desc *xshutdown; struct gpio_desc *xshutdown;
......
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