Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/C/Linux/drivers/gpu/drm/i915/display/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 104 kB image not shown  

Quelle  intel_sdvo.c   Sprache: C

 
/*
 * Copyright 2006 Dave Airlie <airlied@linux.ie>
 * Copyright © 2006-2007 Intel Corporation
 *   Jesse Barnes <jesse.barnes@intel.com>
 *
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 * and/or sell copies of the Software, and to permit persons to whom the
 * Software is furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice (including the next
 * paragraph) shall be included in all copies or substantial portions of the
 * Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 * DEALINGS IN THE SOFTWARE.
 *
 * Authors:
 * Eric Anholt <eric@anholt.net>
 */


#include <linux/delay.h>
#include <linux/export.h>
#include <linux/i2c.h>
#include <linux/slab.h>

#include <drm/display/drm_hdmi_helper.h>
#include <drm/drm_atomic_helper.h>
#include <drm/drm_crtc.h>
#include <drm/drm_edid.h>
#include <drm/drm_eld.h>
#include <drm/drm_print.h>
#include <drm/drm_probe_helper.h>

#include "intel_atomic.h"
#include "intel_audio.h"
#include "intel_connector.h"
#include "intel_crtc.h"
#include "intel_de.h"
#include "intel_display_driver.h"
#include "intel_display_regs.h"
#include "intel_display_types.h"
#include "intel_fdi.h"
#include "intel_fifo_underrun.h"
#include "intel_gmbus.h"
#include "intel_hdmi.h"
#include "intel_hotplug.h"
#include "intel_panel.h"
#include "intel_sdvo.h"
#include "intel_sdvo_regs.h"

#define SDVO_TMDS_MASK (SDVO_OUTPUT_TMDS0 | SDVO_OUTPUT_TMDS1)
#define SDVO_RGB_MASK  (SDVO_OUTPUT_RGB0 | SDVO_OUTPUT_RGB1)
#define SDVO_LVDS_MASK (SDVO_OUTPUT_LVDS0 | SDVO_OUTPUT_LVDS1)
#define SDVO_TV_MASK   (SDVO_OUTPUT_CVBS0 | SDVO_OUTPUT_SVID0 | SDVO_OUTPUT_YPRPB0)

#define SDVO_OUTPUT_MASK (SDVO_TMDS_MASK | SDVO_RGB_MASK | SDVO_LVDS_MASK | SDVO_TV_MASK)

#define IS_TV(c)  ((c)->output_flag & SDVO_TV_MASK)
#define IS_TMDS(c)  ((c)->output_flag & SDVO_TMDS_MASK)
#define IS_LVDS(c)  ((c)->output_flag & SDVO_LVDS_MASK)
#define IS_TV_OR_LVDS(c) ((c)->output_flag & (SDVO_TV_MASK | SDVO_LVDS_MASK))
#define IS_DIGITAL(c)  ((c)->output_flag & (SDVO_TMDS_MASK | SDVO_LVDS_MASK))

#define HAS_DDC(c)  ((c)->output_flag & (SDVO_RGB_MASK | SDVO_TMDS_MASK | \
           SDVO_LVDS_MASK))

static const char * const tv_format_names[] = {
 "NTSC_M"   , "NTSC_J"  , "NTSC_443",
 "PAL_B"    , "PAL_D"   , "PAL_G"   ,
 "PAL_H"    , "PAL_I"   , "PAL_M"   ,
 "PAL_N"    , "PAL_NC"  , "PAL_60"  ,
 "SECAM_B"  , "SECAM_D" , "SECAM_G" ,
 "SECAM_K"  , "SECAM_K1""SECAM_L" ,
 "SECAM_60"
};

#define TV_FORMAT_NUM  ARRAY_SIZE(tv_format_names)

struct intel_sdvo;

struct intel_sdvo_ddc {
 struct i2c_adapter ddc;
 struct intel_sdvo *sdvo;
 u8 ddc_bus;
};

struct intel_sdvo {
 struct intel_encoder base;

 struct i2c_adapter *i2c;
 u8 target_addr;

 struct intel_sdvo_ddc ddc[3];

 /* Register for the SDVO device: SDVOB or SDVOC */
 i915_reg_t sdvo_reg;

 /*
 * Capabilities of the SDVO device returned by
 * intel_sdvo_get_capabilities()
 */

 struct intel_sdvo_caps caps;

 u8 colorimetry_cap;

 /* Pixel clock limitations reported by the SDVO device, in kHz */
 int pixel_clock_min, pixel_clock_max;

 /*
 * Hotplug activation bits for this device
 */

 u16 hotplug_active;

 /*
 * the sdvo flag gets lost in round trip: dtd->adjusted_mode->dtd
 */

 u8 dtd_sdvo_flags;
};

struct intel_sdvo_connector {
 struct intel_connector base;

 /* Mark the type of connector */
 u16 output_flag;

 /* This contains all current supported TV format */
 u8 tv_format_supported[TV_FORMAT_NUM];
 int   format_supported_num;
 struct drm_property *tv_format;

 /* add the property for the SDVO-TV */
 struct drm_property *left;
 struct drm_property *right;
 struct drm_property *top;
 struct drm_property *bottom;
 struct drm_property *hpos;
 struct drm_property *vpos;
 struct drm_property *contrast;
 struct drm_property *saturation;
 struct drm_property *hue;
 struct drm_property *sharpness;
 struct drm_property *flicker_filter;
 struct drm_property *flicker_filter_adaptive;
 struct drm_property *flicker_filter_2d;
 struct drm_property *tv_chroma_filter;
 struct drm_property *tv_luma_filter;
 struct drm_property *dot_crawl;

 /* add the property for the SDVO-TV/LVDS */
 struct drm_property *brightness;

 /* this is to get the range of margin.*/
 u32 max_hscan, max_vscan;

 /**
 * This is set if we treat the device as HDMI, instead of DVI.
 */

 bool is_hdmi;
};

struct intel_sdvo_connector_state {
 /* base.base: tv.saturation/contrast/hue/brightness */
 struct intel_digital_connector_state base;

 struct {
  unsigned overscan_h, overscan_v, hpos, vpos, sharpness;
  unsigned flicker_filter, flicker_filter_2d, flicker_filter_adaptive;
  unsigned chroma_filter, luma_filter, dot_crawl;
 } tv;
};

static struct intel_sdvo *to_sdvo(struct intel_encoder *encoder)
{
 return container_of(encoder, struct intel_sdvo, base);
}

static struct intel_sdvo *intel_attached_sdvo(struct intel_connector *connector)
{
 return to_sdvo(intel_attached_encoder(connector));
}

static struct intel_sdvo_connector *
to_intel_sdvo_connector(struct drm_connector *connector)
{
 return container_of(connector, struct intel_sdvo_connector, base.base);
}

#define to_intel_sdvo_connector_state(conn_state) \
 container_of_const((conn_state), struct intel_sdvo_connector_state, base.base)

static bool
intel_sdvo_output_setup(struct intel_sdvo *intel_sdvo);
static bool
intel_sdvo_tv_create_property(struct intel_sdvo *intel_sdvo,
         struct intel_sdvo_connector *intel_sdvo_connector,
         int type);
static bool
intel_sdvo_create_enhance_property(struct intel_sdvo *intel_sdvo,
       struct intel_sdvo_connector *intel_sdvo_connector);

/*
 * Writes the SDVOB or SDVOC with the given value, but always writes both
 * SDVOB and SDVOC to work around apparent hardware issues (according to
 * comments in the BIOS).
 */

static void intel_sdvo_write_sdvox(struct intel_sdvo *intel_sdvo, u32 val)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);
 u32 bval = val, cval = val;
 int i;

 if (HAS_PCH_SPLIT(display)) {
  intel_de_write(display, intel_sdvo->sdvo_reg, val);
  intel_de_posting_read(display, intel_sdvo->sdvo_reg);
  /*
 * HW workaround, need to write this twice for issue
 * that may result in first write getting masked.
 */

  if (HAS_PCH_IBX(display)) {
   intel_de_write(display, intel_sdvo->sdvo_reg, val);
   intel_de_posting_read(display, intel_sdvo->sdvo_reg);
  }
  return;
 }

 if (intel_sdvo->base.port == PORT_B)
  cval = intel_de_read(display, GEN3_SDVOC);
 else
  bval = intel_de_read(display, GEN3_SDVOB);

 /*
 * Write the registers twice for luck. Sometimes,
 * writing them only once doesn't appear to 'stick'.
 * The BIOS does this too. Yay, magic
 */

 for (i = 0; i < 2; i++) {
  intel_de_write(display, GEN3_SDVOB, bval);
  intel_de_posting_read(display, GEN3_SDVOB);

  intel_de_write(display, GEN3_SDVOC, cval);
  intel_de_posting_read(display, GEN3_SDVOC);
 }
}

static bool intel_sdvo_read_byte(struct intel_sdvo *intel_sdvo, u8 addr, u8 *ch)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);
 struct i2c_msg msgs[] = {
  {
   .addr = intel_sdvo->target_addr,
   .flags = 0,
   .len = 1,
   .buf = &addr,
  },
  {
   .addr = intel_sdvo->target_addr,
   .flags = I2C_M_RD,
   .len = 1,
   .buf = ch,
  }
 };
 int ret;

 if ((ret = i2c_transfer(intel_sdvo->i2c, msgs, 2)) == 2)
  return true;

 drm_dbg_kms(display->drm, "i2c transfer returned %d\n", ret);
 return false;
}

#define SDVO_CMD_NAME_ENTRY(cmd_) { .cmd = SDVO_CMD_ ## cmd_, .name = #cmd_ }

