--- /dev/null
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * CrOS EC ANX7688 HDMI->DP bridge driver
+ *
+ * Copyright 2020 Google LLC
+ */
+
+#include <drm/drm_bridge.h>
+#include <drm/drm_print.h>
+#include <linux/i2c.h>
+#include <linux/module.h>
+#include <linux/regmap.h>
+#include <linux/types.h>
+
+/* Register addresses */
+#define ANX7688_VENDOR_ID_REG          0x00
+#define ANX7688_DEVICE_ID_REG          0x02
+
+#define ANX7688_FW_VERSION_REG         0x80
+
+#define ANX7688_DP_BANDWIDTH_REG       0x85
+#define ANX7688_DP_LANE_COUNT_REG      0x86
+
+#define ANX7688_VENDOR_ID              0x1f29
+#define ANX7688_DEVICE_ID              0x7688
+
+/* First supported firmware version (0.85) */
+#define ANX7688_MINIMUM_FW_VERSION     0x0085
+
+static const struct regmap_config cros_ec_anx7688_regmap_config = {
+       .reg_bits = 8,
+       .val_bits = 8,
+};
+
+struct cros_ec_anx7688 {
+       struct i2c_client *client;
+       struct regmap *regmap;
+       struct drm_bridge bridge;
+       bool filter;
+};
+
+static inline struct cros_ec_anx7688 *
+bridge_to_cros_ec_anx7688(struct drm_bridge *bridge)
+{
+       return container_of(bridge, struct cros_ec_anx7688, bridge);
+}
+
+static bool cros_ec_anx7688_bridge_mode_fixup(struct drm_bridge *bridge,
+                                             const struct drm_display_mode *mode,
+                                             struct drm_display_mode *adjusted_mode)
+{
+       struct cros_ec_anx7688 *anx = bridge_to_cros_ec_anx7688(bridge);
+       int totalbw, requiredbw;
+       u8 dpbw, lanecount;
+       u8 regs[2];
+       int ret;
+
+       if (!anx->filter)
+               return true;
+
+       /* Read both regs 0x85 (bandwidth) and 0x86 (lane count). */
+       ret = regmap_bulk_read(anx->regmap, ANX7688_DP_BANDWIDTH_REG, regs, 2);
+       if (ret < 0) {
+               DRM_ERROR("Failed to read bandwidth/lane count\n");
+               return false;
+       }
+       dpbw = regs[0];
+       lanecount = regs[1];
+
+       /* Maximum 0x19 bandwidth (6.75 Gbps Turbo mode), 2 lanes */
+       if (dpbw > 0x19 || lanecount > 2) {
+               DRM_ERROR("Invalid bandwidth/lane count (%02x/%d)\n", dpbw,
+                         lanecount);
+               return false;
+       }
+
+       /* Compute available bandwidth (kHz) */
+       totalbw = dpbw * lanecount * 270000 * 8 / 10;
+
+       /* Required bandwidth (8 bpc, kHz) */
+       requiredbw = mode->clock * 8 * 3;
+
+       DRM_DEBUG_KMS("DP bandwidth: %d kHz (%02x/%d); mode requires %d Khz\n",
+                     totalbw, dpbw, lanecount, requiredbw);
+
+       if (totalbw == 0) {
+               DRM_ERROR("Bandwidth/lane count are 0, not rejecting modes\n");
+               return true;
+       }
+
+       return totalbw >= requiredbw;
+}
+
+static const struct drm_bridge_funcs cros_ec_anx7688_bridge_funcs = {
+       .mode_fixup = cros_ec_anx7688_bridge_mode_fixup,
+};
+
+static int cros_ec_anx7688_bridge_probe(struct i2c_client *client)
+{
+       struct device *dev = &client->dev;
+       struct cros_ec_anx7688 *anx7688;
+       u16 vendor, device, fw_version;
+       u8 buffer[4];
+       int ret;
+
+       anx7688 = devm_kzalloc(dev, sizeof(*anx7688), GFP_KERNEL);
+       if (!anx7688)
+               return -ENOMEM;
+
+       anx7688->client = client;
+       i2c_set_clientdata(client, anx7688);
+
+       anx7688->regmap = devm_regmap_init_i2c(client, &cros_ec_anx7688_regmap_config);
+       if (IS_ERR(anx7688->regmap)) {
+               ret = PTR_ERR(anx7688->regmap);
+               dev_err(dev, "regmap i2c init failed: %d\n", ret);
+               return ret;
+       }
+
+       /* Read both vendor and device id (4 bytes). */
+       ret = regmap_bulk_read(anx7688->regmap, ANX7688_VENDOR_ID_REG,
+                              buffer, 4);
+       if (ret) {
+               dev_err(dev, "Failed to read chip vendor/device id\n");
+               return ret;
+       }
+
+       vendor = (u16)buffer[1] << 8 | buffer[0];
+       device = (u16)buffer[3] << 8 | buffer[2];
+       if (vendor != ANX7688_VENDOR_ID || device != ANX7688_DEVICE_ID) {
+               dev_err(dev, "Invalid vendor/device id %04x/%04x\n",
+                       vendor, device);
+               return -ENODEV;
+       }
+
+       ret = regmap_bulk_read(anx7688->regmap, ANX7688_FW_VERSION_REG,
+                              buffer, 2);
+       if (ret) {
+               dev_err(dev, "Failed to read firmware version\n");
+               return ret;
+       }
+
+       fw_version = (u16)buffer[0] << 8 | buffer[1];
+       dev_info(dev, "ANX7688 firmware version 0x%04x\n", fw_version);
+
+       anx7688->bridge.of_node = dev->of_node;
+
+       /* FW version >= 0.85 supports bandwidth/lane count registers */
+       if (fw_version >= ANX7688_MINIMUM_FW_VERSION)
+               anx7688->filter = true;
+       else
+               /* Warn, but not fail, for backwards compatibility */
+               DRM_WARN("Old ANX7688 FW version (0x%04x), not filtering\n",
+                        fw_version);
+
+       anx7688->bridge.funcs = &cros_ec_anx7688_bridge_funcs;
+       drm_bridge_add(&anx7688->bridge);
+
+       return 0;
+}
+
+static int cros_ec_anx7688_bridge_remove(struct i2c_client *client)
+{
+       struct cros_ec_anx7688 *anx7688 = i2c_get_clientdata(client);
+
+       drm_bridge_remove(&anx7688->bridge);
+
+       return 0;
+}
+
+static const struct of_device_id cros_ec_anx7688_bridge_match_table[] = {
+       { .compatible = "google,cros-ec-anx7688" },
+       { }
+};
+MODULE_DEVICE_TABLE(of, cros_ec_anx7688_bridge_match_table);
+
+static struct i2c_driver cros_ec_anx7688_bridge_driver = {
+       .probe_new = cros_ec_anx7688_bridge_probe,
+       .remove = cros_ec_anx7688_bridge_remove,
+       .driver = {
+               .name = "cros-ec-anx7688-bridge",
+               .of_match_table = cros_ec_anx7688_bridge_match_table,
+       },
+};
+
+module_i2c_driver(cros_ec_anx7688_bridge_driver);
+
+MODULE_DESCRIPTION("ChromeOS EC ANX7688 HDMI->DP bridge driver");
+MODULE_AUTHOR("Nicolas Boichat <drinkcat@chromium.org>");
+MODULE_AUTHOR("Enric Balletbo i Serra <enric.balletbo@collabora.com>");
+MODULE_LICENSE("GPL");