#ifndef IMX8_CCM_H
#define IMX8_CCM_H
#include <dev/clk/clk.h>
#include <dev/clk/clk_div.h>
#include <dev/clk/clk_fixed.h>
#include <dev/clk/clk_gate.h>
#include <dev/clk/clk_link.h>
int imx_ccm_attach(device_t);
int imx_ccm_detach(device_t);
struct imx_ccm_softc {
device_t dev;
struct resource *mem_res;
struct clkdom *clkdom;
struct mtx mtx;
struct imx_clk *clks;
int nclks;
};
DECLARE_CLASS(imx_ccm_driver);
enum imx_clk_type {
IMX_CLK_UNDEFINED = 0,
IMX_CLK_FIXED,
IMX_CLK_LINK,
IMX_CLK_MUX,
IMX_CLK_GATE,
IMX_CLK_COMPOSITE,
IMX_CLK_SSCG_PLL,
IMX_CLK_FRAC_PLL,
IMX_CLK_DIV,
};
struct imx_clk {
enum imx_clk_type type;
union {
struct clk_fixed_def *fixed;
struct clk_link_def *link;
struct imx_clk_mux_def *mux;
struct imx_clk_gate_def *gate;
struct imx_clk_composite_def *composite;
struct imx_clk_sscg_pll_def *sscg_pll;
struct imx_clk_frac_pll_def *frac_pll;
struct clk_div_def *div;
} clk;
};
#define LINK(_id, _name) \
{ \
.type = IMX_CLK_LINK, \
.clk.link = &(struct clk_link_def) { \
.clkdef.id = _id, \
.clkdef.name = _name, \
.clkdef.parent_names = NULL, \
.clkdef.parent_cnt = 0, \
.clkdef.flags = CLK_NODE_STATIC_STRINGS, \
}, \
}
#define MUX(_id, _name, _pn, _f, _mo, _ms, _mw) \
{ \
.type = IMX_CLK_MUX, \
.clk.mux = &(struct imx_clk_mux_def) { \
.clkdef.id = _id, \
.clkdef.name = _name, \
.clkdef.parent_names = _pn, \
.clkdef.parent_cnt = nitems(_pn), \
.clkdef.flags = CLK_NODE_STATIC_STRINGS, \
.offset = _mo, \
.shift = _ms, \
.width = _mw, \
.mux_flags = _f, \
}, \
}
#define FIXED(_id, _name, _freq) \
{ \
.type = IMX_CLK_FIXED, \
.clk.fixed = &(struct clk_fixed_def) { \
.clkdef.id = _id, \
.clkdef.name = _name, \
.clkdef.flags = CLK_NODE_STATIC_STRINGS, \
.freq = _freq, \
}, \
}
#define FFACT(_id, _name, _pname, _mult, _div) \
{ \
.type = IMX_CLK_FIXED, \
.clk.fixed = &(struct clk_fixed_def) { \
.clkdef.id = _id, \
.clkdef.name = _name, \
.clkdef.parent_names = (const char *[]){_pname}, \
.clkdef.parent_cnt = 1, \
.clkdef.flags = CLK_NODE_STATIC_STRINGS, \
.mult = _mult, \
.div = _div, \
}, \
}
#define GATE(_id, _name, _pname, _o, _shift) \
{ \
.type = IMX_CLK_GATE, \
.clk.gate = &(struct imx_clk_gate_def) { \
.clkdef.id = _id, \
.clkdef.name = _name, \
.clkdef.parent_names = (const char *[]){_pname}, \
.clkdef.parent_cnt = 1, \
.clkdef.flags = CLK_NODE_STATIC_STRINGS, \
.offset = _o, \
.shift = _shift, \
.mask = 1, \
}, \
}
#define ROOT_GATE(_id, _name, _pname, _reg) \
{ \
.type = IMX_CLK_GATE, \
.clk.gate = &(struct imx_clk_gate_def) { \
.clkdef.id = _id, \
.clkdef.name = _name, \
.clkdef.parent_names = (const char *[]){_pname}, \
.clkdef.parent_cnt = 1, \
.clkdef.flags = CLK_NODE_STATIC_STRINGS, \
.offset = _reg, \
.shift = 0, \
.mask = 3, \
}, \
}
#define COMPOSITE(_id, _name, _pn, _o, _flags) \
{ \
.type = IMX_CLK_COMPOSITE, \
.clk.composite = &(struct imx_clk_composite_def) { \
.clkdef.id = _id, \
.clkdef.name = _name, \
.clkdef.parent_names = _pn, \
.clkdef.parent_cnt = nitems(_pn), \
.clkdef.flags = CLK_NODE_STATIC_STRINGS, \
.offset = _o, \
.flags = _flags, \
}, \
}
#define SSCG_PLL(_id, _name, _pn, _o) \
{ \
.type = IMX_CLK_SSCG_PLL, \
.clk.composite = &(struct imx_clk_composite_def) { \
.clkdef.id = _id, \
.clkdef.name = _name, \
.clkdef.parent_names = _pn, \
.clkdef.parent_cnt = nitems(_pn), \
.clkdef.flags = CLK_NODE_STATIC_STRINGS, \
.offset = _o, \
}, \
}
#define FRAC_PLL(_id, _name, _pname, _o) \
{ \
.type = IMX_CLK_FRAC_PLL, \
.clk.frac_pll = &(struct imx_clk_frac_pll_def) { \
.clkdef.id = _id, \
.clkdef.name = _name, \
.clkdef.parent_names = (const char *[]){_pname}, \
.clkdef.parent_cnt = 1, \
.clkdef.flags = CLK_NODE_STATIC_STRINGS, \
.offset = _o, \
}, \
}
#define DIV(_id, _name, _pname, _o, _shift, _width) \
{ \
.type = IMX_CLK_DIV, \
.clk.div = &(struct clk_div_def) { \
.clkdef.id = _id, \
.clkdef.name = _name, \
.clkdef.parent_names = (const char *[]){_pname}, \
.clkdef.parent_cnt = 1, \
.clkdef.flags = CLK_NODE_STATIC_STRINGS, \
.offset = _o, \
.i_shift = _shift, \
.i_width = _width, \
}, \
}
#endif