/** Mapping of command numbers to names, for debug output */
static const struct {
 u8 cmd;
 const char *name;
} __packed sdvo_cmd_names[] = {
 SDVO_CMD_NAME_ENTRY(RESET),
 SDVO_CMD_NAME_ENTRY(GET_DEVICE_CAPS),
 SDVO_CMD_NAME_ENTRY(GET_FIRMWARE_REV),
 SDVO_CMD_NAME_ENTRY(GET_TRAINED_INPUTS),
 SDVO_CMD_NAME_ENTRY(GET_ACTIVE_OUTPUTS),
 SDVO_CMD_NAME_ENTRY(SET_ACTIVE_OUTPUTS),
 SDVO_CMD_NAME_ENTRY(GET_IN_OUT_MAP),
 SDVO_CMD_NAME_ENTRY(SET_IN_OUT_MAP),
 SDVO_CMD_NAME_ENTRY(GET_ATTACHED_DISPLAYS),
 SDVO_CMD_NAME_ENTRY(GET_HOT_PLUG_SUPPORT),
 SDVO_CMD_NAME_ENTRY(SET_ACTIVE_HOT_PLUG),
 SDVO_CMD_NAME_ENTRY(GET_ACTIVE_HOT_PLUG),
 SDVO_CMD_NAME_ENTRY(GET_INTERRUPT_EVENT_SOURCE),
 SDVO_CMD_NAME_ENTRY(SET_TARGET_INPUT),
 SDVO_CMD_NAME_ENTRY(SET_TARGET_OUTPUT),
 SDVO_CMD_NAME_ENTRY(GET_INPUT_TIMINGS_PART1),
 SDVO_CMD_NAME_ENTRY(GET_INPUT_TIMINGS_PART2),
 SDVO_CMD_NAME_ENTRY(SET_INPUT_TIMINGS_PART1),
 SDVO_CMD_NAME_ENTRY(SET_INPUT_TIMINGS_PART2),
 SDVO_CMD_NAME_ENTRY(SET_OUTPUT_TIMINGS_PART1),
 SDVO_CMD_NAME_ENTRY(SET_OUTPUT_TIMINGS_PART2),
 SDVO_CMD_NAME_ENTRY(GET_OUTPUT_TIMINGS_PART1),
 SDVO_CMD_NAME_ENTRY(GET_OUTPUT_TIMINGS_PART2),
 SDVO_CMD_NAME_ENTRY(CREATE_PREFERRED_INPUT_TIMING),
 SDVO_CMD_NAME_ENTRY(GET_PREFERRED_INPUT_TIMING_PART1),
 SDVO_CMD_NAME_ENTRY(GET_PREFERRED_INPUT_TIMING_PART2),
 SDVO_CMD_NAME_ENTRY(GET_INPUT_PIXEL_CLOCK_RANGE),
 SDVO_CMD_NAME_ENTRY(GET_OUTPUT_PIXEL_CLOCK_RANGE),
 SDVO_CMD_NAME_ENTRY(GET_SUPPORTED_CLOCK_RATE_MULTS),
 SDVO_CMD_NAME_ENTRY(GET_CLOCK_RATE_MULT),
 SDVO_CMD_NAME_ENTRY(SET_CLOCK_RATE_MULT),
 SDVO_CMD_NAME_ENTRY(GET_SUPPORTED_TV_FORMATS),
 SDVO_CMD_NAME_ENTRY(GET_TV_FORMAT),
 SDVO_CMD_NAME_ENTRY(SET_TV_FORMAT),
 SDVO_CMD_NAME_ENTRY(GET_SUPPORTED_POWER_STATES),
 SDVO_CMD_NAME_ENTRY(GET_POWER_STATE),
 SDVO_CMD_NAME_ENTRY(SET_ENCODER_POWER_STATE),
 SDVO_CMD_NAME_ENTRY(SET_DISPLAY_POWER_STATE),
 SDVO_CMD_NAME_ENTRY(SET_CONTROL_BUS_SWITCH),
 SDVO_CMD_NAME_ENTRY(GET_SDTV_RESOLUTION_SUPPORT),
 SDVO_CMD_NAME_ENTRY(GET_SCALED_HDTV_RESOLUTION_SUPPORT),
 SDVO_CMD_NAME_ENTRY(GET_SUPPORTED_ENHANCEMENTS),

 /* Add the op code for SDVO enhancements */
 SDVO_CMD_NAME_ENTRY(GET_MAX_HPOS),
 SDVO_CMD_NAME_ENTRY(GET_HPOS),
 SDVO_CMD_NAME_ENTRY(SET_HPOS),
 SDVO_CMD_NAME_ENTRY(GET_MAX_VPOS),
 SDVO_CMD_NAME_ENTRY(GET_VPOS),
 SDVO_CMD_NAME_ENTRY(SET_VPOS),
 SDVO_CMD_NAME_ENTRY(GET_MAX_SATURATION),
 SDVO_CMD_NAME_ENTRY(GET_SATURATION),
 SDVO_CMD_NAME_ENTRY(SET_SATURATION),
 SDVO_CMD_NAME_ENTRY(GET_MAX_HUE),
 SDVO_CMD_NAME_ENTRY(GET_HUE),
 SDVO_CMD_NAME_ENTRY(SET_HUE),
 SDVO_CMD_NAME_ENTRY(GET_MAX_CONTRAST),
 SDVO_CMD_NAME_ENTRY(GET_CONTRAST),
 SDVO_CMD_NAME_ENTRY(SET_CONTRAST),
 SDVO_CMD_NAME_ENTRY(GET_MAX_BRIGHTNESS),
 SDVO_CMD_NAME_ENTRY(GET_BRIGHTNESS),
 SDVO_CMD_NAME_ENTRY(SET_BRIGHTNESS),
 SDVO_CMD_NAME_ENTRY(GET_MAX_OVERSCAN_H),
 SDVO_CMD_NAME_ENTRY(GET_OVERSCAN_H),
 SDVO_CMD_NAME_ENTRY(SET_OVERSCAN_H),
 SDVO_CMD_NAME_ENTRY(GET_MAX_OVERSCAN_V),
 SDVO_CMD_NAME_ENTRY(GET_OVERSCAN_V),
 SDVO_CMD_NAME_ENTRY(SET_OVERSCAN_V),
 SDVO_CMD_NAME_ENTRY(GET_MAX_FLICKER_FILTER),
 SDVO_CMD_NAME_ENTRY(GET_FLICKER_FILTER),
 SDVO_CMD_NAME_ENTRY(SET_FLICKER_FILTER),
 SDVO_CMD_NAME_ENTRY(GET_MAX_FLICKER_FILTER_ADAPTIVE),
 SDVO_CMD_NAME_ENTRY(GET_FLICKER_FILTER_ADAPTIVE),
 SDVO_CMD_NAME_ENTRY(SET_FLICKER_FILTER_ADAPTIVE),
 SDVO_CMD_NAME_ENTRY(GET_MAX_FLICKER_FILTER_2D),
 SDVO_CMD_NAME_ENTRY(GET_FLICKER_FILTER_2D),
 SDVO_CMD_NAME_ENTRY(SET_FLICKER_FILTER_2D),
 SDVO_CMD_NAME_ENTRY(GET_MAX_SHARPNESS),
 SDVO_CMD_NAME_ENTRY(GET_SHARPNESS),
 SDVO_CMD_NAME_ENTRY(SET_SHARPNESS),
 SDVO_CMD_NAME_ENTRY(GET_DOT_CRAWL),
 SDVO_CMD_NAME_ENTRY(SET_DOT_CRAWL),
 SDVO_CMD_NAME_ENTRY(GET_MAX_TV_CHROMA_FILTER),
 SDVO_CMD_NAME_ENTRY(GET_TV_CHROMA_FILTER),
 SDVO_CMD_NAME_ENTRY(SET_TV_CHROMA_FILTER),
 SDVO_CMD_NAME_ENTRY(GET_MAX_TV_LUMA_FILTER),
 SDVO_CMD_NAME_ENTRY(GET_TV_LUMA_FILTER),
 SDVO_CMD_NAME_ENTRY(SET_TV_LUMA_FILTER),

 /* HDMI op code */
 SDVO_CMD_NAME_ENTRY(GET_SUPP_ENCODE),
 SDVO_CMD_NAME_ENTRY(GET_ENCODE),
 SDVO_CMD_NAME_ENTRY(SET_ENCODE),
 SDVO_CMD_NAME_ENTRY(SET_PIXEL_REPLI),
 SDVO_CMD_NAME_ENTRY(GET_PIXEL_REPLI),
 SDVO_CMD_NAME_ENTRY(GET_COLORIMETRY_CAP),
 SDVO_CMD_NAME_ENTRY(SET_COLORIMETRY),
 SDVO_CMD_NAME_ENTRY(GET_COLORIMETRY),
 SDVO_CMD_NAME_ENTRY(GET_AUDIO_ENCRYPT_PREFER),
 SDVO_CMD_NAME_ENTRY(SET_AUDIO_STAT),
 SDVO_CMD_NAME_ENTRY(GET_AUDIO_STAT),
 SDVO_CMD_NAME_ENTRY(GET_HBUF_INDEX),
 SDVO_CMD_NAME_ENTRY(SET_HBUF_INDEX),
 SDVO_CMD_NAME_ENTRY(GET_HBUF_INFO),
 SDVO_CMD_NAME_ENTRY(GET_HBUF_AV_SPLIT),
 SDVO_CMD_NAME_ENTRY(SET_HBUF_AV_SPLIT),
 SDVO_CMD_NAME_ENTRY(GET_HBUF_TXRATE),
 SDVO_CMD_NAME_ENTRY(SET_HBUF_TXRATE),
 SDVO_CMD_NAME_ENTRY(SET_HBUF_DATA),
 SDVO_CMD_NAME_ENTRY(GET_HBUF_DATA),
};

#undef SDVO_CMD_NAME_ENTRY

static const char *sdvo_cmd_name(u8 cmd)
{
 int i;

 for (i = 0; i < ARRAY_SIZE(sdvo_cmd_names); i++) {
  if (cmd == sdvo_cmd_names[i].cmd)
   return sdvo_cmd_names[i].name;
 }

 return NULL;
}

#define SDVO_NAME(svdo) ((svdo)->base.port == PORT_B ? "SDVOB" : "SDVOC")

static void intel_sdvo_debug_write(struct intel_sdvo *intel_sdvo, u8 cmd,
       const void *args, int args_len)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);
 const char *cmd_name;
 int i, pos = 0;
 char buffer[64];

#define BUF_PRINT(args...) \
 pos += snprintf(buffer + pos, max_t(intsizeof(buffer) - pos, 0), args)

 for (i = 0; i < args_len; i++) {
  BUF_PRINT("%02X ", ((u8 *)args)[i]);
 }
 for (; i < 8; i++) {
  BUF_PRINT(" ");
 }

 cmd_name = sdvo_cmd_name(cmd);
 if (cmd_name)
  BUF_PRINT("(%s)", cmd_name);
 else
  BUF_PRINT("(%02X)", cmd);

 drm_WARN_ON(display->drm, pos >= sizeof(buffer) - 1);
#undef BUF_PRINT

 drm_dbg_kms(display->drm, "%s: W: %02X %s\n", SDVO_NAME(intel_sdvo),
      cmd, buffer);
}

static const char * const cmd_status_names[] = {
 [SDVO_CMD_STATUS_POWER_ON] = "Power on",
 [SDVO_CMD_STATUS_SUCCESS] = "Success",
 [SDVO_CMD_STATUS_NOTSUPP] = "Not supported",
 [SDVO_CMD_STATUS_INVALID_ARG] = "Invalid arg",
 [SDVO_CMD_STATUS_PENDING] = "Pending",
 [SDVO_CMD_STATUS_TARGET_NOT_SPECIFIED] = "Target not specified",
 [SDVO_CMD_STATUS_SCALING_NOT_SUPP] = "Scaling not supported",
};

static const char *sdvo_cmd_status(u8 status)
{
 if (status < ARRAY_SIZE(cmd_status_names))
  return cmd_status_names[status];
 else
  return NULL;
}

