forked from mirror/qemu
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
211 lines
4.4 KiB
C
211 lines
4.4 KiB
C
/*
|
|
* 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
|