ASoC: wm8962: Relax bit clock divider searching
[sfrench/cifs-2.6.git] / sound / soc / codecs / wm8962.c
index ce4666a74793e35884e67dc6724bf36ab3246f01..34080f497584d39cbcdc83d83726199a7cf0159b 100644 (file)
@@ -2403,6 +2403,7 @@ static const int sysclk_rates[] = {
 static void wm8962_configure_bclk(struct snd_soc_component *component)
 {
        struct wm8962_priv *wm8962 = snd_soc_component_get_drvdata(component);
+       int best, min_diff, diff;
        int dspclk, i;
        int clocking2 = 0;
        int clocking4 = 0;
@@ -2473,23 +2474,25 @@ static void wm8962_configure_bclk(struct snd_soc_component *component)
 
        dev_dbg(component->dev, "DSPCLK is %dHz, BCLK %d\n", dspclk, wm8962->bclk);
 
-       /* We're expecting an exact match */
+       /* Search a proper bclk, not exact match. */
+       best = 0;
+       min_diff = INT_MAX;
        for (i = 0; i < ARRAY_SIZE(bclk_divs); i++) {
                if (bclk_divs[i] < 0)
                        continue;
 
-               if (dspclk / bclk_divs[i] == wm8962->bclk) {
-                       dev_dbg(component->dev, "Selected BCLK_DIV %d for %dHz\n",
-                               bclk_divs[i], wm8962->bclk);
-                       clocking2 |= i;
+               diff = (dspclk / bclk_divs[i]) - wm8962->bclk;
+               if (diff < 0) /* Table is sorted */
                        break;
+               if (diff < min_diff) {
+                       best = i;
+                       min_diff = diff;
                }
        }
-       if (i == ARRAY_SIZE(bclk_divs)) {
-               dev_err(component->dev, "Unsupported BCLK ratio %d\n",
-                       dspclk / wm8962->bclk);
-               return;
-       }
+       wm8962->bclk = dspclk / bclk_divs[best];
+       clocking2 |= best;
+       dev_dbg(component->dev, "Selected BCLK_DIV %d for %dHz\n",
+               bclk_divs[best], wm8962->bclk);
 
        aif2 |= wm8962->bclk / wm8962->lrclk;
        dev_dbg(component->dev, "Selected LRCLK divisor %d for %dHz\n",