static bool __intel_sdvo_write_cmd(struct intel_sdvo *intel_sdvo, u8 cmd,
       const void *args, int args_len,
       bool unlocked)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);
 u8 *buf, status;
 struct i2c_msg *msgs;
 int i, ret = true;

 /* Would be simpler to allocate both in one go ? */
 buf = kzalloc(args_len * 2 + 2, GFP_KERNEL);
 if (!buf)
  return false;

 msgs = kcalloc(args_len + 3, sizeof(*msgs), GFP_KERNEL);
 if (!msgs) {
  kfree(buf);
  return false;
 }

 intel_sdvo_debug_write(intel_sdvo, cmd, args, args_len);

 for (i = 0; i < args_len; i++) {
  msgs[i].addr = intel_sdvo->target_addr;
  msgs[i].flags = 0;
  msgs[i].len = 2;
  msgs[i].buf = buf + 2 *i;
  buf[2*i + 0] = SDVO_I2C_ARG_0 - i;
  buf[2*i + 1] = ((u8*)args)[i];
 }
 msgs[i].addr = intel_sdvo->target_addr;
 msgs[i].flags = 0;
 msgs[i].len = 2;
 msgs[i].buf = buf + 2*i;
 buf[2*i + 0] = SDVO_I2C_OPCODE;
 buf[2*i + 1] = cmd;

 /* the following two are to read the response */
 status = SDVO_I2C_CMD_STATUS;
 msgs[i+1].addr = intel_sdvo->target_addr;
 msgs[i+1].flags = 0;
 msgs[i+1].len = 1;
 msgs[i+1].buf = &status;

 msgs[i+2].addr = intel_sdvo->target_addr;
 msgs[i+2].flags = I2C_M_RD;
 msgs[i+2].len = 1;
 msgs[i+2].buf = &status;

 if (unlocked)
  ret = i2c_transfer(intel_sdvo->i2c, msgs, i+3);
 else
  ret = __i2c_transfer(intel_sdvo->i2c, msgs, i+3);
 if (ret < 0) {
  drm_dbg_kms(display->drm, "I2c transfer returned %d\n", ret);
  ret = false;
  goto out;
 }
 if (ret != i+3) {
  /* failure in I2C transfer */
  drm_dbg_kms(display->drm, "I2c transfer returned %d/%d\n", ret, i + 3);
  ret = false;
 }

out:
 kfree(msgs);
 kfree(buf);
 return ret;
}

static bool intel_sdvo_write_cmd(struct intel_sdvo *intel_sdvo, u8 cmd,
     const void *args, int args_len)
{
 return __intel_sdvo_write_cmd(intel_sdvo, cmd, args, args_len, true);
}

static bool intel_sdvo_read_response(struct intel_sdvo *intel_sdvo,
         void *response, int response_len)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);
 const char *cmd_status;
 u8 retry = 15; /* 5 quick checks, followed by 10 long checks */
 u8 status;
 int i, pos = 0;
 char buffer[64];

 buffer[0] = '\0';

 /*
 * The documentation states that all commands will be
 * processed within 15µs, and that we need only poll
 * the status byte a maximum of 3 times in order for the
 * command to be complete.
 *
 * Check 5 times in case the hardware failed to read the docs.
 *
 * Also beware that the first response by many devices is to
 * reply PENDING and stall for time. TVs are notorious for
 * requiring longer than specified to complete their replies.
 * Originally (in the DDX long ago), the delay was only ever 15ms
 * with an additional delay of 30ms applied for TVs added later after
 * many experiments. To accommodate both sets of delays, we do a
 * sequence of slow checks if the device is falling behind and fails
 * to reply within 5*15µs.
 */

 if (!intel_sdvo_read_byte(intel_sdvo,
      SDVO_I2C_CMD_STATUS,
      &status))
  goto log_fail;

 while ((status == SDVO_CMD_STATUS_PENDING ||
  status == SDVO_CMD_STATUS_TARGET_NOT_SPECIFIED) && --retry) {
  if (retry < 10)
   msleep(15);
  else
   udelay(15);

  if (!intel_sdvo_read_byte(intel_sdvo,
       SDVO_I2C_CMD_STATUS,
       &status))
   goto log_fail;
 }

#define BUF_PRINT(args...) \
 pos += snprintf(buffer + pos, max_t(intsizeof(buffer) - pos, 0), args)

 cmd_status = sdvo_cmd_status(status);
 if (cmd_status)
  BUF_PRINT("(%s)", cmd_status);
 else
  BUF_PRINT("(??? %d)", status);

 if (status != SDVO_CMD_STATUS_SUCCESS)
  goto log_fail;

 /* Read the command response */
 for (i = 0; i < response_len; i++) {
  if (!intel_sdvo_read_byte(intel_sdvo,
       SDVO_I2C_RETURN_0 + i,
       &((u8 *)response)[i]))
   goto log_fail;
  BUF_PRINT(" %02X", ((u8 *)response)[i]);
 }

 drm_WARN_ON(display->drm, pos >= sizeof(buffer) - 1);
#undef BUF_PRINT

 drm_dbg_kms(display->drm, "%s: R: %s\n",
      SDVO_NAME(intel_sdvo), buffer);
 return true;

log_fail:
 drm_dbg_kms(display->drm, "%s: R: ... failed %s\n",
      SDVO_NAME(intel_sdvo), buffer);
 return false;
}

static int intel_sdvo_get_pixel_multiplier(const struct drm_display_mode *adjusted_mode)
{
 if (adjusted_mode->crtc_clock >= 100000)
  return 1;
 else if (adjusted_mode->crtc_clock >= 50000)
  return 2;
 else
  return 4;
}

static bool __intel_sdvo_set_control_bus_switch(struct intel_sdvo *intel_sdvo,
      u8 ddc_bus)
{
 /* This must be the immediately preceding write before the i2c xfer */
 return __intel_sdvo_write_cmd(intel_sdvo,
          SDVO_CMD_SET_CONTROL_BUS_SWITCH,
          &ddc_bus, 1, false);
}

static bool intel_sdvo_set_value(struct intel_sdvo *intel_sdvo, u8 cmd, const void *data, int len)
{
 if (!intel_sdvo_write_cmd(intel_sdvo, cmd, data, len))
  return false;

 return intel_sdvo_read_response(intel_sdvo, NULL, 0);
}

static bool
intel_sdvo_get_value(struct intel_sdvo *intel_sdvo, u8 cmd, void *value, int len)
{
 if (!intel_sdvo_write_cmd(intel_sdvo, cmd, NULL, 0))
  return false;

 return intel_sdvo_read_response(intel_sdvo, value, len);
}

static bool intel_sdvo_set_target_input(struct intel_sdvo *intel_sdvo)
{
 struct intel_sdvo_set_target_input_args targets = {};
 return intel_sdvo_set_value(intel_sdvo,
        SDVO_CMD_SET_TARGET_INPUT,
        &targets, sizeof(targets));
}

/*
 * Return whether each input is trained.
 *
 * This function is making an assumption about the layout of the response,
 * which should be checked against the docs.
 */

static bool intel_sdvo_get_trained_inputs(struct intel_sdvo *intel_sdvo, bool *input_1bool *input_2)
{
 struct intel_sdvo_get_trained_inputs_response response;

 BUILD_BUG_ON(sizeof(response) != 1);
 if (!intel_sdvo_get_value(intel_sdvo, SDVO_CMD_GET_TRAINED_INPUTS,
      &response, sizeof(response)))
  return false;

 *input_1 = response.input0_trained;
 *input_2 = response.input1_trained;
 return true;
}

static bool intel_sdvo_set_active_outputs(struct intel_sdvo *intel_sdvo,
       u16 outputs)
{
 return intel_sdvo_set_value(intel_sdvo,
        SDVO_CMD_SET_ACTIVE_OUTPUTS,
        &outputs, sizeof(outputs));
}

static bool intel_sdvo_get_active_outputs(struct intel_sdvo *intel_sdvo,
       u16 *outputs)
{
 return intel_sdvo_get_value(intel_sdvo,
        SDVO_CMD_GET_ACTIVE_OUTPUTS,
        outputs, sizeof(*outputs));
}

static bool intel_sdvo_set_encoder_power_state(struct intel_sdvo *intel_sdvo,
            int mode)
{
 u8 state = SDVO_ENCODER_STATE_ON;

 switch (mode) {
 case DRM_MODE_DPMS_ON:
  state = SDVO_ENCODER_STATE_ON;
  break;
 case DRM_MODE_DPMS_STANDBY:
  state = SDVO_ENCODER_STATE_STANDBY;
  break;
 case DRM_MODE_DPMS_SUSPEND:
  state = SDVO_ENCODER_STATE_SUSPEND;
  break;
 case DRM_MODE_DPMS_OFF:
  state = SDVO_ENCODER_STATE_OFF;
  break;
 }

 return intel_sdvo_set_value(intel_sdvo,
        SDVO_CMD_SET_ENCODER_POWER_STATE, &state, sizeof(state));
}

static bool intel_sdvo_get_input_pixel_clock_range(struct intel_sdvo *intel_sdvo,
         int *clock_min,
         int *clock_max)
{
 struct intel_sdvo_pixel_clock_range clocks;

 BUILD_BUG_ON(sizeof(clocks) != 4);
 if (!intel_sdvo_get_value(intel_sdvo,
      SDVO_CMD_GET_INPUT_PIXEL_CLOCK_RANGE,
      &clocks, sizeof(clocks)))
  return false;

 /* Convert the values from units of 10 kHz to kHz. */
 *clock_min = clocks.min * 10;
 *clock_max = clocks.max * 10;
 return true;
}

static bool intel_sdvo_set_target_output(struct intel_sdvo *intel_sdvo,
      u16 outputs)
{
 return intel_sdvo_set_value(intel_sdvo,
        SDVO_CMD_SET_TARGET_OUTPUT,
        &outputs, sizeof(outputs));
}

static bool intel_sdvo_set_timing(struct intel_sdvo *intel_sdvo, u8 cmd,
      struct intel_sdvo_dtd *dtd)
{
 return intel_sdvo_set_value(intel_sdvo, cmd, &dtd->part1, sizeof(dtd->part1)) &&
  intel_sdvo_set_value(intel_sdvo, cmd + 1, &dtd->part2, sizeof(dtd->part2));
}

static bool intel_sdvo_get_timing(struct intel_sdvo *intel_sdvo, u8 cmd,
      struct intel_sdvo_dtd *dtd)
{
 return intel_sdvo_get_value(intel_sdvo, cmd, &dtd->part1, sizeof(dtd->part1)) &&
  intel_sdvo_get_value(intel_sdvo, cmd + 1, &dtd->part2, sizeof(dtd->part2));
}

static bool intel_sdvo_set_input_timing(struct intel_sdvo *intel_sdvo,
     struct intel_sdvo_dtd *dtd)
{
 return intel_sdvo_set_timing(intel_sdvo,
         SDVO_CMD_SET_INPUT_TIMINGS_PART1, dtd);
}

static bool intel_sdvo_set_output_timing(struct intel_sdvo *intel_sdvo,
      struct intel_sdvo_dtd *dtd)
{
 return intel_sdvo_set_timing(intel_sdvo,
         SDVO_CMD_SET_OUTPUT_TIMINGS_PART1, dtd);
}

static bool intel_sdvo_get_input_timing(struct intel_sdvo *intel_sdvo,
     struct intel_sdvo_dtd *dtd)
{
 return intel_sdvo_get_timing(intel_sdvo,
         SDVO_CMD_GET_INPUT_TIMINGS_PART1, dtd);
}

