#define AR0521_PLL_MIN          (320 * 1000 * 1000)
 #define AR0521_PLL_MAX         (1280 * 1000 * 1000)
 
-/* Effective pixel clocks, the registers may be DDR */
+/* Effective pixel sample rate on the pixel array. */
 #define AR0521_PIXEL_CLOCK_RATE         (184 * 1000 * 1000)
 #define AR0521_PIXEL_CLOCK_MIN  (168 * 1000 * 1000)
 #define AR0521_PIXEL_CLOCK_MAX  (414 * 1000 * 1000)
        unsigned int lane_count;
        u16 total_width;
        u16 total_height;
-       u16 pll_pre;
-       u16 pll_mult;
-       u16 pll_pre2;
-       u16 pll_mult2;
+       struct {
+               u16 pre;
+               u16 mult;
+               u16 pre2;
+               u16 mult2;
+               u16 vt_pix;
+       } pll;
+
        bool streaming;
 };
 
        return div_u64(v + d - 1, d);
 }
 
+static int ar0521_code_to_bpp(struct ar0521_dev *sensor)
+{
+       switch (sensor->fmt.code) {
+       case MEDIA_BUS_FMT_SGRBG8_1X8:
+               return 8;
+       }
+
+       return -EINVAL;
+}
+
 /* Data must be BE16, the first value is the register address */
 static int ar0521_write_regs(struct ar0521_dev *sensor, const __be16 *data,
                             unsigned int count)
        return ar0521_write_regs(sensor, regs, ARRAY_SIZE(regs));
 }
 
-static u32 calc_pll(struct ar0521_dev *sensor, int num, u32 freq, u16 *pre_ptr,
-                   u16 *mult_ptr)
+static u32 calc_pll(struct ar0521_dev *sensor, u32 freq, u16 *pre_ptr, u16 *mult_ptr)
 {
        u16 pre = 1, mult = 1, new_pre;
        u32 pll = AR0521_PLL_MAX + 1;
        return pll;
 }
 
-#define DIV 4
 static void ar0521_calc_mode(struct ar0521_dev *sensor)
 {
-       unsigned int speed_mod = 4 / sensor->lane_count; /* 1 with 4 DDR lanes */
-       u16 total_width = max(sensor->fmt.width + AR0521_WIDTH_BLANKING_MIN,
-                             AR0521_TOTAL_WIDTH_MIN);
-       u16 total_height = sensor->fmt.height + AR0521_HEIGHT_BLANKING_MIN;
-
-       /* Calculate approximate pixel clock first */
-       u64 pix_clk = AR0521_PIXEL_CLOCK_RATE;
-
-       /* PLL1 drives pixel clock - dual rate */
-       pix_clk = calc_pll(sensor, 1, pix_clk * (DIV / 2), &sensor->pll_pre,
-                          &sensor->pll_mult);
-       pix_clk = div64_round(pix_clk, (DIV / 2));
-       calc_pll(sensor, 2, pix_clk * (DIV / 2) * speed_mod, &sensor->pll_pre2,
-                &sensor->pll_mult2);
-
-       sensor->total_width = total_width;
-       sensor->total_height = total_height;
+       unsigned int pixel_clock;
+       u16 pre, mult;
+       u32 vco;
+       int bpp;
+
+       /*
+        * PLL1 and PLL2 are computed equally even if the application note
+        * suggests a slower PLL1 clock. Maintain pll1 and pll2 divider and
+        * multiplier separated to later specialize the calculation procedure.
+        *
+        * PLL1:
+        * - mclk -> / pre_div1 * pre_mul1 = VCO1 = COUNTER_CLOCK
+        *
+        * PLL2:
+        * - mclk -> / pre_div * pre_mul = VCO
+        *
+        *   VCO -> / vt_pix = PIXEL_CLOCK
+        *   VCO -> / vt_pix / 2 = WORD_CLOCK
+        *   VCO -> / op_sys = SERIAL_CLOCK
+        *
+        * With:
+        * - vt_pix = bpp / 2
+        * - WORD_CLOCK = PIXEL_CLOCK / 2
+        * - SERIAL_CLOCK = MIPI data rate (Mbps / lane) = WORD_CLOCK * bpp
+        *   NOTE: this implies the MIPI clock is divided internally by 2
+        *         to account for DDR.
+        *
+        * As op_sys_div is fixed to 1:
+        *
+        * SERIAL_CLOCK = VCO
+        * VCO = 2 * MIPI_CLK
+        * VCO = PIXEL_CLOCK * bpp / 2
+        *
+        * In the clock tree:
+        * MIPI_CLK = PIXEL_CLOCK * bpp / 2 / 2
+        *
+        * Generic pixel_rate to bus clock frequencey equation:
+        * MIPI_CLK = V4L2_CID_PIXEL_RATE * bpp / lanes / 2
+        *
+        * From which we derive the PIXEL_CLOCK to use in the clock tree:
+        * PIXEL_CLOCK = V4L2_CID_PIXEL_RATE * 2 / lanes
+        *
+        * Documented clock ranges:
+        *   WORD_CLOCK = (35MHz - 120 MHz)
+        *   PIXEL_CLOCK = (84MHz - 207MHz)
+        *   VCO = (320MHz - 1280MHz)
+        *
+        * TODO: in case we have less data lanes we have to reduce the desired
+        * VCO not to exceed the limits specified by the datasheet and
+        * consequentially reduce the obtained pixel clock.
+        */
+       pixel_clock = AR0521_PIXEL_CLOCK_RATE * 2 / sensor->lane_count;
+       bpp = ar0521_code_to_bpp(sensor);
+       sensor->pll.vt_pix = bpp / 2;
+       vco = pixel_clock * sensor->pll.vt_pix;
+
+       calc_pll(sensor, vco, &pre, &mult);
+
+       sensor->pll.pre = sensor->pll.pre2 = pre;
+       sensor->pll.mult = sensor->pll.mult2 = mult;
 }
 
 static int ar0521_write_mode(struct ar0521_dev *sensor)
 {
        __be16 pll_regs[] = {
                be(AR0521_REG_VT_PIX_CLK_DIV),
-               /* 0x300 */ be(4), /* vt_pix_clk_div = number of bits / 2 */
+               /* 0x300 */ be(sensor->pll.vt_pix), /* vt_pix_clk_div = bpp / 2 */
                /* 0x302 */ be(1), /* vt_sys_clk_div */
-               /* 0x304 */ be((sensor->pll_pre2 << 8) | sensor->pll_pre),
-               /* 0x306 */ be((sensor->pll_mult2 << 8) | sensor->pll_mult),
-               /* 0x308 */ be(8), /* op_pix_clk_div = 2 * vt_pix_clk_div */
+               /* 0x304 */ be((sensor->pll.pre2 << 8) | sensor->pll.pre),
+               /* 0x306 */ be((sensor->pll.mult2 << 8) | sensor->pll.mult),
+               /* 0x308 */ be(sensor->pll.vt_pix * 2), /* op_pix_clk_div = 2 * vt_pix_clk_div */
                /* 0x30A */ be(1)  /* op_sys_clk_div */
        };
        int ret;