xref: /openbmc/linux/drivers/gpu/drm/bridge/cros-ec-anx7688.c (revision 0791faebfe750292a8a842b64795a390ca4a3b51)
144602b10SEnric Balletbo i Serra // SPDX-License-Identifier: GPL-2.0-only
244602b10SEnric Balletbo i Serra /*
344602b10SEnric Balletbo i Serra  * CrOS EC ANX7688 HDMI->DP bridge driver
444602b10SEnric Balletbo i Serra  *
544602b10SEnric Balletbo i Serra  * Copyright 2020 Google LLC
644602b10SEnric Balletbo i Serra  */
744602b10SEnric Balletbo i Serra 
844602b10SEnric Balletbo i Serra #include <drm/drm_bridge.h>
944602b10SEnric Balletbo i Serra #include <drm/drm_print.h>
1044602b10SEnric Balletbo i Serra #include <linux/i2c.h>
1144602b10SEnric Balletbo i Serra #include <linux/module.h>
1244602b10SEnric Balletbo i Serra #include <linux/regmap.h>
1344602b10SEnric Balletbo i Serra #include <linux/types.h>
1444602b10SEnric Balletbo i Serra 
1544602b10SEnric Balletbo i Serra /* Register addresses */
1644602b10SEnric Balletbo i Serra #define ANX7688_VENDOR_ID_REG		0x00
1744602b10SEnric Balletbo i Serra #define ANX7688_DEVICE_ID_REG		0x02
1844602b10SEnric Balletbo i Serra 
1944602b10SEnric Balletbo i Serra #define ANX7688_FW_VERSION_REG		0x80
2044602b10SEnric Balletbo i Serra 
2144602b10SEnric Balletbo i Serra #define ANX7688_DP_BANDWIDTH_REG	0x85
2244602b10SEnric Balletbo i Serra #define ANX7688_DP_LANE_COUNT_REG	0x86
2344602b10SEnric Balletbo i Serra 
2444602b10SEnric Balletbo i Serra #define ANX7688_VENDOR_ID		0x1f29
2544602b10SEnric Balletbo i Serra #define ANX7688_DEVICE_ID		0x7688
2644602b10SEnric Balletbo i Serra 
2744602b10SEnric Balletbo i Serra /* First supported firmware version (0.85) */
2844602b10SEnric Balletbo i Serra #define ANX7688_MINIMUM_FW_VERSION	0x0085
2944602b10SEnric Balletbo i Serra 
3044602b10SEnric Balletbo i Serra static const struct regmap_config cros_ec_anx7688_regmap_config = {
3144602b10SEnric Balletbo i Serra 	.reg_bits = 8,
3244602b10SEnric Balletbo i Serra 	.val_bits = 8,
3344602b10SEnric Balletbo i Serra };
3444602b10SEnric Balletbo i Serra 
3544602b10SEnric Balletbo i Serra struct cros_ec_anx7688 {
3644602b10SEnric Balletbo i Serra 	struct i2c_client *client;
3744602b10SEnric Balletbo i Serra 	struct regmap *regmap;
3844602b10SEnric Balletbo i Serra 	struct drm_bridge bridge;
3944602b10SEnric Balletbo i Serra 	bool filter;
4044602b10SEnric Balletbo i Serra };
4144602b10SEnric Balletbo i Serra 
4244602b10SEnric Balletbo i Serra static inline struct cros_ec_anx7688 *
bridge_to_cros_ec_anx7688(struct drm_bridge * bridge)4344602b10SEnric Balletbo i Serra bridge_to_cros_ec_anx7688(struct drm_bridge *bridge)
4444602b10SEnric Balletbo i Serra {
4544602b10SEnric Balletbo i Serra 	return container_of(bridge, struct cros_ec_anx7688, bridge);
4644602b10SEnric Balletbo i Serra }
4744602b10SEnric Balletbo i Serra 
cros_ec_anx7688_bridge_mode_fixup(struct drm_bridge * bridge,const struct drm_display_mode * mode,struct drm_display_mode * adjusted_mode)4844602b10SEnric Balletbo i Serra static bool cros_ec_anx7688_bridge_mode_fixup(struct drm_bridge *bridge,
4944602b10SEnric Balletbo i Serra 					      const struct drm_display_mode *mode,
5044602b10SEnric Balletbo i Serra 					      struct drm_display_mode *adjusted_mode)
5144602b10SEnric Balletbo i Serra {
5244602b10SEnric Balletbo i Serra 	struct cros_ec_anx7688 *anx = bridge_to_cros_ec_anx7688(bridge);
5344602b10SEnric Balletbo i Serra 	int totalbw, requiredbw;
5444602b10SEnric Balletbo i Serra 	u8 dpbw, lanecount;
5544602b10SEnric Balletbo i Serra 	u8 regs[2];
5644602b10SEnric Balletbo i Serra 	int ret;
5744602b10SEnric Balletbo i Serra 
5844602b10SEnric Balletbo i Serra 	if (!anx->filter)
5944602b10SEnric Balletbo i Serra 		return true;
6044602b10SEnric Balletbo i Serra 
6144602b10SEnric Balletbo i Serra 	/* Read both regs 0x85 (bandwidth) and 0x86 (lane count). */
6244602b10SEnric Balletbo i Serra 	ret = regmap_bulk_read(anx->regmap, ANX7688_DP_BANDWIDTH_REG, regs, 2);
6344602b10SEnric Balletbo i Serra 	if (ret < 0) {
6444602b10SEnric Balletbo i Serra 		DRM_ERROR("Failed to read bandwidth/lane count\n");
6544602b10SEnric Balletbo i Serra 		return false;
6644602b10SEnric Balletbo i Serra 	}
6744602b10SEnric Balletbo i Serra 	dpbw = regs[0];
6844602b10SEnric Balletbo i Serra 	lanecount = regs[1];
6944602b10SEnric Balletbo i Serra 
7044602b10SEnric Balletbo i Serra 	/* Maximum 0x19 bandwidth (6.75 Gbps Turbo mode), 2 lanes */
7144602b10SEnric Balletbo i Serra 	if (dpbw > 0x19 || lanecount > 2) {
7244602b10SEnric Balletbo i Serra 		DRM_ERROR("Invalid bandwidth/lane count (%02x/%d)\n", dpbw,
7344602b10SEnric Balletbo i Serra 			  lanecount);
7444602b10SEnric Balletbo i Serra 		return false;
7544602b10SEnric Balletbo i Serra 	}
7644602b10SEnric Balletbo i Serra 
7744602b10SEnric Balletbo i Serra 	/* Compute available bandwidth (kHz) */
7844602b10SEnric Balletbo i Serra 	totalbw = dpbw * lanecount * 270000 * 8 / 10;
7944602b10SEnric Balletbo i Serra 
8044602b10SEnric Balletbo i Serra 	/* Required bandwidth (8 bpc, kHz) */
8144602b10SEnric Balletbo i Serra 	requiredbw = mode->clock * 8 * 3;
8244602b10SEnric Balletbo i Serra 
8344602b10SEnric Balletbo i Serra 	DRM_DEBUG_KMS("DP bandwidth: %d kHz (%02x/%d); mode requires %d Khz\n",
8444602b10SEnric Balletbo i Serra 		      totalbw, dpbw, lanecount, requiredbw);
8544602b10SEnric Balletbo i Serra 
8644602b10SEnric Balletbo i Serra 	if (totalbw == 0) {
8744602b10SEnric Balletbo i Serra 		DRM_ERROR("Bandwidth/lane count are 0, not rejecting modes\n");
8844602b10SEnric Balletbo i Serra 		return true;
8944602b10SEnric Balletbo i Serra 	}
9044602b10SEnric Balletbo i Serra 
9144602b10SEnric Balletbo i Serra 	return totalbw >= requiredbw;
9244602b10SEnric Balletbo i Serra }
9344602b10SEnric Balletbo i Serra 
9444602b10SEnric Balletbo i Serra static const struct drm_bridge_funcs cros_ec_anx7688_bridge_funcs = {
9544602b10SEnric Balletbo i Serra 	.mode_fixup = cros_ec_anx7688_bridge_mode_fixup,
9644602b10SEnric Balletbo i Serra };
9744602b10SEnric Balletbo i Serra 
cros_ec_anx7688_bridge_probe(struct i2c_client * client)9844602b10SEnric Balletbo i Serra static int cros_ec_anx7688_bridge_probe(struct i2c_client *client)
9944602b10SEnric Balletbo i Serra {
10044602b10SEnric Balletbo i Serra 	struct device *dev = &client->dev;
10144602b10SEnric Balletbo i Serra 	struct cros_ec_anx7688 *anx7688;
10244602b10SEnric Balletbo i Serra 	u16 vendor, device, fw_version;
10344602b10SEnric Balletbo i Serra 	u8 buffer[4];
10444602b10SEnric Balletbo i Serra 	int ret;
10544602b10SEnric Balletbo i Serra 
10644602b10SEnric Balletbo i Serra 	anx7688 = devm_kzalloc(dev, sizeof(*anx7688), GFP_KERNEL);
10744602b10SEnric Balletbo i Serra 	if (!anx7688)
10844602b10SEnric Balletbo i Serra 		return -ENOMEM;
10944602b10SEnric Balletbo i Serra 
11044602b10SEnric Balletbo i Serra 	anx7688->client = client;
11144602b10SEnric Balletbo i Serra 	i2c_set_clientdata(client, anx7688);
11244602b10SEnric Balletbo i Serra 
11344602b10SEnric Balletbo i Serra 	anx7688->regmap = devm_regmap_init_i2c(client, &cros_ec_anx7688_regmap_config);
11444602b10SEnric Balletbo i Serra 	if (IS_ERR(anx7688->regmap)) {
11544602b10SEnric Balletbo i Serra 		ret = PTR_ERR(anx7688->regmap);
11644602b10SEnric Balletbo i Serra 		dev_err(dev, "regmap i2c init failed: %d\n", ret);
11744602b10SEnric Balletbo i Serra 		return ret;
11844602b10SEnric Balletbo i Serra 	}
11944602b10SEnric Balletbo i Serra 
12044602b10SEnric Balletbo i Serra 	/* Read both vendor and device id (4 bytes). */
12144602b10SEnric Balletbo i Serra 	ret = regmap_bulk_read(anx7688->regmap, ANX7688_VENDOR_ID_REG,
12244602b10SEnric Balletbo i Serra 			       buffer, 4);
12344602b10SEnric Balletbo i Serra 	if (ret) {
12444602b10SEnric Balletbo i Serra 		dev_err(dev, "Failed to read chip vendor/device id\n");
12544602b10SEnric Balletbo i Serra 		return ret;
12644602b10SEnric Balletbo i Serra 	}
12744602b10SEnric Balletbo i Serra 
12844602b10SEnric Balletbo i Serra 	vendor = (u16)buffer[1] << 8 | buffer[0];
12944602b10SEnric Balletbo i Serra 	device = (u16)buffer[3] << 8 | buffer[2];
13044602b10SEnric Balletbo i Serra 	if (vendor != ANX7688_VENDOR_ID || device != ANX7688_DEVICE_ID) {
13144602b10SEnric Balletbo i Serra 		dev_err(dev, "Invalid vendor/device id %04x/%04x\n",
13244602b10SEnric Balletbo i Serra 			vendor, device);
13344602b10SEnric Balletbo i Serra 		return -ENODEV;
13444602b10SEnric Balletbo i Serra 	}
13544602b10SEnric Balletbo i Serra 
13644602b10SEnric Balletbo i Serra 	ret = regmap_bulk_read(anx7688->regmap, ANX7688_FW_VERSION_REG,
13744602b10SEnric Balletbo i Serra 			       buffer, 2);
13844602b10SEnric Balletbo i Serra 	if (ret) {
13944602b10SEnric Balletbo i Serra 		dev_err(dev, "Failed to read firmware version\n");
14044602b10SEnric Balletbo i Serra 		return ret;
14144602b10SEnric Balletbo i Serra 	}
14244602b10SEnric Balletbo i Serra 
14344602b10SEnric Balletbo i Serra 	fw_version = (u16)buffer[0] << 8 | buffer[1];
14444602b10SEnric Balletbo i Serra 	dev_info(dev, "ANX7688 firmware version 0x%04x\n", fw_version);
14544602b10SEnric Balletbo i Serra 
14644602b10SEnric Balletbo i Serra 	anx7688->bridge.of_node = dev->of_node;
14744602b10SEnric Balletbo i Serra 
14844602b10SEnric Balletbo i Serra 	/* FW version >= 0.85 supports bandwidth/lane count registers */
14944602b10SEnric Balletbo i Serra 	if (fw_version >= ANX7688_MINIMUM_FW_VERSION)
15044602b10SEnric Balletbo i Serra 		anx7688->filter = true;
15144602b10SEnric Balletbo i Serra 	else
15244602b10SEnric Balletbo i Serra 		/* Warn, but not fail, for backwards compatibility */
15344602b10SEnric Balletbo i Serra 		DRM_WARN("Old ANX7688 FW version (0x%04x), not filtering\n",
15444602b10SEnric Balletbo i Serra 			 fw_version);
15544602b10SEnric Balletbo i Serra 
15644602b10SEnric Balletbo i Serra 	anx7688->bridge.funcs = &cros_ec_anx7688_bridge_funcs;
15744602b10SEnric Balletbo i Serra 	drm_bridge_add(&anx7688->bridge);
15844602b10SEnric Balletbo i Serra 
15944602b10SEnric Balletbo i Serra 	return 0;
16044602b10SEnric Balletbo i Serra }
16144602b10SEnric Balletbo i Serra 
cros_ec_anx7688_bridge_remove(struct i2c_client * client)162ed5c2f5fSUwe Kleine-König static void cros_ec_anx7688_bridge_remove(struct i2c_client *client)
16344602b10SEnric Balletbo i Serra {
16444602b10SEnric Balletbo i Serra 	struct cros_ec_anx7688 *anx7688 = i2c_get_clientdata(client);
16544602b10SEnric Balletbo i Serra 
16644602b10SEnric Balletbo i Serra 	drm_bridge_remove(&anx7688->bridge);
16744602b10SEnric Balletbo i Serra }
16844602b10SEnric Balletbo i Serra 
16944602b10SEnric Balletbo i Serra static const struct of_device_id cros_ec_anx7688_bridge_match_table[] = {
17044602b10SEnric Balletbo i Serra 	{ .compatible = "google,cros-ec-anx7688" },
17144602b10SEnric Balletbo i Serra 	{ }
17244602b10SEnric Balletbo i Serra };
17344602b10SEnric Balletbo i Serra MODULE_DEVICE_TABLE(of, cros_ec_anx7688_bridge_match_table);
17444602b10SEnric Balletbo i Serra 
17544602b10SEnric Balletbo i Serra static struct i2c_driver cros_ec_anx7688_bridge_driver = {
176*332af828SUwe Kleine-König 	.probe = cros_ec_anx7688_bridge_probe,
17744602b10SEnric Balletbo i Serra 	.remove = cros_ec_anx7688_bridge_remove,
17844602b10SEnric Balletbo i Serra 	.driver = {
17944602b10SEnric Balletbo i Serra 		.name = "cros-ec-anx7688-bridge",
18044602b10SEnric Balletbo i Serra 		.of_match_table = cros_ec_anx7688_bridge_match_table,
18144602b10SEnric Balletbo i Serra 	},
18244602b10SEnric Balletbo i Serra };
18344602b10SEnric Balletbo i Serra 
18444602b10SEnric Balletbo i Serra module_i2c_driver(cros_ec_anx7688_bridge_driver);
18544602b10SEnric Balletbo i Serra 
18644602b10SEnric Balletbo i Serra MODULE_DESCRIPTION("ChromeOS EC ANX7688 HDMI->DP bridge driver");
18744602b10SEnric Balletbo i Serra MODULE_AUTHOR("Nicolas Boichat <drinkcat@chromium.org>");
18844602b10SEnric Balletbo i Serra MODULE_AUTHOR("Enric Balletbo i Serra <enric.balletbo@collabora.com>");
18944602b10SEnric Balletbo i Serra MODULE_LICENSE("GPL");
190