static bool
intel_sdvo_create_preferred_input_timing(struct intel_sdvo *intel_sdvo,
      struct intel_sdvo_connector *intel_sdvo_connector,
      const struct drm_display_mode *mode)
{
 struct intel_sdvo_preferred_input_timing_args args;

 memset(&args, 0, sizeof(args));
 args.clock = mode->clock / 10;
 args.width = mode->hdisplay;
 args.height = mode->vdisplay;
 args.interlace = 0;

 if (IS_LVDS(intel_sdvo_connector)) {
  const struct drm_display_mode *fixed_mode =
   intel_panel_fixed_mode(&intel_sdvo_connector->base, mode);

  if (fixed_mode->hdisplay != args.width ||
      fixed_mode->vdisplay != args.height)
   args.scaled = 1;
 }

 return intel_sdvo_set_value(intel_sdvo,
        SDVO_CMD_CREATE_PREFERRED_INPUT_TIMING,
        &args, sizeof(args));
}

static bool intel_sdvo_get_preferred_input_timing(struct intel_sdvo *intel_sdvo,
        struct intel_sdvo_dtd *dtd)
{
 BUILD_BUG_ON(sizeof(dtd->part1) != 8);
 BUILD_BUG_ON(sizeof(dtd->part2) != 8);
 return intel_sdvo_get_value(intel_sdvo, SDVO_CMD_GET_PREFERRED_INPUT_TIMING_PART1,
        &dtd->part1, sizeof(dtd->part1)) &&
  intel_sdvo_get_value(intel_sdvo, SDVO_CMD_GET_PREFERRED_INPUT_TIMING_PART2,
         &dtd->part2, sizeof(dtd->part2));
}

static bool intel_sdvo_set_clock_rate_mult(struct intel_sdvo *intel_sdvo, u8 val)
{
 return intel_sdvo_set_value(intel_sdvo, SDVO_CMD_SET_CLOCK_RATE_MULT, &val, 1);
}

static void intel_sdvo_get_dtd_from_mode(struct intel_sdvo_dtd *dtd,
      const struct drm_display_mode *mode)
{
 u16 width, height;
 u16 h_blank_len, h_sync_len, v_blank_len, v_sync_len;
 u16 h_sync_offset, v_sync_offset;
 int mode_clock;

 memset(dtd, 0, sizeof(*dtd));

 width = mode->hdisplay;
 height = mode->vdisplay;

 /* do some mode translations */
 h_blank_len = mode->htotal - mode->hdisplay;
 h_sync_len = mode->hsync_end - mode->hsync_start;

 v_blank_len = mode->vtotal - mode->vdisplay;
 v_sync_len = mode->vsync_end - mode->vsync_start;

 h_sync_offset = mode->hsync_start - mode->hdisplay;
 v_sync_offset = mode->vsync_start - mode->vdisplay;

 mode_clock = mode->clock;
 mode_clock /= 10;
 dtd->part1.clock = mode_clock;

 dtd->part1.h_active = width & 0xff;
 dtd->part1.h_blank = h_blank_len & 0xff;
 dtd->part1.h_high = (((width >> 8) & 0xf) << 4) |
  ((h_blank_len >> 8) & 0xf);
 dtd->part1.v_active = height & 0xff;
 dtd->part1.v_blank = v_blank_len & 0xff;
 dtd->part1.v_high = (((height >> 8) & 0xf) << 4) |
  ((v_blank_len >> 8) & 0xf);

 dtd->part2.h_sync_off = h_sync_offset & 0xff;
 dtd->part2.h_sync_width = h_sync_len & 0xff;
 dtd->part2.v_sync_off_width = (v_sync_offset & 0xf) << 4 |
  (v_sync_len & 0xf);
 dtd->part2.sync_off_width_high = ((h_sync_offset & 0x300) >> 2) |
  ((h_sync_len & 0x300) >> 4) | ((v_sync_offset & 0x30) >> 2) |
  ((v_sync_len & 0x30) >> 4);

 dtd->part2.dtd_flags = 0x18;
 if (mode->flags & DRM_MODE_FLAG_INTERLACE)
  dtd->part2.dtd_flags |= DTD_FLAG_INTERLACE;
 if (mode->flags & DRM_MODE_FLAG_PHSYNC)
  dtd->part2.dtd_flags |= DTD_FLAG_HSYNC_POSITIVE;
 if (mode->flags & DRM_MODE_FLAG_PVSYNC)
  dtd->part2.dtd_flags |= DTD_FLAG_VSYNC_POSITIVE;

 dtd->part2.v_sync_off_high = v_sync_offset & 0xc0;
}

static void intel_sdvo_get_mode_from_dtd(struct drm_display_mode *pmode,
      const struct intel_sdvo_dtd *dtd)
{
 struct drm_display_mode mode = {};

 mode.hdisplay = dtd->part1.h_active;
 mode.hdisplay += ((dtd->part1.h_high >> 4) & 0x0f) << 8;
 mode.hsync_start = mode.hdisplay + dtd->part2.h_sync_off;
 mode.hsync_start += (dtd->part2.sync_off_width_high & 0xc0) << 2;
 mode.hsync_end = mode.hsync_start + dtd->part2.h_sync_width;
 mode.hsync_end += (dtd->part2.sync_off_width_high & 0x30) << 4;
 mode.htotal = mode.hdisplay + dtd->part1.h_blank;
 mode.htotal += (dtd->part1.h_high & 0xf) << 8;

 mode.vdisplay = dtd->part1.v_active;
 mode.vdisplay += ((dtd->part1.v_high >> 4) & 0x0f) << 8;
 mode.vsync_start = mode.vdisplay;
 mode.vsync_start += (dtd->part2.v_sync_off_width >> 4) & 0xf;
 mode.vsync_start += (dtd->part2.sync_off_width_high & 0x0c) << 2;
 mode.vsync_start += dtd->part2.v_sync_off_high & 0xc0;
 mode.vsync_end = mode.vsync_start +
  (dtd->part2.v_sync_off_width & 0xf);
 mode.vsync_end += (dtd->part2.sync_off_width_high & 0x3) << 4;
 mode.vtotal = mode.vdisplay + dtd->part1.v_blank;
 mode.vtotal += (dtd->part1.v_high & 0xf) << 8;

 mode.clock = dtd->part1.clock * 10;

 if (dtd->part2.dtd_flags & DTD_FLAG_INTERLACE)
  mode.flags |= DRM_MODE_FLAG_INTERLACE;
 if (dtd->part2.dtd_flags & DTD_FLAG_HSYNC_POSITIVE)
  mode.flags |= DRM_MODE_FLAG_PHSYNC;
 else
  mode.flags |= DRM_MODE_FLAG_NHSYNC;
 if (dtd->part2.dtd_flags & DTD_FLAG_VSYNC_POSITIVE)
  mode.flags |= DRM_MODE_FLAG_PVSYNC;
 else
  mode.flags |= DRM_MODE_FLAG_NVSYNC;

 drm_mode_set_crtcinfo(&mode, 0);

 drm_mode_copy(pmode, &mode);
}

static bool intel_sdvo_check_supp_encode(struct intel_sdvo *intel_sdvo)
{
 struct intel_sdvo_encode encode;

 BUILD_BUG_ON(sizeof(encode) != 2);
 return intel_sdvo_get_value(intel_sdvo,
        SDVO_CMD_GET_SUPP_ENCODE,
        &encode, sizeof(encode));
}

static bool intel_sdvo_set_encode(struct intel_sdvo *intel_sdvo,
      u8 mode)
{
 return intel_sdvo_set_value(intel_sdvo, SDVO_CMD_SET_ENCODE, &mode, 1);
}

static bool intel_sdvo_set_colorimetry(struct intel_sdvo *intel_sdvo,
           u8 mode)
{
 return intel_sdvo_set_value(intel_sdvo, SDVO_CMD_SET_COLORIMETRY, &mode, 1);
}

static bool intel_sdvo_set_pixel_replication(struct intel_sdvo *intel_sdvo,
          u8 pixel_repeat)
{
 return intel_sdvo_set_value(intel_sdvo, SDVO_CMD_SET_PIXEL_REPLI,
        &pixel_repeat, 1);
}

static bool intel_sdvo_set_audio_state(struct intel_sdvo *intel_sdvo,
           u8 audio_state)
{
 return intel_sdvo_set_value(intel_sdvo, SDVO_CMD_SET_AUDIO_STAT,
        &audio_state, 1);
}

static bool intel_sdvo_get_hbuf_size(struct intel_sdvo *intel_sdvo,
         u8 *hbuf_size)
{
 if (!intel_sdvo_get_value(intel_sdvo, SDVO_CMD_GET_HBUF_INFO,
      hbuf_size, 1))
  return false;

 /* Buffer size is 0 based, hooray! However zero means zero. */
 if (*hbuf_size)
  (*hbuf_size)++;

 return true;
}

#if 0
static void intel_sdvo_dump_hdmi_buf(struct intel_sdvo *intel_sdvo)
{
 int i, j;
 u8 set_buf_index[2];
 u8 av_split;
 u8 buf_size;
 u8 buf[48];
 u8 *pos;

 intel_sdvo_get_value(encoder, SDVO_CMD_GET_HBUF_AV_SPLIT, &av_split, 1);

 for (i = 0; i <= av_split; i++) {
  set_buf_index[0] = i; set_buf_index[1] = 0;
  intel_sdvo_write_cmd(encoder, SDVO_CMD_SET_HBUF_INDEX,
         set_buf_index, 2);
  intel_sdvo_write_cmd(encoder, SDVO_CMD_GET_HBUF_INFO, NULL, 0);
  intel_sdvo_read_response(encoder, &buf_size, 1);

  pos = buf;
  for (j = 0; j <= buf_size; j += 8) {
   intel_sdvo_write_cmd(encoder, SDVO_CMD_GET_HBUF_DATA,
          NULL, 0);
   intel_sdvo_read_response(encoder, pos, 8);
   pos += 8;
  }
 }
}
#endif

static bool intel_sdvo_write_infoframe(struct intel_sdvo *intel_sdvo,
           unsigned int if_index, u8 tx_rate,
           const u8 *data, unsigned int length)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);
 u8 set_buf_index[2] = { if_index, 0 };
 u8 hbuf_size, tmp[8];
 int i;

 if (!intel_sdvo_set_value(intel_sdvo,
      SDVO_CMD_SET_HBUF_INDEX,
      set_buf_index, 2))
  return false;

 if (!intel_sdvo_get_hbuf_size(intel_sdvo, &hbuf_size))
  return false;

 drm_dbg_kms(display->drm,
      "writing sdvo hbuf: %i, length %u, hbuf_size: %i\n",
      if_index, length, hbuf_size);

 if (hbuf_size < length)
  return false;

 for (i = 0; i < hbuf_size; i += 8) {
  memset(tmp, 0, 8);
  if (i < length)
   memcpy(tmp, data + i, min_t(unsigned, 8, length - i));

  if (!intel_sdvo_set_value(intel_sdvo,
       SDVO_CMD_SET_HBUF_DATA,
       tmp, 8))
   return false;
 }

 return intel_sdvo_set_value(intel_sdvo,
        SDVO_CMD_SET_HBUF_TXRATE,
        &tx_rate, 1);
}

