cros-ec-anx7688.c (5059B)
1// SPDX-License-Identifier: GPL-2.0-only 2/* 3 * CrOS EC ANX7688 HDMI->DP bridge driver 4 * 5 * Copyright 2020 Google LLC 6 */ 7 8#include <drm/drm_bridge.h> 9#include <drm/drm_print.h> 10#include <linux/i2c.h> 11#include <linux/module.h> 12#include <linux/regmap.h> 13#include <linux/types.h> 14 15/* Register addresses */ 16#define ANX7688_VENDOR_ID_REG 0x00 17#define ANX7688_DEVICE_ID_REG 0x02 18 19#define ANX7688_FW_VERSION_REG 0x80 20 21#define ANX7688_DP_BANDWIDTH_REG 0x85 22#define ANX7688_DP_LANE_COUNT_REG 0x86 23 24#define ANX7688_VENDOR_ID 0x1f29 25#define ANX7688_DEVICE_ID 0x7688 26 27/* First supported firmware version (0.85) */ 28#define ANX7688_MINIMUM_FW_VERSION 0x0085 29 30static const struct regmap_config cros_ec_anx7688_regmap_config = { 31 .reg_bits = 8, 32 .val_bits = 8, 33}; 34 35struct cros_ec_anx7688 { 36 struct i2c_client *client; 37 struct regmap *regmap; 38 struct drm_bridge bridge; 39 bool filter; 40}; 41 42static inline struct cros_ec_anx7688 * 43bridge_to_cros_ec_anx7688(struct drm_bridge *bridge) 44{ 45 return container_of(bridge, struct cros_ec_anx7688, bridge); 46} 47 48static bool cros_ec_anx7688_bridge_mode_fixup(struct drm_bridge *bridge, 49 const struct drm_display_mode *mode, 50 struct drm_display_mode *adjusted_mode) 51{ 52 struct cros_ec_anx7688 *anx = bridge_to_cros_ec_anx7688(bridge); 53 int totalbw, requiredbw; 54 u8 dpbw, lanecount; 55 u8 regs[2]; 56 int ret; 57 58 if (!anx->filter) 59 return true; 60 61 /* Read both regs 0x85 (bandwidth) and 0x86 (lane count). */ 62 ret = regmap_bulk_read(anx->regmap, ANX7688_DP_BANDWIDTH_REG, regs, 2); 63 if (ret < 0) { 64 DRM_ERROR("Failed to read bandwidth/lane count\n"); 65 return false; 66 } 67 dpbw = regs[0]; 68 lanecount = regs[1]; 69 70 /* Maximum 0x19 bandwidth (6.75 Gbps Turbo mode), 2 lanes */ 71 if (dpbw > 0x19 || lanecount > 2) { 72 DRM_ERROR("Invalid bandwidth/lane count (%02x/%d)\n", dpbw, 73 lanecount); 74 return false; 75 } 76 77 /* Compute available bandwidth (kHz) */ 78 totalbw = dpbw * lanecount * 270000 * 8 / 10; 79 80 /* Required bandwidth (8 bpc, kHz) */ 81 requiredbw = mode->clock * 8 * 3; 82 83 DRM_DEBUG_KMS("DP bandwidth: %d kHz (%02x/%d); mode requires %d Khz\n", 84 totalbw, dpbw, lanecount, requiredbw); 85 86 if (totalbw == 0) { 87 DRM_ERROR("Bandwidth/lane count are 0, not rejecting modes\n"); 88 return true; 89 } 90 91 return totalbw >= requiredbw; 92} 93 94static const struct drm_bridge_funcs cros_ec_anx7688_bridge_funcs = { 95 .mode_fixup = cros_ec_anx7688_bridge_mode_fixup, 96}; 97 98static int cros_ec_anx7688_bridge_probe(struct i2c_client *client) 99{ 100 struct device *dev = &client->dev; 101 struct cros_ec_anx7688 *anx7688; 102 u16 vendor, device, fw_version; 103 u8 buffer[4]; 104 int ret; 105 106 anx7688 = devm_kzalloc(dev, sizeof(*anx7688), GFP_KERNEL); 107 if (!anx7688) 108 return -ENOMEM; 109 110 anx7688->client = client; 111 i2c_set_clientdata(client, anx7688); 112 113 anx7688->regmap = devm_regmap_init_i2c(client, &cros_ec_anx7688_regmap_config); 114 if (IS_ERR(anx7688->regmap)) { 115 ret = PTR_ERR(anx7688->regmap); 116 dev_err(dev, "regmap i2c init failed: %d\n", ret); 117 return ret; 118 } 119 120 /* Read both vendor and device id (4 bytes). */ 121 ret = regmap_bulk_read(anx7688->regmap, ANX7688_VENDOR_ID_REG, 122 buffer, 4); 123 if (ret) { 124 dev_err(dev, "Failed to read chip vendor/device id\n"); 125 return ret; 126 } 127 128 vendor = (u16)buffer[1] << 8 | buffer[0]; 129 device = (u16)buffer[3] << 8 | buffer[2]; 130 if (vendor != ANX7688_VENDOR_ID || device != ANX7688_DEVICE_ID) { 131 dev_err(dev, "Invalid vendor/device id %04x/%04x\n", 132 vendor, device); 133 return -ENODEV; 134 } 135 136 ret = regmap_bulk_read(anx7688->regmap, ANX7688_FW_VERSION_REG, 137 buffer, 2); 138 if (ret) { 139 dev_err(dev, "Failed to read firmware version\n"); 140 return ret; 141 } 142 143 fw_version = (u16)buffer[0] << 8 | buffer[1]; 144 dev_info(dev, "ANX7688 firmware version 0x%04x\n", fw_version); 145 146 anx7688->bridge.of_node = dev->of_node; 147 148 /* FW version >= 0.85 supports bandwidth/lane count registers */ 149 if (fw_version >= ANX7688_MINIMUM_FW_VERSION) 150 anx7688->filter = true; 151 else 152 /* Warn, but not fail, for backwards compatibility */ 153 DRM_WARN("Old ANX7688 FW version (0x%04x), not filtering\n", 154 fw_version); 155 156 anx7688->bridge.funcs = &cros_ec_anx7688_bridge_funcs; 157 drm_bridge_add(&anx7688->bridge); 158 159 return 0; 160} 161 162static int cros_ec_anx7688_bridge_remove(struct i2c_client *client) 163{ 164 struct cros_ec_anx7688 *anx7688 = i2c_get_clientdata(client); 165 166 drm_bridge_remove(&anx7688->bridge); 167 168 return 0; 169} 170 171static const struct of_device_id cros_ec_anx7688_bridge_match_table[] = { 172 { .compatible = "google,cros-ec-anx7688" }, 173 { } 174}; 175MODULE_DEVICE_TABLE(of, cros_ec_anx7688_bridge_match_table); 176 177static struct i2c_driver cros_ec_anx7688_bridge_driver = { 178 .probe_new = cros_ec_anx7688_bridge_probe, 179 .remove = cros_ec_anx7688_bridge_remove, 180 .driver = { 181 .name = "cros-ec-anx7688-bridge", 182 .of_match_table = cros_ec_anx7688_bridge_match_table, 183 }, 184}; 185 186module_i2c_driver(cros_ec_anx7688_bridge_driver); 187 188MODULE_DESCRIPTION("ChromeOS EC ANX7688 HDMI->DP bridge driver"); 189MODULE_AUTHOR("Nicolas Boichat <drinkcat@chromium.org>"); 190MODULE_AUTHOR("Enric Balletbo i Serra <enric.balletbo@collabora.com>"); 191MODULE_LICENSE("GPL");