[alsa-devel] [v4, 1/2] video: hdmi: add helper functions for N and CTS

Doug Anderson dianders at chromium.org
Mon Jun 6 18:34:47 CEST 2016


Hi,

On Thu, Apr 21, 2016 at 8:29 AM, Arnaud Pouliquen <dianders at chromium.org> wrote:
> Add helper functions to compute HDMI CTS and N parameters.
> Implementation is based on HDMI 1.4b specification.

It would be super nice to have this somewhere common.  Any idea who
would land this?


> +static const struct hdmi_audio_acr hdmi_audio_standard_acr[3][13] = {
> +       [HDMI_AUDIO_N_CTS_32KHZ] = {
> +               /* N and CTS values for 32 kHz rate*/
> +               {  25174825, {  4576,  28125, 0 } }, /* 25.20/1.001  MHz */
> +               {  25200000, {  4096,  25200, 0 } }, /* 25.20        MHz */
> +               {  27000000, {  4096,  27000, 0 } }, /* 27.00        MHz */
> +               {  27027000, {  4096,  27027, 0 } }, /* 27.00*1.001  MHz */
> +               {  54000000, {  4096,  54000, 0 } }, /* 54.00        MHz */
> +               {  54054000, {  4096,  54054, 0 } }, /* 54.00*1.001  MHz */
> +               {  74175824, { 11648, 210937, 50 } }, /* 74.25/1.001 MHz */
> +               {  74250000, {  4096,  74250, 0 } }, /* 74.25        MHz */
> +               { 148351648, { 11648, 421875, 0 } }, /* 148.50/1.001 MHz */
> +               { 148500000, {  4096, 148500, 0 } }, /* 148.50       MHz */
> +               { 296703296, {  5824, 421875, 0 } }, /* 297/1.001 MHz (truncated)*/
> +               { 296703297, {  5824, 421875, 0 } }, /* 297/1.001 MHz (rounded)*/
> +               { 297000000, {  3072, 222750, 0 } }, /* 297          MHz */

One thing to note is that for all but the non-integral clock rates and
the rates >= ~297MHz, all of this can be done programmatically.
...the function I came up with to do that is pretty slow, so a table
is still useful in general unless you want to try to optimize things,
but it might be nice to have the function available as a fallback?
Specifically many TVs will allow audio to work with rates other than
the ones in the HDMI spec.

You can see the full implementation we used on some devices I worked
on at <https://chromium.googlesource.com/chromiumos/third_party/kernel/+/chromeos-3.14/drivers/gpu/drm/bridge/dw_hdmi.c>.
Specifically the function for computing N:

static unsigned int hdmi_compute_n(struct dw_hdmi *hdmi,
           unsigned long pixel_clk)
{
  unsigned int freq = hdmi->sample_rate;
  unsigned int min_n = DIV_ROUND_UP((128 * freq), 1500);
  unsigned int max_n = (128 * freq) / 300;
  unsigned int ideal_n = (128 * freq) / 1000;
  unsigned int best_n_distance = ideal_n;
  unsigned int best_n = 0;
  u64 best_diff = U64_MAX;
  int n;
  /* If the ideal N could satisfy the audio math, then just take it */
  if (hdmi_audio_math_diff(freq, ideal_n, pixel_clk) == 0)
    return ideal_n;
  for (n = min_n; n <= max_n; n++) {
    u64 diff = hdmi_audio_math_diff(freq, n, pixel_clk);
    if (diff < best_diff || (diff == best_diff &&
        abs(n - ideal_n) < best_n_distance)) {
      best_n = n;
      best_diff = diff;
      best_n_distance = abs(best_n - ideal_n);
    }
    /*
     * The best N already satisfy the audio math, and also be
     * the closest value to ideal N, so just cut the loop.
     */
    if ((best_diff == 0) && (abs(n - ideal_n) > best_n_distance))
      break;
  }
  return best_n;
}

I believe this function written by Yakir Yang based on a bit of python
I had coded up.  The python has the advantage that it will come up
with the right N/CTS even for fractional clock rates, like
25.20/1.001:

def DIV_ROUND_UP(x, y): return (x + y - 1) / y
def calc(freq, tmds):
  min_n = DIV_ROUND_UP((128 * freq), 1500)
  max_n = (128 * freq) / 300
  ideal_n = (128 * freq) / 1000
  best = 0xffffffffffffffff
  for n in xrange(min_n, max_n + 1):
    cts = int(round((tmds * n / (128. * freq))))
    diff = abs(tmds * n - cts * (128. * freq))
    if (diff < best) or \
       (diff == best and
        abs(n - ideal_n) < abs(best_n - ideal_n)):
      best = diff
      best_n = n

  # Want a number that's close to an integer here
  print tmds, freq, best_n, tmds * (best_n) / (128. * freq)

  n = best_n
  cts = (tmds * n) / (128 * freq)
  print ">>> ((128 * %d) * %d) / %d." % (freq, cts, n)
  print "%f" % (((128 * freq) * cts) / n)
  print

25174825.1748 32000 4576 28125.0
>>> ((128 * 32000) * 28125) / 4576.
25174825.174825

25174825.1748 44100 7007 31250.0
>>> ((128 * 44100) * 31250) / 7007.
25174825.174825

25174825.1748 48000 6864 28125.0
>>> ((128 * 48000) * 28125) / 6864.
25174825.174825


One other thing to note is that if your HDMI block doesn't happen to
make _exactly_ the right clock then these values aren't right.  For
instance, if you end up making 25174825 Hz instead of 25200000 / 1.001
Hz that different N/CTS values are ideal.  The numbers below are the
result of my python but (as you can see) things don't match up
properly.

25174825 32000 4405 27074.0000305
>>> ((128 * 32000) * 27074) / 4405.
25174824.000000

25174825 44100 9073 40464.0000044
>>> ((128 * 44100) * 40464) / 9073.
25174824.000000

25174825 48000 15503 63522.9999959
>>> ((128 * 48000) * 63522) / 15503.
25174428.000000


In my particular case we could make 25,176,471 which we thought was
close enough to the proper clock rate, but still deserves better N/CTS
rates.

  /* 25176471 for 25.175 MHz = 428000000 / 17. */
  { .tmds = 25177000, .n_32k = 4352, .n_44k1 = 14994, .n_48k = 6528, },


> +       /*
> +        * Pre-defined frequency not found. Compute CTS using formula:
> +        * CTS = (Ftdms_clk * N) / (128 * audio_fs)
> +        */
> +       val = (u64)tmds_clk * n_cts->n;
> +       n_cts->cts = div64_u64(val, 128UL * audio_fs);
> +
> +       n_cts->cts_1_ratio = 0;
> +       min = (u64)n_cts->cts * 128UL * audio_fs;
> +       if (min < val) {
> +               /*
> +                * Non-accurate value for CTS
> +                * compute ratio, needed by user to alternate in ACR
> +                * between CTS and CTS + 1 value.
> +                */
> +               n_cts->cts_1_ratio = ((u32)(val - min)) * 100 /
> +                                    (128 * audio_fs);
> +       }

This fallback isn't nearly as nice and will likely lead to audio
reconstitution problems.  IIRC the problem was periodic audio cutouts
of you listened long enough.


-Doug


More information about the Alsa-devel mailing list