static ssize_t intel_sdvo_read_infoframe(struct intel_sdvo *intel_sdvo,
      unsigned int if_index,
      u8 *data, unsigned int length)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);
 u8 set_buf_index[2] = { if_index, 0 };
 u8 hbuf_size, tx_rate, av_split;
 int i;

 if (!intel_sdvo_get_value(intel_sdvo,
      SDVO_CMD_GET_HBUF_AV_SPLIT,
      &av_split, 1))
  return -ENXIO;

 if (av_split < if_index)
  return 0;

 if (!intel_sdvo_set_value(intel_sdvo,
      SDVO_CMD_SET_HBUF_INDEX,
      set_buf_index, 2))
  return -ENXIO;

 if (!intel_sdvo_get_value(intel_sdvo,
      SDVO_CMD_GET_HBUF_TXRATE,
      &tx_rate, 1))
  return -ENXIO;

 /* TX_DISABLED doesn't mean disabled for ELD */
 if (if_index != SDVO_HBUF_INDEX_ELD && tx_rate == SDVO_HBUF_TX_DISABLED)
  return 0;

 if (!intel_sdvo_get_hbuf_size(intel_sdvo, &hbuf_size))
  return false;

 drm_dbg_kms(display->drm,
      "reading sdvo hbuf: %i, length %u, hbuf_size: %i\n",
      if_index, length, hbuf_size);

 hbuf_size = min_t(unsigned int, length, hbuf_size);

 for (i = 0; i < hbuf_size; i += 8) {
  if (!intel_sdvo_write_cmd(intel_sdvo, SDVO_CMD_GET_HBUF_DATA, NULL, 0))
   return -ENXIO;
  if (!intel_sdvo_read_response(intel_sdvo, &data[i],
           min_t(unsigned int, 8, hbuf_size - i)))
   return -ENXIO;
 }

 return hbuf_size;
}

static bool intel_sdvo_compute_avi_infoframe(struct intel_sdvo *intel_sdvo,
          struct intel_crtc_state *crtc_state,
          struct drm_connector_state *conn_state)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);
 struct hdmi_avi_infoframe *frame = &crtc_state->infoframes.avi.avi;
 const struct drm_display_mode *adjusted_mode =
  &crtc_state->hw.adjusted_mode;
 int ret;

 if (!crtc_state->has_hdmi_sink)
  return true;

 crtc_state->infoframes.enable |=
  intel_hdmi_infoframe_enable(HDMI_INFOFRAME_TYPE_AVI);

 ret = drm_hdmi_avi_infoframe_from_display_mode(frame,
             conn_state->connector,
             adjusted_mode);
 if (ret)
  return false;

 drm_hdmi_avi_infoframe_quant_range(frame,
        conn_state->connector,
        adjusted_mode,
        crtc_state->limited_color_range ?
        HDMI_QUANTIZATION_RANGE_LIMITED :
        HDMI_QUANTIZATION_RANGE_FULL);

 ret = hdmi_avi_infoframe_check(frame);
 if (drm_WARN_ON(display->drm, ret))
  return false;

 return true;
}

static bool intel_sdvo_set_avi_infoframe(struct intel_sdvo *intel_sdvo,
      const struct intel_crtc_state *crtc_state)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);
 u8 sdvo_data[HDMI_INFOFRAME_SIZE(AVI)];
 const union hdmi_infoframe *frame = &crtc_state->infoframes.avi;
 ssize_t len;

 if ((crtc_state->infoframes.enable &
      intel_hdmi_infoframe_enable(HDMI_INFOFRAME_TYPE_AVI)) == 0)
  return true;

 if (drm_WARN_ON(display->drm,
   frame->any.type != HDMI_INFOFRAME_TYPE_AVI))
  return false;

 len = hdmi_infoframe_pack_only(frame, sdvo_data, sizeof(sdvo_data));
 if (drm_WARN_ON(display->drm, len < 0))
  return false;

 return intel_sdvo_write_infoframe(intel_sdvo, SDVO_HBUF_INDEX_AVI_IF,
       SDVO_HBUF_TX_VSYNC,
       sdvo_data, len);
}

static void intel_sdvo_get_avi_infoframe(struct intel_sdvo *intel_sdvo,
      struct intel_crtc_state *crtc_state)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);
 u8 sdvo_data[HDMI_INFOFRAME_SIZE(AVI)];
 union hdmi_infoframe *frame = &crtc_state->infoframes.avi;
 ssize_t len;
 int ret;

 if (!crtc_state->has_hdmi_sink)
  return;

 len = intel_sdvo_read_infoframe(intel_sdvo, SDVO_HBUF_INDEX_AVI_IF,
     sdvo_data, sizeof(sdvo_data));
 if (len < 0) {
  drm_dbg_kms(display->drm, "failed to read AVI infoframe\n");
  return;
 } else if (len == 0) {
  return;
 }

 crtc_state->infoframes.enable |=
  intel_hdmi_infoframe_enable(HDMI_INFOFRAME_TYPE_AVI);

 ret = hdmi_infoframe_unpack(frame, sdvo_data, len);
 if (ret) {
  drm_dbg_kms(display->drm, "Failed to unpack AVI infoframe\n");
  return;
 }

 if (frame->any.type != HDMI_INFOFRAME_TYPE_AVI)
  drm_dbg_kms(display->drm,
       "Found the wrong infoframe type 0x%x (expected 0x%02x)\n",
       frame->any.type, HDMI_INFOFRAME_TYPE_AVI);
}

static void intel_sdvo_get_eld(struct intel_sdvo *intel_sdvo,
          struct intel_crtc_state *crtc_state)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);
 ssize_t len;
 u8 val;

 if (!crtc_state->has_audio)
  return;

 if (!intel_sdvo_get_value(intel_sdvo, SDVO_CMD_GET_AUDIO_STAT, &val, 1))
  return;

 if ((val & SDVO_AUDIO_ELD_VALID) == 0)
  return;

 len = intel_sdvo_read_infoframe(intel_sdvo, SDVO_HBUF_INDEX_ELD,
     crtc_state->eld, sizeof(crtc_state->eld));
 if (len < 0)
  drm_dbg_kms(display->drm, "failed to read ELD\n");
}

static bool intel_sdvo_set_tv_format(struct intel_sdvo *intel_sdvo,
         const struct drm_connector_state *conn_state)
{
 struct intel_sdvo_tv_format format;
 u32 format_map;

 format_map = 1 << conn_state->tv.legacy_mode;
 memset(&format, 0, sizeof(format));
 memcpy(&format, &format_map, min(sizeof(format), sizeof(format_map)));

 BUILD_BUG_ON(sizeof(format) != 6);
 return intel_sdvo_set_value(intel_sdvo,
        SDVO_CMD_SET_TV_FORMAT,
        &format, sizeof(format));
}

static bool
intel_sdvo_set_output_timings_from_mode(struct intel_sdvo *intel_sdvo,
     struct intel_sdvo_connector *intel_sdvo_connector,
     const struct drm_display_mode *mode)
{
 struct intel_sdvo_dtd output_dtd;

 if (!intel_sdvo_set_target_output(intel_sdvo,
       intel_sdvo_connector->output_flag))
  return false;

 intel_sdvo_get_dtd_from_mode(&output_dtd, mode);
 if (!intel_sdvo_set_output_timing(intel_sdvo, &output_dtd))
  return false;

 return true;
}

/*
 * Asks the sdvo controller for the preferred input mode given the output mode.
 * Unfortunately we have to set up the full output mode to do that.
 */

static bool
intel_sdvo_get_preferred_input_mode(struct intel_sdvo *intel_sdvo,
        struct intel_sdvo_connector *intel_sdvo_connector,
        const struct drm_display_mode *mode,
        struct drm_display_mode *adjusted_mode)
{
 struct intel_sdvo_dtd input_dtd;

 /* Reset the input timing to the screen. Assume always input 0. */
 if (!intel_sdvo_set_target_input(intel_sdvo))
  return false;

 if (!intel_sdvo_create_preferred_input_timing(intel_sdvo,
            intel_sdvo_connector,
            mode))
  return false;

 if (!intel_sdvo_get_preferred_input_timing(intel_sdvo,
         &input_dtd))
  return false;

 intel_sdvo_get_mode_from_dtd(adjusted_mode, &input_dtd);
 intel_sdvo->dtd_sdvo_flags = input_dtd.part2.sdvo_flags;

 return true;
}

static int i9xx_adjust_sdvo_tv_clock(struct intel_crtc_state *pipe_config)
{
 struct intel_display *display = to_intel_display(pipe_config);
 unsigned int dotclock = pipe_config->hw.adjusted_mode.crtc_clock;
 struct dpll *clock = &pipe_config->dpll;

 /*
 * SDVO TV has fixed PLL values depend on its clock range,
 * this mirrors vbios setting.
 */

 if (dotclock >= 100000 && dotclock < 140500) {
  clock->p1 = 2;
  clock->p2 = 10;
  clock->n = 3;
  clock->m1 = 16;
  clock->m2 = 8;
 } else if (dotclock >= 140500 && dotclock <= 200000) {
  clock->p1 = 1;
  clock->p2 = 10;
  clock->n = 6;
  clock->m1 = 12;
  clock->m2 = 8;
 } else {
  drm_dbg_kms(display->drm,
       "SDVO TV clock out of range: %i\n", dotclock);
  return -EINVAL;
 }

 pipe_config->clock_set = true;

 return 0;
}

static bool intel_has_hdmi_sink(struct intel_sdvo_connector *intel_sdvo_connector,
    const struct drm_connector_state *conn_state)
{
 struct drm_connector *connector = conn_state->connector;

 return intel_sdvo_connector->is_hdmi &&
  connector->display_info.is_hdmi &&
  READ_ONCE(to_intel_digital_connector_state(conn_state)->force_audio) != HDMI_AUDIO_OFF_DVI;
}

static bool intel_sdvo_limited_color_range(struct intel_encoder *encoder,
        const struct intel_crtc_state *crtc_state,
        const struct drm_connector_state *conn_state)
{
 struct intel_sdvo *intel_sdvo = to_sdvo(encoder);

 if ((intel_sdvo->colorimetry_cap & SDVO_COLORIMETRY_RGB220) == 0)
  return false;

 return intel_hdmi_limited_color_range(crtc_state, conn_state);
}

static bool intel_sdvo_has_audio(struct intel_encoder *encoder,
     const struct intel_crtc_state *crtc_state,
     const struct drm_connector_state *conn_state)
{
 struct drm_connector *connector = conn_state->connector;
 struct intel_sdvo_connector *intel_sdvo_connector =
  to_intel_sdvo_connector(connector);
 const struct intel_digital_connector_state *intel_conn_state =
  to_intel_digital_connector_state(conn_state);

 if (!crtc_state->has_hdmi_sink)
  return false;

 if (intel_conn_state->force_audio == HDMI_AUDIO_AUTO)
  return intel_sdvo_connector->is_hdmi &&
   connector->display_info.has_audio;
 else
  return intel_conn_state->force_audio == HDMI_AUDIO_ON;
}

