On 5 February 2025 10:18:00 EET, Yongbang Shi <shiyongbang@xxxxxxxxxx> wrote: >> On Mon, Jan 27, 2025 at 11:20:23AM +0800, Yongbang Shi wrote: >>> From: Baihan Li <libaihan@xxxxxxxxxx> >>> >>> Create 3 files in drm debugfs: >> This definitely needs to be split. > >Hi Dmitry, > >Right, I got it. I will split any patch which has mutiple fileds changing blended together. > > >>> colorbar-cfg: Get/Set colorbar cfg >> What does that mean? >> >It's a dp's color bar output, and we have a configuration that >we can set color bar's color type and stripe movement. What is a DP colour bar? > > >>> hibmc-dp: Get dp link status >>> hibmc-dp-edid: Print edid information >> edid-decode /sys/class/drm/card0-DP-1/edid ? > >Yeah, we can directly use "cat" to print edid info by it. I will add comments and >example in next series git log. What is the benefit in having a nonstandard EDID decoder in the kernel if you can use cat to get hexdump from sysfs and then use any of the tools available for EDID deciding? > > >>> Signed-off-by: Baihan Li <libaihan@xxxxxxxxxx> >>> Signed-off-by: Yongbang Shi <shiyongbang@xxxxxxxxxx> >>> --- >>> drivers/gpu/drm/hisilicon/hibmc/Makefile | 3 +- >>> drivers/gpu/drm/hisilicon/hibmc/dp/dp_comm.h | 3 + >>> drivers/gpu/drm/hisilicon/hibmc/dp/dp_hw.c | 58 +++++ >>> drivers/gpu/drm/hisilicon/hibmc/dp/dp_hw.h | 44 ++++ >>> drivers/gpu/drm/hisilicon/hibmc/dp/dp_link.c | 40 +++- >>> drivers/gpu/drm/hisilicon/hibmc/dp/dp_reg.h | 4 +- >>> .../drm/hisilicon/hibmc/hibmc_drm_debugfs.c | 214 ++++++++++++++++++ >>> .../gpu/drm/hisilicon/hibmc/hibmc_drm_dp.c | 2 + >>> .../gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c | 3 + >>> .../gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h | 2 + >>> 10 files changed, 363 insertions(+), 10 deletions(-) >>> create mode 100644 drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_debugfs.c >>> >>> diff --git a/drivers/gpu/drm/hisilicon/hibmc/Makefile b/drivers/gpu/drm/hisilicon/hibmc/Makefile >>> index 35a74cc10c80..c14f5182c067 100644 >>> --- a/drivers/gpu/drm/hisilicon/hibmc/Makefile >>> +++ b/drivers/gpu/drm/hisilicon/hibmc/Makefile >>> @@ -1,5 +1,6 @@ >>> # SPDX-License-Identifier: GPL-2.0-only >>> hibmc-drm-y := hibmc_drm_drv.o hibmc_drm_de.o hibmc_drm_vdac.o hibmc_drm_i2c.o \ >>> - dp/dp_aux.o dp/dp_link.o dp/dp_hw.o dp/dp_phy.o hibmc_drm_dp.o >>> + dp/dp_aux.o dp/dp_link.o dp/dp_hw.o dp/dp_phy.o hibmc_drm_dp.o \ >>> + hibmc_drm_debugfs.o >>> obj-$(CONFIG_DRM_HISI_HIBMC) += hibmc-drm.o >>> diff --git a/drivers/gpu/drm/hisilicon/hibmc/dp/dp_comm.h b/drivers/gpu/drm/hisilicon/hibmc/dp/dp_comm.h >>> index 7edcecd5a5f0..67f6c81a35ed 100644 >>> --- a/drivers/gpu/drm/hisilicon/hibmc/dp/dp_comm.h >>> +++ b/drivers/gpu/drm/hisilicon/hibmc/dp/dp_comm.h >>> @@ -26,6 +26,9 @@ struct hibmc_link_status { >>> struct hibmc_link_cap { >>> u8 link_rate; >>> u8 lanes; >>> + int rx_dpcd_revision; >>> + bool is_tps3; >>> + bool is_tps4; >>> }; >>> struct hibmc_dp_link { >>> diff --git a/drivers/gpu/drm/hisilicon/hibmc/dp/dp_hw.c b/drivers/gpu/drm/hisilicon/hibmc/dp/dp_hw.c >>> index 50050908606f..9c8b91ff0e3b 100644 >>> --- a/drivers/gpu/drm/hisilicon/hibmc/dp/dp_hw.c >>> +++ b/drivers/gpu/drm/hisilicon/hibmc/dp/dp_hw.c >>> @@ -226,3 +226,61 @@ int hibmc_dp_mode_set(struct hibmc_dp *dp, struct drm_display_mode *mode) >>> return 0; >>> } >>> + >>> +u8 hibmc_dp_get_link_rate(struct hibmc_dp *dp) >>> +{ >>> + return dp->dp_dev->link.cap.link_rate; >>> +} >>> + >>> +u8 hibmc_dp_get_lanes(struct hibmc_dp *dp) >>> +{ >>> + return dp->dp_dev->link.cap.lanes; >>> +} >>> + >>> +int hibmc_dp_get_dpcd(struct hibmc_dp *dp) >>> +{ >>> + return dp->dp_dev->link.cap.rx_dpcd_revision; >>> +} >>> + >>> +static const struct hibmc_dp_color_raw g_rgb_raw[] = { >>> + {CBAR_COLOR_BAR, 0x000, 0x000, 0x000}, >>> + {CBAR_WHITE, 0xfff, 0xfff, 0xfff}, >>> + {CBAR_RED, 0xfff, 0x000, 0x000}, >>> + {CBAR_ORANGE, 0xfff, 0x800, 0x000}, >>> + {CBAR_YELLOW, 0xfff, 0xfff, 0x000}, >>> + {CBAR_GREEN, 0x000, 0xfff, 0x000}, >>> + {CBAR_CYAN, 0x000, 0x800, 0x800}, >>> + {CBAR_BLUE, 0x000, 0x000, 0xfff}, >>> + {CBAR_PURPLE, 0x800, 0x000, 0x800}, >>> + {CBAR_BLACK, 0x000, 0x000, 0x000}, >>> +}; >>> + >>> +void hibmc_dp_set_cbar(struct hibmc_dp *dp, const struct hibmc_dp_cbar_cfg *cfg) >>> +{ >>> + struct hibmc_dp_dev *dp_dev = dp->dp_dev; >>> + struct hibmc_dp_color_raw raw_data; >>> + >>> + if (cfg->enable) { >>> + hibmc_dp_reg_write_field(dp_dev, HIBMC_DP_COLOR_BAR_CTRL, BIT(9), >>> + cfg->self_timing); >>> + hibmc_dp_reg_write_field(dp_dev, HIBMC_DP_COLOR_BAR_CTRL, GENMASK(8, 1), >>> + cfg->dynamic_rate); >>> + if (cfg->pattern == CBAR_COLOR_BAR) { >>> + hibmc_dp_reg_write_field(dp_dev, HIBMC_DP_COLOR_BAR_CTRL, BIT(10), 0); >>> + } else { >>> + raw_data = g_rgb_raw[cfg->pattern]; >>> + drm_dbg_dp(dp->drm_dev, "r:%x g:%x b:%x\n", raw_data.r_value, >>> + raw_data.g_value, raw_data.b_value); >>> + hibmc_dp_reg_write_field(dp_dev, HIBMC_DP_COLOR_BAR_CTRL, BIT(10), 1); >>> + hibmc_dp_reg_write_field(dp_dev, HIBMC_DP_COLOR_BAR_CTRL, GENMASK(23, 12), >>> + raw_data.r_value); >>> + hibmc_dp_reg_write_field(dp_dev, HIBMC_DP_COLOR_BAR_CTRL1, GENMASK(23, 12), >>> + raw_data.g_value); >>> + hibmc_dp_reg_write_field(dp_dev, HIBMC_DP_COLOR_BAR_CTRL1, GENMASK(11, 0), >>> + raw_data.b_value); >>> + } >>> + } >>> + >>> + hibmc_dp_reg_write_field(dp_dev, HIBMC_DP_COLOR_BAR_CTRL, BIT(0), cfg->enable); >>> + writel(HIBMC_DP_SYNC_EN_MASK, dp_dev->base + HIBMC_DP_TIMING_SYNC_CTRL); >>> +} >>> diff --git a/drivers/gpu/drm/hisilicon/hibmc/dp/dp_hw.h b/drivers/gpu/drm/hisilicon/hibmc/dp/dp_hw.h >>> index 53b6d0beecea..f2f59f2feb3c 100644 >>> --- a/drivers/gpu/drm/hisilicon/hibmc/dp/dp_hw.h >>> +++ b/drivers/gpu/drm/hisilicon/hibmc/dp/dp_hw.h >>> @@ -14,6 +14,44 @@ >>> struct hibmc_dp_dev; >>> +enum hibmc_dp_cbar_pattern { >>> + CBAR_COLOR_BAR, >>> + CBAR_WHITE, >>> + CBAR_RED, >>> + CBAR_ORANGE, >>> + CBAR_YELLOW, >>> + CBAR_GREEN, >>> + CBAR_CYAN, >>> + CBAR_BLUE, >>> + CBAR_PURPLE, >>> + CBAR_BLACK, >>> +}; >>> + >>> +struct hibmc_dp_color_raw { >>> + enum hibmc_dp_cbar_pattern pattern; >>> + u32 r_value; >>> + u32 g_value; >>> + u32 b_value; >>> +}; >>> + >>> +struct hibmc_dp_cbar_cfg { >>> + bool enable; >>> + bool self_timing; >>> + u8 dynamic_rate; /* 0:static, 1-255(frame):dynamic */ >>> + enum hibmc_dp_cbar_pattern pattern; >>> +}; >>> + >>> +enum hibmc_dp_hpd_status { >>> + HIBMC_DP_HPD_DETECTING, >>> + HIBMC_DP_HPD_IN, >>> + HIBMC_DP_HPD_OUT, >>> + HIBMC_DP_HPD_SHORT, /* Short hpd (irq_hpd) */ >>> + HIBMC_DP_HPD_DET_FAIL, >>> + HIBMC_DP_HPD_IN_SIMULATE, >>> + HIBMC_DP_HPD_OUT_SIMULATE, >>> + HIBMC_DP_HPD_SHORT_SIMULATE, >>> +}; >>> + >>> struct hibmc_dp { >>> struct hibmc_dp_dev *dp_dev; >>> struct drm_device *drm_dev; >>> @@ -21,10 +59,16 @@ struct hibmc_dp { >>> struct drm_connector connector; >>> void __iomem *mmio; >>> struct drm_dp_aux aux; >>> + struct hibmc_dp_cbar_cfg cfg; >>> + bool is_inited; >>> }; >>> int hibmc_dp_hw_init(struct hibmc_dp *dp); >>> int hibmc_dp_mode_set(struct hibmc_dp *dp, struct drm_display_mode *mode); >>> void hibmc_dp_display_en(struct hibmc_dp *dp, bool enable); >>> +int hibmc_dp_get_dpcd(struct hibmc_dp *dp); >>> +u8 hibmc_dp_get_link_rate(struct hibmc_dp *dp); >>> +u8 hibmc_dp_get_lanes(struct hibmc_dp *dp); >>> +void hibmc_dp_set_cbar(struct hibmc_dp *dp, const struct hibmc_dp_cbar_cfg *cfg); >>> #endif >>> diff --git a/drivers/gpu/drm/hisilicon/hibmc/dp/dp_link.c b/drivers/gpu/drm/hisilicon/hibmc/dp/dp_link.c >>> index 695cb9c0b643..20849f1ebd0c 100644 >>> --- a/drivers/gpu/drm/hisilicon/hibmc/dp/dp_link.c >>> +++ b/drivers/gpu/drm/hisilicon/hibmc/dp/dp_link.c >>> @@ -4,9 +4,11 @@ >>> #include <linux/delay.h> >>> #include <drm/drm_device.h> >>> #include <drm/drm_print.h> >>> + >>> #include "dp_comm.h" >>> #include "dp_reg.h" >>> #include "dp_phy.h" >>> +#include "dp_config.h" >>> #define HIBMC_EQ_MAX_RETRY 5 >>> @@ -42,11 +44,7 @@ static int hibmc_dp_link_training_configure(struct hibmc_dp_dev *dp) >>> return ret >= 0 ? -EIO : ret; >>> } >>> - ret = drm_dp_read_dpcd_caps(dp->aux, dp->dpcd); >>> - if (ret) >>> - drm_err(dp->dev, "dp aux read dpcd failed, ret: %d\n", ret); >>> - >>> - return ret; >>> + return 0; >>> } >>> static int hibmc_dp_link_set_pattern(struct hibmc_dp_dev *dp, int pattern) >>> @@ -189,15 +187,17 @@ static int hibmc_dp_link_training_cr(struct hibmc_dp_dev *dp) >>> bool level_changed; >>> u32 voltage_tries; >>> u32 cr_tries; >>> + u32 max_cr; >>> int ret; >>> /* >>> * DP 1.4 spec define 10 for maxtries value, for pre DP 1.4 version set a limit of 80 >>> * (4 voltage levels x 4 preemphasis levels x 5 identical voltage retries) >>> */ >>> + max_cr = dp->link.cap.rx_dpcd_revision >= DP_DPCD_REV_14 ? 10 : 80; >>> voltage_tries = 1; >>> - for (cr_tries = 0; cr_tries < 80; cr_tries++) { >>> + for (cr_tries = 0; cr_tries < max_cr; cr_tries++) { >>> drm_dp_link_train_clock_recovery_delay(dp->aux, dp->dpcd); >>> ret = drm_dp_dpcd_read_link_status(dp->aux, lane_status); >>> @@ -234,7 +234,7 @@ static int hibmc_dp_link_training_cr(struct hibmc_dp_dev *dp) >>> voltage_tries = level_changed ? 1 : voltage_tries + 1; >>> } >>> - drm_err(dp->dev, "dp link training clock recovery 80 times failed\n"); >>> + drm_err(dp->dev, "dp link training clock recovery %u times failed\n", max_cr); >>> dp->link.status.clock_recovered = false; >>> return 0; >>> @@ -244,9 +244,17 @@ static int hibmc_dp_link_training_channel_eq(struct hibmc_dp_dev *dp) >>> { >>> u8 lane_status[DP_LINK_STATUS_SIZE] = {0}; >>> u8 eq_tries; >>> + int tps; >>> int ret; >>> - ret = hibmc_dp_link_set_pattern(dp, DP_TRAINING_PATTERN_2); >>> + if (dp->link.cap.is_tps4) >>> + tps = DP_TRAINING_PATTERN_4; >>> + else if (dp->link.cap.is_tps3) >>> + tps = DP_TRAINING_PATTERN_3; >>> + else >>> + tps = DP_TRAINING_PATTERN_2; >>> + >>> + ret = hibmc_dp_link_set_pattern(dp, tps); >>> if (ret) >>> return ret; >>> @@ -313,11 +321,27 @@ static int hibmc_dp_link_downgrade_training_eq(struct hibmc_dp_dev *dp) >>> return hibmc_dp_link_reduce_rate(dp); >>> } >>> +static void hibmc_dp_update_caps(struct hibmc_dp_dev *dp) >>> +{ >>> + dp->link.cap.rx_dpcd_revision = dp->dpcd[DP_DPCD_REV]; >>> + >>> + dp->link.cap.is_tps3 = (dp->dpcd[DP_DPCD_REV] >= DP_DPCD_REV_13) && >>> + (dp->dpcd[DP_MAX_LANE_COUNT] & DP_TPS3_SUPPORTED); >>> + dp->link.cap.is_tps4 = (dp->dpcd[DP_DPCD_REV] >= DP_DPCD_REV_14) && >>> + (dp->dpcd[DP_MAX_DOWNSPREAD] & DP_TPS4_SUPPORTED); >>> +} >>> + >>> int hibmc_dp_link_training(struct hibmc_dp_dev *dp) >>> { >>> struct hibmc_dp_link *link = &dp->link; >>> int ret; >>> + ret = drm_dp_read_dpcd_caps(dp->aux, dp->dpcd); >>> + if (ret) >>> + drm_err(dp->dev, "dp aux read dpcd failed, ret: %d\n", ret); >>> + >>> + hibmc_dp_update_caps(dp); >>> + >>> while (true) { >>> ret = hibmc_dp_link_training_cr_pre(dp); >>> if (ret) >>> diff --git a/drivers/gpu/drm/hisilicon/hibmc/dp/dp_reg.h b/drivers/gpu/drm/hisilicon/hibmc/dp/dp_reg.h >>> index 99ba9c951c41..c43ad6b30c2c 100644 >>> --- a/drivers/gpu/drm/hisilicon/hibmc/dp/dp_reg.h >>> +++ b/drivers/gpu/drm/hisilicon/hibmc/dp/dp_reg.h >>> @@ -23,6 +23,8 @@ >>> #define HIBMC_DP_VIDEO_MSA1 0x11c >>> #define HIBMC_DP_VIDEO_MSA2 0x120 >>> #define HIBMC_DP_VIDEO_HORIZONTAL_SIZE 0X124 >>> +#define HIBMC_DP_COLOR_BAR_CTRL 0x260 >>> +#define HIBMC_DP_COLOR_BAR_CTRL1 0x264 >>> #define HIBMC_DP_TIMING_GEN_CONFIG0 0x26c >>> #define HIBMC_DP_TIMING_GEN_CONFIG2 0x274 >>> #define HIBMC_DP_TIMING_GEN_CONFIG3 0x278 >>> @@ -72,6 +74,6 @@ >>> #define HIBMC_DP_CFG_STREAM_TU_SYMBOL_FRAC_SIZE GENMASK(9, 6) >>> #define HIBMC_DP_CFG_STREAM_HTOTAL_SIZE GENMASK(31, 16) >>> #define HIBMC_DP_CFG_STREAM_HBLANK_SIZE GENMASK(15, 0) >>> -#define HIBMC_DP_CFG_STREAM_SYNC_CALIBRATION GENMASK(31, 20) >>> +#define HIBMC_DP_CFG_STREAM_SYNC_CALIBRATION GENMASK(31, 20) >>> #endif >>> diff --git a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_debugfs.c b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_debugfs.c >>> new file mode 100644 >>> index 000000000000..f6885399c2b3 >>> --- /dev/null >>> +++ b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_debugfs.c >>> @@ -0,0 +1,214 @@ >>> +// SPDX-License-Identifier: GPL-2.0-or-later >>> +// Copyright (c) 2024 Hisilicon Limited. >>> + >>> +#include <linux/debugfs.h> >>> +#include <linux/device.h> >>> +#include <linux/seq_file.h> >>> +#include <linux/pci.h> >>> + >>> +#include <drm/drm_drv.h> >>> +#include <drm/drm_file.h> >>> +#include <drm/drm_debugfs.h> >>> +#include <drm/drm_edid.h> >>> + >>> +#include "hibmc_drm_drv.h" >>> + >>> +static void hibmc_dump_edid(struct seq_file *m, const struct edid *edid) >>> +{ >>> + const struct detailed_pixel_timing *pixel_data; >>> + int i; >>> + >>> + seq_puts(m, "EDID:\n"); >>> + seq_printf(m, "\theader: 0x%02x 0x%02x 0x%02x 0x%02x 0x%02x 0x%02x 0x%02x 0x%02x\n", >>> + edid->header[0], edid->header[1], edid->header[2], edid->header[3], >>> + edid->header[4], edid->header[5], edid->header[6], edid->header[7]); >>> + >>> + seq_puts(m, "Vendor & product info:\n"); >>> + seq_printf(m, "\tmfg_id: 0x%02x 0x%02x\n", edid->mfg_id[0], edid->mfg_id[1]); >>> + seq_printf(m, "\tprod_code: 0x%02x 0x%02x\n", edid->prod_code[0], edid->prod_code[1]); >>> + seq_printf(m, "\tserial: 0x%08x\n", edid->serial); >>> + seq_printf(m, "\tmfg_week/year: 0x%02x 0x%02x\n", edid->mfg_week, edid->mfg_year); >>> + >>> + seq_puts(m, "EDID version:\n"); >>> + seq_printf(m, "\tversion: 0x%02x\n", edid->version); >>> + seq_printf(m, "\trevision: 0x%02x\n", edid->revision); >>> + >>> + seq_puts(m, "Display info:\n"); >>> + seq_printf(m, "\tinput: 0x%02x\n", edid->input); >>> + seq_printf(m, "\twidth_cm: 0x%02x\n", edid->width_cm); >>> + seq_printf(m, "\theight_cm: 0x%02x\n", edid->height_cm); >>> + seq_printf(m, "\tgamma: 0x%02x\n", edid->gamma); >>> + seq_printf(m, "\tfeatures: 0x%02x\n", edid->features); >>> + >>> + seq_puts(m, "Color characteristics:\n"); >>> + seq_printf(m, "\tred_green_lo: 0x%02x\n", edid->red_green_lo); >>> + seq_printf(m, "\tblue/black_white_lo: 0x%02x\n", *(&edid->red_green_lo) + 1); >>> + seq_printf(m, "\tred_x/y: 0x%02x 0x%02x\n", edid->red_x, edid->red_y); >>> + seq_printf(m, "\tgreen_x/y: 0x%02x 0x%02x\n", edid->green_x, edid->green_y); >>> + seq_printf(m, "\tblue_x/y: 0x%02x 0x%02x\n", edid->blue_x, edid->blue_y); >>> + seq_printf(m, "\twhite_x/y: 0x%02x 0x%02x\n", edid->white_x, edid->white_y); >>> + >>> + seq_puts(m, "Est. timings and mfg rsvd timings:\n"); >>> + seq_printf(m, "\test_timings_t1/2: 0x%02x 0x%02x\n", >>> + edid->established_timings.t1, edid->established_timings.t2); >>> + >>> + seq_puts(m, "Standard timings 1-8:\n"); >>> + for (i = 0; i < ARRAY_SIZE(edid->standard_timings); i++) { >>> + seq_printf(m, "\tstandard_timings[%d] hsize/vfreq_aspect: 0x%02x 0x%02x\n", >>> + i, edid->standard_timings[i].hsize, >>> + edid->standard_timings[i].vfreq_aspect); >>> + } >>> + >>> + seq_puts(m, "Detailing timings 1-4:\n"); >>> + for (i = 0; i < ARRAY_SIZE(edid->detailed_timings); i++) { >>> + pixel_data = &edid->detailed_timings[i].data.pixel_data; >>> + seq_printf(m, "\tdetailed_timing[%d] pixel_clock: 0x%04x\n", >>> + i, edid->detailed_timings[i].pixel_clock); >>> + seq_printf(m, "\tdetailed_timing[%d] hactive: %u\n", i, >>> + (pixel_data->hactive_hblank_hi & 0xf0) << 4 | pixel_data->hactive_lo); >>> + seq_printf(m, "\tdetailed_timing[%d] vactive: %u\n", i, >>> + (pixel_data->vactive_vblank_hi & 0xf0) << 4 | pixel_data->vactive_lo); >>> + } >>> + >>> + seq_puts(m, "Others:\n"); >>> + seq_printf(m, "\textensions: 0x%02x\n", edid->extensions); >>> + seq_printf(m, "\tchecksum: 0x%02x\n", edid->checksum); >>> +} >>> + >>> +static int hibmc_dp_edid_show(struct seq_file *m, void *arg) >>> +{ >>> + struct drm_info_node *node = m->private; >>> + struct drm_device *dev = node->minor->dev; >>> + struct hibmc_drm_private *priv = to_hibmc_drm_private(dev); >>> + struct edid *edid; >>> + char name[20]; >>> + int idx; >>> + >>> + if (!drm_dev_enter(dev, &idx)) >>> + return -ENODEV; >>> + >>> + edid = drm_get_edid(&priv->dp.connector, &priv->dp.aux.ddc); >>> + if (edid) { >>> + drm_edid_get_monitor_name(edid, name, ARRAY_SIZE(name)); >>> + seq_printf(m, "Monitor name: %s\n", name); >>> + hibmc_dump_edid(m, edid); >>> + kfree(edid); >>> + } else { >>> + seq_puts(m, "No connector available!\n"); >>> + } >>> + >>> + drm_dev_exit(idx); >>> + >>> + return 0; >>> +} >>> + >>> +static int hibmc_dp_show(struct seq_file *m, void *arg) >>> +{ >>> + struct drm_info_node *node = m->private; >>> + struct drm_device *dev = node->minor->dev; >>> + struct hibmc_drm_private *priv = to_hibmc_drm_private(dev); >>> + int idx; >>> + >>> + if (!drm_dev_enter(dev, &idx)) >>> + return -ENODEV; >>> + >>> + seq_printf(m, "enable lanes: %u\n", hibmc_dp_get_lanes(&priv->dp)); >>> + seq_printf(m, "link rate: %d\n", hibmc_dp_get_link_rate(&priv->dp) * 27); >>> + seq_printf(m, "vfresh: %d\n", drm_mode_vrefresh(&priv->crtc.mode)); >>> + seq_printf(m, "dpcd version: 0x%x\n", hibmc_dp_get_dpcd(&priv->dp)); >>> + >>> + drm_dev_exit(idx); >>> + >>> + return 0; >>> +} >>> + >>> +static ssize_t hibmc_control_write(struct file *file, const char __user *user_buf, >>> + size_t size, loff_t *ppos) >>> +{ >>> + struct hibmc_drm_private *priv = file_inode(file)->i_private; >>> + struct hibmc_dp_cbar_cfg *cfg = &priv->dp.cfg; >>> + u32 input = 0; >>> + int ret, idx; >>> + u8 val; >>> + >>> + ret = kstrtou32_from_user(user_buf, size, 0, &input); >>> + if (ret) >>> + return ret; >>> + >>> + val = FIELD_GET(GENMASK(13, 10), input); >>> + if (val > 9) >>> + return -EINVAL; >>> + cfg->pattern = val; >>> + cfg->enable = FIELD_GET(BIT(0), input); >>> + cfg->self_timing = FIELD_GET(BIT(1), input); >>> + cfg->dynamic_rate = FIELD_GET(GENMASK(9, 2), input); >>> + >>> + ret = drm_dev_enter(&priv->dev, &idx); >>> + if (!ret) >>> + return -ENODEV; >>> + >>> + hibmc_dp_set_cbar(&priv->dp, cfg); >>> + >>> + drm_dev_exit(idx); >>> + >>> + return size; >>> +} >>> + >>> +static int hibmc_dp_dbgfs_show(struct seq_file *m, void *arg) >>> +{ >>> + struct hibmc_drm_private *priv = m->private; >>> + struct hibmc_dp_cbar_cfg *cfg = &priv->dp.cfg; >>> + u32 output = 0; >>> + int idx; >>> + >>> + if (!drm_dev_enter(&priv->dev, &idx)) >>> + return -ENODEV; >>> + >>> + /* bit[0]: 0: enable colorbar, 1: disable colorbar >>> + * bit[1]: 0: timing follows XDP, 1: internal self timing >>> + * bit[2,9]: 0: static colorbar image, >>> + * 1~255: right shifting a type of color per (1~255)frames >>> + * bit[10,13]: 0~9: color bar, white, red, orange, >>> + * yellow, green, cyan, bule, pupper, black >>> + */ >>> + output = cfg->enable | (cfg->self_timing << 1) | >>> + (cfg->dynamic_rate << 2) | (cfg->pattern << 10); >>> + >>> + drm_dev_exit(idx); >>> + >>> + seq_printf(m, "hibmc dp colorbar cfg: %u\n", output); >>> + >>> + return 0; >>> +} >>> + >>> +static int hibmc_open(struct inode *inode, struct file *filp) >>> +{ >>> + return single_open(filp, hibmc_dp_dbgfs_show, inode->i_private); >>> +} >>> + >>> +static const struct file_operations hibmc_dbg_fops = { >>> + .owner = THIS_MODULE, >>> + .write = hibmc_control_write, >>> + .read = seq_read, >>> + .open = hibmc_open, >>> + .llseek = seq_lseek, >>> + .release = single_release, >>> +}; >>> + >>> +static struct drm_info_list hibmc_debugfs_list[] = { >>> + { "hibmc-dp", hibmc_dp_show }, >>> + { "hibmc-dp-edid", hibmc_dp_edid_show }, >>> +}; >>> + >>> +void hibmc_debugfs_register(struct hibmc_drm_private *priv) >>> +{ >>> + struct drm_connector *dp_conn = &priv->dp.connector; >>> + struct drm_minor *minor = priv->dev.primary; >>> + >>> + /* create the file in drm directory, so we don't need to remove manually */ >>> + debugfs_create_file("colorbar-cfg", 0200, >>> + dp_conn->debugfs_entry, priv, &hibmc_dbg_fops); >>> + >>> + drm_debugfs_create_files(hibmc_debugfs_list, ARRAY_SIZE(hibmc_debugfs_list), >>> + minor->debugfs_root, minor); >>> +} >>> diff --git a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_dp.c b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_dp.c >>> index fac8485a69d9..cc1f9ee0656f 100644 >>> --- a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_dp.c >>> +++ b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_dp.c >>> @@ -146,5 +146,7 @@ int hibmc_dp_init(struct hibmc_drm_private *priv) >>> drm_connector_attach_encoder(connector, encoder); >>> + dp->is_inited = true; >>> + >>> return 0; >>> } >>> diff --git a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c >>> index bade693d9730..3d4d5185c523 100644 >>> --- a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c >>> +++ b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c >>> @@ -352,6 +352,9 @@ static int hibmc_pci_probe(struct pci_dev *pdev, >>> goto err_unload; >>> } >>> + if (priv->dp.is_inited) >>> + hibmc_debugfs_register(priv); >> Please use debugfs_init() callback for that >> >> >>> + >>> drm_client_setup(dev, NULL); >>> return 0; >>> diff --git a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h >>> index 3ddd71aada66..ff61efb8a2ab 100644 >>> --- a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h >>> +++ b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h >>> @@ -69,4 +69,6 @@ int hibmc_ddc_create(struct drm_device *drm_dev, struct hibmc_vdac *connector); >>> int hibmc_dp_init(struct hibmc_drm_private *priv); >>> +void hibmc_debugfs_register(struct hibmc_drm_private *priv); >>> + >>> #endif >>> -- >>> 2.33.0 >>>