/*
 * BCM2835 CPRMAN clock manager
 *
 * Copyright (c) 2020 Luc Michel <luc@lmichel.fr>
 *
 * SPDX-License-Identifier: GPL-2.0-or-later
 */

#ifndef HW_MISC_BCM2835_CPRMAN_H
#define HW_MISC_BCM2835_CPRMAN_H

#include "hw/sysbus.h"
#include "hw/qdev-clock.h"

#define TYPE_BCM2835_CPRMAN "bcm2835-cprman"

typedef struct BCM2835CprmanState BCM2835CprmanState;

DECLARE_INSTANCE_CHECKER(BCM2835CprmanState, CPRMAN,
                         TYPE_BCM2835_CPRMAN)

#define CPRMAN_NUM_REGS (0x2000 / sizeof(uint32_t))

typedef enum CprmanPll {
    CPRMAN_PLLA = 0,
    CPRMAN_PLLC,
    CPRMAN_PLLD,
    CPRMAN_PLLH,
    CPRMAN_PLLB,

    CPRMAN_NUM_PLL
} CprmanPll;

typedef enum CprmanPllChannel {
    CPRMAN_PLLA_CHANNEL_DSI0 = 0,
    CPRMAN_PLLA_CHANNEL_CORE,
    CPRMAN_PLLA_CHANNEL_PER,
    CPRMAN_PLLA_CHANNEL_CCP2,

    CPRMAN_PLLC_CHANNEL_CORE2,
    CPRMAN_PLLC_CHANNEL_CORE1,
    CPRMAN_PLLC_CHANNEL_PER,
    CPRMAN_PLLC_CHANNEL_CORE0,

    CPRMAN_PLLD_CHANNEL_DSI0,
    CPRMAN_PLLD_CHANNEL_CORE,
    CPRMAN_PLLD_CHANNEL_PER,
    CPRMAN_PLLD_CHANNEL_DSI1,

    CPRMAN_PLLH_CHANNEL_AUX,
    CPRMAN_PLLH_CHANNEL_RCAL,
    CPRMAN_PLLH_CHANNEL_PIX,

    CPRMAN_PLLB_CHANNEL_ARM,

    CPRMAN_NUM_PLL_CHANNEL,

    /* Special values used when connecting clock sources to clocks */
    CPRMAN_CLOCK_SRC_NORMAL = -1,
    CPRMAN_CLOCK_SRC_FORCE_GROUND = -2,
    CPRMAN_CLOCK_SRC_DSI0HSCK = -3,
} CprmanPllChannel;

typedef enum CprmanClockMux {
    CPRMAN_CLOCK_GNRIC,
    CPRMAN_CLOCK_VPU,
    CPRMAN_CLOCK_SYS,
    CPRMAN_CLOCK_PERIA,
    CPRMAN_CLOCK_PERII,
    CPRMAN_CLOCK_H264,
    CPRMAN_CLOCK_ISP,
    CPRMAN_CLOCK_V3D,
    CPRMAN_CLOCK_CAM0,
    CPRMAN_CLOCK_CAM1,
    CPRMAN_CLOCK_CCP2,
    CPRMAN_CLOCK_DSI0E,
    CPRMAN_CLOCK_DSI0P,
    CPRMAN_CLOCK_DPI,
    CPRMAN_CLOCK_GP0,
    CPRMAN_CLOCK_GP1,
    CPRMAN_CLOCK_GP2,
    CPRMAN_CLOCK_HSM,
    CPRMAN_CLOCK_OTP,
    CPRMAN_CLOCK_PCM,
    CPRMAN_CLOCK_PWM,
    CPRMAN_CLOCK_SLIM,
    CPRMAN_CLOCK_SMI,
    CPRMAN_CLOCK_TEC,
    CPRMAN_CLOCK_TD0,
    CPRMAN_CLOCK_TD1,
    CPRMAN_CLOCK_TSENS,
    CPRMAN_CLOCK_TIMER,
    CPRMAN_CLOCK_UART,
    CPRMAN_CLOCK_VEC,
    CPRMAN_CLOCK_PULSE,
    CPRMAN_CLOCK_SDC,
    CPRMAN_CLOCK_ARM,
    CPRMAN_CLOCK_AVEO,
    CPRMAN_CLOCK_EMMC,
    CPRMAN_CLOCK_EMMC2,

    CPRMAN_NUM_CLOCK_MUX
} CprmanClockMux;

typedef enum CprmanClockMuxSource {
    CPRMAN_CLOCK_SRC_GND = 0,
    CPRMAN_CLOCK_SRC_XOSC,
    CPRMAN_CLOCK_SRC_TD0,
    CPRMAN_CLOCK_SRC_TD1,
    CPRMAN_CLOCK_SRC_PLLA,
    CPRMAN_CLOCK_SRC_PLLC,
    CPRMAN_CLOCK_SRC_PLLD,
    CPRMAN_CLOCK_SRC_PLLH,
    CPRMAN_CLOCK_SRC_PLLC_CORE1,
    CPRMAN_CLOCK_SRC_PLLC_CORE2,

    CPRMAN_NUM_CLOCK_MUX_SRC
} CprmanClockMuxSource;

typedef struct CprmanPllState {
    /*< private >*/
    DeviceState parent_obj;

    /*< public >*/
    CprmanPll id;

    uint32_t *reg_cm;
    uint32_t *reg_a2w_ctrl;
    uint32_t *reg_a2w_ana; /* ANA[0] .. ANA[3] */
    uint32_t prediv_mask; /* prediv bit in ana[1] */
    uint32_t *reg_a2w_frac;

    Clock *xosc_in;
    Clock *out;
} CprmanPllState;

typedef struct CprmanPllChannelState {
    /*< private >*/
    DeviceState parent_obj;

    /*< public >*/
    CprmanPllChannel id;
    CprmanPll parent;

    uint32_t *reg_cm;
    uint32_t hold_mask;
    uint32_t load_mask;
    uint32_t *reg_a2w_ctrl;
    int fixed_divider;

    Clock *pll_in;
    Clock *out;
} CprmanPllChannelState;

typedef struct CprmanClockMuxState {
    /*< private >*/
    DeviceState parent_obj;

    /*< public >*/
    CprmanClockMux id;

    uint32_t *reg_ctl;
    uint32_t *reg_div;
    int int_bits;
    int frac_bits;

    Clock *srcs[CPRMAN_NUM_CLOCK_MUX_SRC];
    Clock *out;

    /*
     * Used by clock srcs update callback to retrieve both the clock and the
     * source number.
     */
    struct CprmanClockMuxState *backref[CPRMAN_NUM_CLOCK_MUX_SRC];
} CprmanClockMuxState;

typedef struct CprmanDsi0HsckMuxState {
    /*< private >*/
    DeviceState parent_obj;

    /*< public >*/
    CprmanClockMux id;

    uint32_t *reg_cm;

    Clock *plla_in;
    Clock *plld_in;
    Clock *out;
} CprmanDsi0HsckMuxState;

struct BCM2835CprmanState {
    /*< private >*/
    SysBusDevice parent_obj;

    /*< public >*/
    MemoryRegion iomem;

    CprmanPllState plls[CPRMAN_NUM_PLL];
    CprmanPllChannelState channels[CPRMAN_NUM_PLL_CHANNEL];
    CprmanClockMuxState clock_muxes[CPRMAN_NUM_CLOCK_MUX];
    CprmanDsi0HsckMuxState dsi0hsck_mux;

    uint32_t regs[CPRMAN_NUM_REGS];
    uint32_t xosc_freq;

    Clock *xosc;
    Clock *gnd;
};

#endif