static int intel_sdvo_compute_config(struct intel_encoder *encoder,
         struct intel_crtc_state *pipe_config,
         struct drm_connector_state *conn_state)
{
 struct intel_display *display = to_intel_display(encoder);
 struct intel_sdvo *intel_sdvo = to_sdvo(encoder);
 struct intel_sdvo_connector *intel_sdvo_connector =
  to_intel_sdvo_connector(conn_state->connector);
 struct drm_display_mode *adjusted_mode = &pipe_config->hw.adjusted_mode;
 struct drm_display_mode *mode = &pipe_config->hw.mode;

 if (HAS_PCH_SPLIT(display)) {
  pipe_config->has_pch_encoder = true;
  if (!intel_fdi_compute_pipe_bpp(pipe_config))
   return -EINVAL;
 }

 drm_dbg_kms(display->drm, "forcing bpc to 8 for SDVO\n");
 /* FIXME: Don't increase pipe_bpp */
 pipe_config->pipe_bpp = 8*3;
 pipe_config->sink_format = INTEL_OUTPUT_FORMAT_RGB;
 pipe_config->output_format = INTEL_OUTPUT_FORMAT_RGB;

 /*
 * We need to construct preferred input timings based on our
 * output timings.  To do that, we have to set the output
 * timings, even though this isn't really the right place in
 * the sequence to do it. Oh well.
 */

 if (IS_TV(intel_sdvo_connector)) {
  if (!intel_sdvo_set_output_timings_from_mode(intel_sdvo,
            intel_sdvo_connector,
            mode))
   return -EINVAL;

  (void) intel_sdvo_get_preferred_input_mode(intel_sdvo,
          intel_sdvo_connector,
          mode,
          adjusted_mode);
  pipe_config->sdvo_tv_clock = true;
 } else if (IS_LVDS(intel_sdvo_connector)) {
  const struct drm_display_mode *fixed_mode =
   intel_panel_fixed_mode(&intel_sdvo_connector->base, mode);
  int ret;

  ret = intel_panel_compute_config(&intel_sdvo_connector->base,
       adjusted_mode);
  if (ret)
   return ret;

  if (!intel_sdvo_set_output_timings_from_mode(intel_sdvo,
            intel_sdvo_connector,
            fixed_mode))
   return -EINVAL;

  (void) intel_sdvo_get_preferred_input_mode(intel_sdvo,
          intel_sdvo_connector,
          mode,
          adjusted_mode);
 }

 if (adjusted_mode->flags & DRM_MODE_FLAG_DBLSCAN)
  return -EINVAL;

 /*
 * Make the CRTC code factor in the SDVO pixel multiplier.  The
 * SDVO device will factor out the multiplier during mode_set.
 */

 pipe_config->pixel_multiplier =
  intel_sdvo_get_pixel_multiplier(adjusted_mode);

 pipe_config->has_hdmi_sink = intel_has_hdmi_sink(intel_sdvo_connector, conn_state);

 pipe_config->has_audio =
  intel_sdvo_has_audio(encoder, pipe_config, conn_state) &&
  intel_audio_compute_config(encoder, pipe_config, conn_state);

 pipe_config->limited_color_range =
  intel_sdvo_limited_color_range(encoder, pipe_config,
            conn_state);

 /* Clock computation needs to happen after pixel multiplier. */
 if (IS_TV(intel_sdvo_connector)) {
  int ret;

  ret = i9xx_adjust_sdvo_tv_clock(pipe_config);
  if (ret)
   return ret;
 }

 if (conn_state->picture_aspect_ratio)
  adjusted_mode->picture_aspect_ratio =
   conn_state->picture_aspect_ratio;

 if (!intel_sdvo_compute_avi_infoframe(intel_sdvo,
           pipe_config, conn_state)) {
  drm_dbg_kms(display->drm, "bad AVI infoframe\n");
  return -EINVAL;
 }

 return 0;
}

#define UPDATE_PROPERTY(input, NAME) \
 do { \
  val = input; \
  intel_sdvo_set_value(intel_sdvo, SDVO_CMD_SET_##NAME, &val, sizeof(val)); \
 } while (0)

static void intel_sdvo_update_props(struct intel_sdvo *intel_sdvo,
        const struct intel_sdvo_connector_state *sdvo_state)
{
 const struct drm_connector_state *conn_state = &sdvo_state->base.base;
 struct intel_sdvo_connector *intel_sdvo_conn =
  to_intel_sdvo_connector(conn_state->connector);
 u16 val;

 if (intel_sdvo_conn->left)
  UPDATE_PROPERTY(sdvo_state->tv.overscan_h, OVERSCAN_H);

 if (intel_sdvo_conn->top)
  UPDATE_PROPERTY(sdvo_state->tv.overscan_v, OVERSCAN_V);

 if (intel_sdvo_conn->hpos)
  UPDATE_PROPERTY(sdvo_state->tv.hpos, HPOS);

 if (intel_sdvo_conn->vpos)
  UPDATE_PROPERTY(sdvo_state->tv.vpos, VPOS);

 if (intel_sdvo_conn->saturation)
  UPDATE_PROPERTY(conn_state->tv.saturation, SATURATION);

 if (intel_sdvo_conn->contrast)
  UPDATE_PROPERTY(conn_state->tv.contrast, CONTRAST);

 if (intel_sdvo_conn->hue)
  UPDATE_PROPERTY(conn_state->tv.hue, HUE);

 if (intel_sdvo_conn->brightness)
  UPDATE_PROPERTY(conn_state->tv.brightness, BRIGHTNESS);

 if (intel_sdvo_conn->sharpness)
  UPDATE_PROPERTY(sdvo_state->tv.sharpness, SHARPNESS);

 if (intel_sdvo_conn->flicker_filter)
  UPDATE_PROPERTY(sdvo_state->tv.flicker_filter, FLICKER_FILTER);

 if (intel_sdvo_conn->flicker_filter_2d)
  UPDATE_PROPERTY(sdvo_state->tv.flicker_filter_2d, FLICKER_FILTER_2D);

 if (intel_sdvo_conn->flicker_filter_adaptive)
  UPDATE_PROPERTY(sdvo_state->tv.flicker_filter_adaptive, FLICKER_FILTER_ADAPTIVE);

 if (intel_sdvo_conn->tv_chroma_filter)
  UPDATE_PROPERTY(sdvo_state->tv.chroma_filter, TV_CHROMA_FILTER);

 if (intel_sdvo_conn->tv_luma_filter)
  UPDATE_PROPERTY(sdvo_state->tv.luma_filter, TV_LUMA_FILTER);

 if (intel_sdvo_conn->dot_crawl)
  UPDATE_PROPERTY(sdvo_state->tv.dot_crawl, DOT_CRAWL);

#undef UPDATE_PROPERTY
}

static void intel_sdvo_pre_enable(struct intel_atomic_state *state,
      struct intel_encoder *intel_encoder,
      const struct intel_crtc_state *crtc_state,
      const struct drm_connector_state *conn_state)
{
 struct intel_display *display = to_intel_display(intel_encoder);
 struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc);
 const struct drm_display_mode *adjusted_mode = &crtc_state->hw.adjusted_mode;
 const struct intel_sdvo_connector_state *sdvo_state =
  to_intel_sdvo_connector_state(conn_state);
 struct intel_sdvo_connector *intel_sdvo_connector =
  to_intel_sdvo_connector(conn_state->connector);
 const struct drm_display_mode *mode = &crtc_state->hw.mode;
 struct intel_sdvo *intel_sdvo = to_sdvo(intel_encoder);
 u32 sdvox;
 struct intel_sdvo_in_out_map in_out;
 struct intel_sdvo_dtd input_dtd, output_dtd;
 int rate;

 intel_sdvo_update_props(intel_sdvo, sdvo_state);

 /*
 * First, set the input mapping for the first input to our controlled
 * output. This is only correct if we're a single-input device, in
 * which case the first input is the output from the appropriate SDVO
 * channel on the motherboard.  In a two-input device, the first input
 * will be SDVOB and the second SDVOC.
 */

 in_out.in0 = intel_sdvo_connector->output_flag;
 in_out.in1 = 0;

 intel_sdvo_set_value(intel_sdvo,
        SDVO_CMD_SET_IN_OUT_MAP,
        &in_out, sizeof(in_out));

 /* Set the output timings to the screen */
 if (!intel_sdvo_set_target_output(intel_sdvo,
       intel_sdvo_connector->output_flag))
  return;

 /* lvds has a special fixed output timing. */
 if (IS_LVDS(intel_sdvo_connector)) {
  const struct drm_display_mode *fixed_mode =
   intel_panel_fixed_mode(&intel_sdvo_connector->base, mode);

  intel_sdvo_get_dtd_from_mode(&output_dtd, fixed_mode);
 } else {
  intel_sdvo_get_dtd_from_mode(&output_dtd, mode);
 }
 if (!intel_sdvo_set_output_timing(intel_sdvo, &output_dtd))
  drm_info(display->drm,
    "Setting output timings on %s failed\n",
    SDVO_NAME(intel_sdvo));

 /* Set the input timing to the screen. Assume always input 0. */
 if (!intel_sdvo_set_target_input(intel_sdvo))
  return;

 if (crtc_state->has_hdmi_sink) {
  intel_sdvo_set_encode(intel_sdvo, SDVO_ENCODE_HDMI);
  intel_sdvo_set_colorimetry(intel_sdvo,
        crtc_state->limited_color_range ?
        SDVO_COLORIMETRY_RGB220 :
        SDVO_COLORIMETRY_RGB256);
  intel_sdvo_set_avi_infoframe(intel_sdvo, crtc_state);
  intel_sdvo_set_pixel_replication(intel_sdvo,
       !!(adjusted_mode->flags &
          DRM_MODE_FLAG_DBLCLK));
 } else
  intel_sdvo_set_encode(intel_sdvo, SDVO_ENCODE_DVI);

 if (IS_TV(intel_sdvo_connector) &&
     !intel_sdvo_set_tv_format(intel_sdvo, conn_state))
  return;

 intel_sdvo_get_dtd_from_mode(&input_dtd, adjusted_mode);

 if (IS_TV(intel_sdvo_connector) || IS_LVDS(intel_sdvo_connector))
  input_dtd.part2.sdvo_flags = intel_sdvo->dtd_sdvo_flags;
 if (!intel_sdvo_set_input_timing(intel_sdvo, &input_dtd))
  drm_info(display->drm,
    "Setting input timings on %s failed\n",
    SDVO_NAME(intel_sdvo));

 switch (crtc_state->pixel_multiplier) {
 default:
  drm_WARN(display->drm, 1,
    "unknown pixel multiplier specified\n");
  fallthrough;
 case 1: rate = SDVO_CLOCK_RATE_MULT_1X; break;
 case 2: rate = SDVO_CLOCK_RATE_MULT_2X; break;
 case 4: rate = SDVO_CLOCK_RATE_MULT_4X; break;
 }
 if (!intel_sdvo_set_clock_rate_mult(intel_sdvo, rate))
  return;

 /* Set the SDVO control regs. */
 if (DISPLAY_VER(display) >= 4) {
  /* The real mode polarity is set by the SDVO commands, using
 * struct intel_sdvo_dtd. */

  sdvox = SDVO_VSYNC_ACTIVE_HIGH | SDVO_HSYNC_ACTIVE_HIGH;
  if (DISPLAY_VER(display) < 5)
   sdvox |= SDVO_BORDER_ENABLE;
 } else {
  sdvox = intel_de_read(display, intel_sdvo->sdvo_reg);
  if (intel_sdvo->base.port == PORT_B)
   sdvox &= SDVOB_PRESERVE_MASK;
  else
   sdvox &= SDVOC_PRESERVE_MASK;
  sdvox |= (9 << 19) | SDVO_BORDER_ENABLE;
 }

 if (HAS_PCH_CPT(display))
  sdvox |= SDVO_PIPE_SEL_CPT(crtc->pipe);
 else
  sdvox |= SDVO_PIPE_SEL(crtc->pipe);

 if (DISPLAY_VER(display) >= 4) {
  /* done in crtc_mode_set as the dpll_md reg must be written early */
 } else if (display->platform.i945g || display->platform.i945gm ||
     display->platform.g33 || display->platform.pineview) {
  /* done in crtc_mode_set as it lives inside the dpll register */
 } else {
  sdvox |= (crtc_state->pixel_multiplier - 1)
   << SDVO_PORT_MULTIPLY_SHIFT;
 }

 if (input_dtd.part2.sdvo_flags & SDVO_NEED_TO_STALL &&
     DISPLAY_VER(display) < 5)
  sdvox |= SDVO_STALL_SELECT;
 intel_sdvo_write_sdvox(intel_sdvo, sdvox);
}

