mirror of
https://github.com/qemu/qemu.git
synced 2025-08-02 04:35:42 +00:00

PLLs are composed of multiple channels. Each channel outputs one clock signal. They are modeled as one device taking the PLL generated clock as input, and outputting a new clock. A channel shares the CM register with its parent PLL, and has its own A2W_CTRL register. A write to the CM register will trigger an update of the PLL and all its channels, while a write to an A2W_CTRL channel register will update the required channel only. Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org> Tested-by: Philippe Mathieu-Daudé <f4bug@amsat.org> Signed-off-by: Luc Michel <luc@lmichel.fr> Tested-by: Guenter Roeck <linux@roeck-us.net> Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
111 lines
2.2 KiB
C
111 lines
2.2 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_CPRMAN_H
|
|
#define HW_MISC_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,
|
|
} CprmanPllChannel;
|
|
|
|
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;
|
|
|
|
struct BCM2835CprmanState {
|
|
/*< private >*/
|
|
SysBusDevice parent_obj;
|
|
|
|
/*< public >*/
|
|
MemoryRegion iomem;
|
|
|
|
CprmanPllState plls[CPRMAN_NUM_PLL];
|
|
CprmanPllChannelState channels[CPRMAN_NUM_PLL_CHANNEL];
|
|
|
|
uint32_t regs[CPRMAN_NUM_REGS];
|
|
uint32_t xosc_freq;
|
|
|
|
Clock *xosc;
|
|
};
|
|
|
|
#endif
|