static bool intel_sdvo_connector_get_hw_state(struct intel_connector *connector)
{
 struct intel_sdvo_connector *intel_sdvo_connector =
  to_intel_sdvo_connector(&connector->base);
 struct intel_sdvo *intel_sdvo = intel_attached_sdvo(connector);
 u16 active_outputs = 0;

 intel_sdvo_get_active_outputs(intel_sdvo, &active_outputs);

 return active_outputs & intel_sdvo_connector->output_flag;
}

bool intel_sdvo_port_enabled(struct intel_display *display,
        i915_reg_t sdvo_reg, enum pipe *pipe)
{
 u32 val;

 val = intel_de_read(display, sdvo_reg);

 /* asserts want to know the pipe even if the port is disabled */
 if (HAS_PCH_CPT(display))
  *pipe = (val & SDVO_PIPE_SEL_MASK_CPT) >> SDVO_PIPE_SEL_SHIFT_CPT;
 else if (display->platform.cherryview)
  *pipe = (val & SDVO_PIPE_SEL_MASK_CHV) >> SDVO_PIPE_SEL_SHIFT_CHV;
 else
  *pipe = (val & SDVO_PIPE_SEL_MASK) >> SDVO_PIPE_SEL_SHIFT;

 return val & SDVO_ENABLE;
}

static bool intel_sdvo_get_hw_state(struct intel_encoder *encoder,
        enum pipe *pipe)
{
 struct intel_display *display = to_intel_display(encoder);
 struct intel_sdvo *intel_sdvo = to_sdvo(encoder);
 u16 active_outputs = 0;
 bool ret;

 intel_sdvo_get_active_outputs(intel_sdvo, &active_outputs);

 ret = intel_sdvo_port_enabled(display, intel_sdvo->sdvo_reg, pipe);

 return ret || active_outputs;
}

static void intel_sdvo_get_config(struct intel_encoder *encoder,
      struct intel_crtc_state *pipe_config)
{
 struct intel_display *display = to_intel_display(encoder);
 struct intel_sdvo *intel_sdvo = to_sdvo(encoder);
 struct intel_sdvo_dtd dtd;
 int encoder_pixel_multiplier = 0;
 int dotclock;
 u32 flags = 0, sdvox;
 u8 val;
 bool ret;

 pipe_config->output_types |= BIT(INTEL_OUTPUT_SDVO);

 sdvox = intel_de_read(display, intel_sdvo->sdvo_reg);

 ret = intel_sdvo_get_input_timing(intel_sdvo, &dtd);
 if (!ret) {
  /*
 * Some sdvo encoders are not spec compliant and don't
 * implement the mandatory get_timings function.
 */

  drm_dbg_kms(display->drm, "failed to retrieve SDVO DTD\n");
  pipe_config->quirks |= PIPE_CONFIG_QUIRK_MODE_SYNC_FLAGS;
 } else {
  if (dtd.part2.dtd_flags & DTD_FLAG_HSYNC_POSITIVE)
   flags |= DRM_MODE_FLAG_PHSYNC;
  else
   flags |= DRM_MODE_FLAG_NHSYNC;

  if (dtd.part2.dtd_flags & DTD_FLAG_VSYNC_POSITIVE)
   flags |= DRM_MODE_FLAG_PVSYNC;
  else
   flags |= DRM_MODE_FLAG_NVSYNC;
 }

 pipe_config->hw.adjusted_mode.flags |= flags;

 /*
 * pixel multiplier readout is tricky: Only on i915g/gm it is stored in
 * the sdvo port register, on all other platforms it is part of the dpll
 * state. Since the general pipe state readout happens before the
 * encoder->get_config we so already have a valid pixel multiplier on all
 * other platforms.
 */

 if (display->platform.i915g || display->platform.i915gm) {
  pipe_config->pixel_multiplier =
   ((sdvox & SDVO_PORT_MULTIPLY_MASK)
    >> SDVO_PORT_MULTIPLY_SHIFT) + 1;
 }

 dotclock = pipe_config->port_clock;

 if (pipe_config->pixel_multiplier)
  dotclock /= pipe_config->pixel_multiplier;

 pipe_config->hw.adjusted_mode.crtc_clock = dotclock;

 /* Cross check the port pixel multiplier with the sdvo encoder state. */
 if (intel_sdvo_get_value(intel_sdvo, SDVO_CMD_GET_CLOCK_RATE_MULT,
     &val, 1)) {
  switch (val) {
  case SDVO_CLOCK_RATE_MULT_1X:
   encoder_pixel_multiplier = 1;
   break;
  case SDVO_CLOCK_RATE_MULT_2X:
   encoder_pixel_multiplier = 2;
   break;
  case SDVO_CLOCK_RATE_MULT_4X:
   encoder_pixel_multiplier = 4;
   break;
  }
 }

 drm_WARN(display->drm,
   encoder_pixel_multiplier != pipe_config->pixel_multiplier,
   "SDVO pixel multiplier mismatch, port: %i, encoder: %i\n",
   pipe_config->pixel_multiplier, encoder_pixel_multiplier);

 if (intel_sdvo_get_value(intel_sdvo, SDVO_CMD_GET_COLORIMETRY,
     &val, 1)) {
  if (val == SDVO_COLORIMETRY_RGB220)
   pipe_config->limited_color_range = true;
 }

 if (intel_sdvo_get_value(intel_sdvo, SDVO_CMD_GET_AUDIO_STAT,
     &val, 1)) {
  if (val & SDVO_AUDIO_PRESENCE_DETECT)
   pipe_config->has_audio = true;
 }

 if (intel_sdvo_get_value(intel_sdvo, SDVO_CMD_GET_ENCODE,
     &val, 1)) {
  if (val == SDVO_ENCODE_HDMI)
   pipe_config->has_hdmi_sink = true;
 }

 intel_sdvo_get_avi_infoframe(intel_sdvo, pipe_config);

 intel_sdvo_get_eld(intel_sdvo, pipe_config);
}

static void intel_sdvo_disable_audio(struct intel_encoder *encoder,
         const struct intel_crtc_state *old_crtc_state,
         const struct drm_connector_state *old_conn_state)
{
 struct intel_sdvo *intel_sdvo = to_sdvo(encoder);

 if (!old_crtc_state->has_audio)
  return;

 intel_sdvo_set_audio_state(intel_sdvo, 0);
}

static void intel_sdvo_enable_audio(struct intel_encoder *encoder,
        const struct intel_crtc_state *crtc_state,
        const struct drm_connector_state *conn_state)
{
 struct intel_sdvo *intel_sdvo = to_sdvo(encoder);
 const u8 *eld = crtc_state->eld;

 if (!crtc_state->has_audio)
  return;

 intel_sdvo_set_audio_state(intel_sdvo, 0);

 intel_sdvo_write_infoframe(intel_sdvo, SDVO_HBUF_INDEX_ELD,
       SDVO_HBUF_TX_DISABLED,
       eld, drm_eld_size(eld));

 intel_sdvo_set_audio_state(intel_sdvo, SDVO_AUDIO_ELD_VALID |
       SDVO_AUDIO_PRESENCE_DETECT);
}

static void intel_disable_sdvo(struct intel_atomic_state *state,
          struct intel_encoder *encoder,
          const struct intel_crtc_state *old_crtc_state,
          const struct drm_connector_state *conn_state)
{
 struct intel_display *display = to_intel_display(encoder);
 struct intel_sdvo *intel_sdvo = to_sdvo(encoder);
 struct intel_crtc *crtc = to_intel_crtc(old_crtc_state->uapi.crtc);
 u32 temp;

 intel_sdvo_set_active_outputs(intel_sdvo, 0);
 if (0)
  intel_sdvo_set_encoder_power_state(intel_sdvo,
         DRM_MODE_DPMS_OFF);

 temp = intel_de_read(display, intel_sdvo->sdvo_reg);

 temp &= ~SDVO_ENABLE;
 intel_sdvo_write_sdvox(intel_sdvo, temp);

 /*
 * HW workaround for IBX, we need to move the port
 * to transcoder A after disabling it to allow the
 * matching DP port to be enabled on transcoder A.
 */

 if (HAS_PCH_IBX(display) && crtc->pipe == PIPE_B) {
  /*
 * We get CPU/PCH FIFO underruns on the other pipe when
 * doing the workaround. Sweep them under the rug.
 */

  intel_set_cpu_fifo_underrun_reporting(display, PIPE_A, false);
  intel_set_pch_fifo_underrun_reporting(display, PIPE_A, false);

  temp &= ~SDVO_PIPE_SEL_MASK;
  temp |= SDVO_ENABLE | SDVO_PIPE_SEL(PIPE_A);
  intel_sdvo_write_sdvox(intel_sdvo, temp);

  temp &= ~SDVO_ENABLE;
  intel_sdvo_write_sdvox(intel_sdvo, temp);

  intel_wait_for_vblank_if_active(display, PIPE_A);
  intel_set_cpu_fifo_underrun_reporting(display, PIPE_A, true);
  intel_set_pch_fifo_underrun_reporting(display, PIPE_A, true);
 }
}

static void pch_disable_sdvo(struct intel_atomic_state *state,
        struct intel_encoder *encoder,
        const struct intel_crtc_state *old_crtc_state,
        const struct drm_connector_state *old_conn_state)
{
}

static void pch_post_disable_sdvo(struct intel_atomic_state *state,
      struct intel_encoder *encoder,
      const struct intel_crtc_state *old_crtc_state,
      const struct drm_connector_state *old_conn_state)
{
 intel_disable_sdvo(state, encoder, old_crtc_state, old_conn_state);
}

static void intel_enable_sdvo(struct intel_atomic_state *state,
         struct intel_encoder *encoder,
         const struct intel_crtc_state *pipe_config,
         const struct drm_connector_state *conn_state)
{
 struct intel_display *display = to_intel_display(encoder);
 struct intel_sdvo *intel_sdvo = to_sdvo(encoder);
 struct intel_sdvo_connector *intel_sdvo_connector =
  to_intel_sdvo_connector(conn_state->connector);
 struct intel_crtc *crtc = to_intel_crtc(pipe_config->uapi.crtc);
 u32 temp;
 bool input1, input2;
 int i;
 bool success;

 temp = intel_de_read(display, intel_sdvo->sdvo_reg);
 temp |= SDVO_ENABLE;
 intel_sdvo_write_sdvox(intel_sdvo, temp);

 for (i = 0; i < 2; i++)
  intel_crtc_wait_for_next_vblank(crtc);

 success = intel_sdvo_get_trained_inputs(intel_sdvo, &input1, &input2);
 /*
 * Warn if the device reported failure to sync.
 *
 * A lot of SDVO devices fail to notify of sync, but it's
 * a given it the status is a success, we succeeded.
 */

 if (success && !input1) {
  drm_dbg_kms(display->drm,
       "First %s output reported failure to sync\n",
       SDVO_NAME(intel_sdvo));
 }

 if (0)
  intel_sdvo_set_encoder_power_state(intel_sdvo,
         DRM_MODE_DPMS_ON);
 intel_sdvo_set_active_outputs(intel_sdvo, intel_sdvo_connector->output_flag);
}

static enum drm_mode_status
intel_sdvo_mode_valid(struct drm_connector *connector,
        const struct drm_display_mode *mode)
{
 struct intel_display *display = to_intel_display(connector->dev);
 struct intel_sdvo *intel_sdvo = intel_attached_sdvo(to_intel_connector(connector));
 struct intel_sdvo_connector *intel_sdvo_connector =
  to_intel_sdvo_connector(connector);
 bool has_hdmi_sink = intel_has_hdmi_sink(intel_sdvo_connector, connector->state);
 int max_dotclk = display->cdclk.max_dotclk_freq;
 enum drm_mode_status status;
 int clock = mode->clock;

 status = intel_cpu_transcoder_mode_valid(display, mode);
 if (status != MODE_OK)
  return status;

 if (clock > max_dotclk)
  return MODE_CLOCK_HIGH;

 if (mode->flags & DRM_MODE_FLAG_DBLCLK) {
  if (!has_hdmi_sink)
   return MODE_CLOCK_LOW;
  clock *= 2;
 }

 if (intel_sdvo->pixel_clock_min > clock)
  return MODE_CLOCK_LOW;

 if (intel_sdvo->pixel_clock_max < clock)
  return MODE_CLOCK_HIGH;

 if (IS_LVDS(intel_sdvo_connector)) {
  enum drm_mode_status status;

  status = intel_panel_mode_valid(&intel_sdvo_connector->base, mode);
  if (status != MODE_OK)
   return status;
 }

 return MODE_OK;
}

static bool intel_sdvo_get_capabilities(struct intel_sdvo *intel_sdvo, struct intel_sdvo_caps *caps)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);

 BUILD_BUG_ON(sizeof(*caps) != 8);
 if (!intel_sdvo_get_value(intel_sdvo,
      SDVO_CMD_GET_DEVICE_CAPS,
      caps, sizeof(*caps)))
  return false;

 drm_dbg_kms(display->drm, "SDVO capabilities:\n"
      " vendor_id: %d\n"
      " device_id: %d\n"
      " device_rev_id: %d\n"
      " sdvo_version_major: %d\n"
      " sdvo_version_minor: %d\n"
      " sdvo_num_inputs: %d\n"
      " smooth_scaling: %d\n"
      " sharp_scaling: %d\n"
      " up_scaling: %d\n"
      " down_scaling: %d\n"
      " stall_support: %d\n"
      " output_flags: %d\n",
      caps->vendor_id,
      caps->device_id,
      caps->device_rev_id,
      caps->sdvo_version_major,
      caps->sdvo_version_minor,
      caps->sdvo_num_inputs,
      caps->smooth_scaling,
      caps->sharp_scaling,
      caps->up_scaling,
      caps->down_scaling,
      caps->stall_support,
      caps->output_flags);

 return true;
}

static u8 intel_sdvo_get_colorimetry_cap(struct intel_sdvo *intel_sdvo)
{
 u8 cap;

 if (!intel_sdvo_get_value(intel_sdvo, SDVO_CMD_GET_COLORIMETRY_CAP,
      &cap, sizeof(cap)))
  return SDVO_COLORIMETRY_RGB256;

 return cap;
}

static u16 intel_sdvo_get_hotplug_support(struct intel_sdvo *intel_sdvo)
{
 struct intel_display *display = to_intel_display(&intel_sdvo->base);
 u16 hotplug;

 if (!HAS_HOTPLUG(display))
  return 0;

 /*
 * HW Erratum: SDVO Hotplug is broken on all i945G chips, there's noise
 * on the line.
 */

 if (display->platform.i945g || display->platform.i945gm)
  return 0;

 if (!intel_sdvo_get_value(intel_sdvo, SDVO_CMD_GET_HOT_PLUG_SUPPORT,
      &hotplug, sizeof(hotplug)))
  return 0;

 return hotplug;
}

static void intel_sdvo_enable_hotplug(struct intel_encoder *encoder)
{
 struct intel_sdvo *intel_sdvo = to_sdvo(encoder);

 intel_sdvo_write_cmd(intel_sdvo, SDVO_CMD_SET_ACTIVE_HOT_PLUG,
        &intel_sdvo->hotplug_active, 2);
}

static enum intel_hotplug_state
intel_sdvo_hotplug(struct intel_encoder *encoder,
     struct intel_connector *connector)
{
 intel_sdvo_enable_hotplug(encoder);

 return intel_encoder_hotplug(encoder, connector);
}

static const struct drm_edid *
intel_sdvo_get_edid(struct drm_connector *connector)
{
 struct i2c_adapter *ddc = connector->ddc;

 if (!ddc)
  return NULL;

 return drm_edid_read_ddc(connector, ddc);
}

/* Mac mini hack -- use the same DDC as the analog connector */
static const struct drm_edid *
intel_sdvo_get_analog_edid(struct drm_connector *connector)
{
 struct intel_display *display = to_intel_display(connector->dev);
 struct i2c_adapter *ddc;

 ddc = intel_gmbus_get_adapter(display, display->vbt.crt_ddc_pin);
 if (!ddc)
  return NULL;

 return drm_edid_read_ddc(connector, ddc);
}

static enum drm_connector_status
intel_sdvo_tmds_sink_detect(struct drm_connector *connector)
{
 enum drm_connector_status status;
 const struct drm_edid *drm_edid;

 drm_edid = intel_sdvo_get_edid(connector);

 /*
 * When there is no edid and no monitor is connected with VGA
 * port, try to use the CRT ddc to read the EDID for DVI-connector.
 */

 if (!drm_edid)
  drm_edid = intel_sdvo_get_analog_edid(connector);

 status = connector_status_unknown;
 if (drm_edid) {
  /* DDC bus is shared, match EDID to connector type */
  if (drm_edid_is_digital(drm_edid))
   status = connector_status_connected;
  else
   status = connector_status_disconnected;
  drm_edid_free(drm_edid);
 }

 return status;
}

static bool
intel_sdvo_connector_matches_edid(struct intel_sdvo_connector *sdvo,
      const struct drm_edid *drm_edid)
{
 bool monitor_is_digital = drm_edid_is_digital(drm_edid);
 bool connector_is_digital = !!IS_DIGITAL(sdvo);

 drm_dbg_kms(sdvo->base.base.dev,
      "connector_is_digital? %d, monitor_is_digital? %d\n",
      connector_is_digital, monitor_is_digital);
 return connector_is_digital == monitor_is_digital;
}

static enum drm_connector_status
intel_sdvo_detect(struct drm_connector *connector, bool force)
{
 struct intel_display *display = to_intel_display(connector->dev);
 struct intel_sdvo *intel_sdvo = intel_attached_sdvo(to_intel_connector(connector));
 struct intel_sdvo_connector *intel_sdvo_connector = to_intel_sdvo_connector(connector);
 enum drm_connector_status ret;
 u16 response;

 drm_dbg_kms(display->drm, "[CONNECTOR:%d:%s]\n",
      connector->base.id, connector->name);

 if (!intel_display_device_enabled(display))
  return connector_status_disconnected;

 if (!intel_display_driver_check_access(display))
  return connector->status;

 if (!intel_sdvo_set_target_output(intel_sdvo,
       intel_sdvo_connector->output_flag))
  return connector_status_unknown;

 if (!intel_sdvo_get_value(intel_sdvo,
      SDVO_CMD_GET_ATTACHED_DISPLAYS,
      &response, 2))
  return connector_status_unknown;

 drm_dbg_kms(display->drm, "SDVO response %d %d [%x]\n",
      response & 0xff, response >> 8,
      intel_sdvo_connector->output_flag);

 if (response == 0)
  return connector_status_disconnected;

 if ((intel_sdvo_connector->output_flag & response) == 0)
  ret = connector_status_disconnected;
 else if (IS_TMDS(intel_sdvo_connector))
  ret = intel_sdvo_tmds_sink_detect(connector);
 else {
  const struct drm_edid *drm_edid;

  /* if we have an edid check it matches the connection */
  drm_edid = intel_sdvo_get_edid(connector);
  if (!drm_edid)
   drm_edid = intel_sdvo_get_analog_edid(connector);
  if (drm_edid) {
   if (intel_sdvo_connector_matches_edid(intel_sdvo_connector,
             drm_edid))
    ret = connector_status_connected;
   else
    ret = connector_status_disconnected;

   drm_edid_free(drm_edid);
  } else {
   ret = connector_status_connected;
  }
 }

 return ret;
}

static int intel_sdvo_get_ddc_modes(struct drm_connector *connector)
{
 struct intel_display *display = to_intel_display(connector->dev);
 int num_modes = 0;
 const struct drm_edid *drm_edid;

 drm_dbg_kms(connector->dev, "[CONNECTOR:%d:%s]\n",
      connector->base.id, connector->name);

 if (!intel_display_driver_check_access(display))
  return drm_edid_connector_add_modes(connector);

 /* set the bus switch and get the modes */
 drm_edid = intel_sdvo_get_edid(connector);

 /*
 * Mac mini hack.  On this device, the DVI-I connector shares one DDC
--> --------------------

--> maximum size reached

--> --------------------

Messung V0.5
C=94 H=94 G=93

¤ Dauer der Verarbeitung: 0.51 Sekunden  (vorverarbeitet)  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.