| /* SPDX-License-Identifier: GPL-2.0-only */ |
| /* |
| * Copyright (c) 2016 Maxime Ripard. All rights reserved. |
| */ |
| |
| #ifndef _CCU_DIV_H_ |
| #define _CCU_DIV_H_ |
| |
| #include <linux/clk-provider.h> |
| |
| #include "ccu_common.h" |
| #include "ccu_mux.h" |
| |
| /** |
| * struct ccu_div_internal - Internal divider description |
| * @shift: Bit offset of the divider in its register |
| * @width: Width of the divider field in its register |
| * @max: Maximum value allowed for that divider. This is the |
| * arithmetic value, not the maximum value to be set in the |
| * register. |
| * @flags: clk_divider flags to apply on this divider |
| * @table: Divider table pointer (if applicable) |
| * |
| * That structure represents a single divider, and is meant to be |
| * embedded in other structures representing the various clock |
| * classes. |
| * |
| * It is basically a wrapper around the clk_divider functions |
| * arguments. |
| */ |
| struct ccu_div_internal { |
| u8 shift; |
| u8 width; |
| |
| u32 max; |
| u32 offset; |
| |
| u32 flags; |
| |
| struct clk_div_table *table; |
| }; |
| |
| #define _SUNXI_CCU_DIV_TABLE_FLAGS(_shift, _width, _table, _flags) \ |
| { \ |
| .shift = _shift, \ |
| .width = _width, \ |
| .flags = _flags, \ |
| .table = _table, \ |
| } |
| |
| #define _SUNXI_CCU_DIV_TABLE(_shift, _width, _table) \ |
| _SUNXI_CCU_DIV_TABLE_FLAGS(_shift, _width, _table, 0) |
| |
| #define _SUNXI_CCU_DIV_OFFSET_MAX_FLAGS(_shift, _width, _off, _max, _flags) \ |
| { \ |
| .shift = _shift, \ |
| .width = _width, \ |
| .flags = _flags, \ |
| .max = _max, \ |
| .offset = _off, \ |
| } |
| |
| #define _SUNXI_CCU_DIV_MAX_FLAGS(_shift, _width, _max, _flags) \ |
| _SUNXI_CCU_DIV_OFFSET_MAX_FLAGS(_shift, _width, 1, _max, _flags) |
| |
| #define _SUNXI_CCU_DIV_FLAGS(_shift, _width, _flags) \ |
| _SUNXI_CCU_DIV_MAX_FLAGS(_shift, _width, 0, _flags) |
| |
| #define _SUNXI_CCU_DIV_MAX(_shift, _width, _max) \ |
| _SUNXI_CCU_DIV_MAX_FLAGS(_shift, _width, _max, 0) |
| |
| #define _SUNXI_CCU_DIV_OFFSET(_shift, _width, _offset) \ |
| _SUNXI_CCU_DIV_OFFSET_MAX_FLAGS(_shift, _width, _offset, 0, 0) |
| |
| #define _SUNXI_CCU_DIV(_shift, _width) \ |
| _SUNXI_CCU_DIV_FLAGS(_shift, _width, 0) |
| |
| struct ccu_div { |
| u32 enable; |
| |
| struct ccu_div_internal div; |
| struct ccu_mux_internal mux; |
| struct ccu_common common; |
| unsigned int fixed_post_div; |
| }; |
| |
| #define SUNXI_CCU_DIV_TABLE_WITH_GATE(_struct, _name, _parent, _reg, \ |
| _shift, _width, \ |
| _table, _gate, _flags) \ |
| struct ccu_div _struct = { \ |
| .div = _SUNXI_CCU_DIV_TABLE(_shift, _width, \ |
| _table), \ |
| .enable = _gate, \ |
| .common = { \ |
| .reg = _reg, \ |
| .hw.init = CLK_HW_INIT(_name, \ |
| _parent, \ |
| &ccu_div_ops, \ |
| _flags), \ |
| } \ |
| } |
| |
| |
| #define SUNXI_CCU_DIV_TABLE(_struct, _name, _parent, _reg, \ |
| _shift, _width, \ |
| _table, _flags) \ |
| SUNXI_CCU_DIV_TABLE_WITH_GATE(_struct, _name, _parent, _reg, \ |
| _shift, _width, _table, 0, \ |
| _flags) |
| |
| #define SUNXI_CCU_DIV_TABLE_HW(_struct, _name, _parent, _reg, \ |
| _shift, _width, \ |
| _table, _flags) \ |
| struct ccu_div _struct = { \ |
| .div = _SUNXI_CCU_DIV_TABLE(_shift, _width, \ |
| _table), \ |
| .common = { \ |
| .reg = _reg, \ |
| .hw.init = CLK_HW_INIT_HW(_name, \ |
| _parent, \ |
| &ccu_div_ops, \ |
| _flags), \ |
| } \ |
| } |
| |
| |
| #define SUNXI_CCU_M_WITH_MUX_TABLE_GATE(_struct, _name, \ |
| _parents, _table, \ |
| _reg, \ |
| _mshift, _mwidth, \ |
| _muxshift, _muxwidth, \ |
| _gate, _flags) \ |
| struct ccu_div _struct = { \ |
| .enable = _gate, \ |
| .div = _SUNXI_CCU_DIV(_mshift, _mwidth), \ |
| .mux = _SUNXI_CCU_MUX_TABLE(_muxshift, _muxwidth, _table), \ |
| .common = { \ |
| .reg = _reg, \ |
| .hw.init = CLK_HW_INIT_PARENTS(_name, \ |
| _parents, \ |
| &ccu_div_ops, \ |
| _flags), \ |
| }, \ |
| } |
| |
| #define SUNXI_CCU_M_WITH_MUX_TABLE_GATE_CLOSEST(_struct, _name, \ |
| _parents, _table, \ |
| _reg, \ |
| _mshift, _mwidth, \ |
| _muxshift, _muxwidth, \ |
| _gate, _flags) \ |
| struct ccu_div _struct = { \ |
| .enable = _gate, \ |
| .div = _SUNXI_CCU_DIV_FLAGS(_mshift, _mwidth, CLK_DIVIDER_ROUND_CLOSEST), \ |
| .mux = _SUNXI_CCU_MUX_TABLE(_muxshift, _muxwidth, _table), \ |
| .common = { \ |
| .reg = _reg, \ |
| .hw.init = CLK_HW_INIT_PARENTS(_name, \ |
| _parents, \ |
| &ccu_div_ops, \ |
| _flags), \ |
| .features = CCU_FEATURE_CLOSEST_RATE, \ |
| }, \ |
| } |
| |
| #define SUNXI_CCU_M_WITH_MUX_GATE(_struct, _name, _parents, _reg, \ |
| _mshift, _mwidth, _muxshift, _muxwidth, \ |
| _gate, _flags) \ |
| SUNXI_CCU_M_WITH_MUX_TABLE_GATE(_struct, _name, \ |
| _parents, NULL, \ |
| _reg, _mshift, _mwidth, \ |
| _muxshift, _muxwidth, \ |
| _gate, _flags) |
| |
| #define SUNXI_CCU_M_WITH_MUX_GATE_CLOSEST(_struct, _name, _parents, \ |
| _reg, _mshift, _mwidth, \ |
| _muxshift, _muxwidth, \ |
| _gate, _flags) \ |
| SUNXI_CCU_M_WITH_MUX_TABLE_GATE_CLOSEST(_struct, _name, \ |
| _parents, NULL, \ |
| _reg, _mshift, _mwidth, \ |
| _muxshift, _muxwidth, \ |
| _gate, _flags) |
| |
| #define SUNXI_CCU_M_WITH_MUX(_struct, _name, _parents, _reg, \ |
| _mshift, _mwidth, _muxshift, _muxwidth, \ |
| _flags) \ |
| SUNXI_CCU_M_WITH_MUX_TABLE_GATE(_struct, _name, \ |
| _parents, NULL, \ |
| _reg, _mshift, _mwidth, \ |
| _muxshift, _muxwidth, \ |
| 0, _flags) |
| |
| |
| #define SUNXI_CCU_M_WITH_GATE(_struct, _name, _parent, _reg, \ |
| _mshift, _mwidth, _gate, \ |
| _flags) \ |
| struct ccu_div _struct = { \ |
| .enable = _gate, \ |
| .div = _SUNXI_CCU_DIV(_mshift, _mwidth), \ |
| .common = { \ |
| .reg = _reg, \ |
| .hw.init = CLK_HW_INIT(_name, \ |
| _parent, \ |
| &ccu_div_ops, \ |
| _flags), \ |
| }, \ |
| } |
| |
| #define SUNXI_CCU_M(_struct, _name, _parent, _reg, _mshift, _mwidth, \ |
| _flags) \ |
| SUNXI_CCU_M_WITH_GATE(_struct, _name, _parent, _reg, \ |
| _mshift, _mwidth, 0, _flags) |
| |
| #define SUNXI_CCU_M_DATA_WITH_MUX_GATE(_struct, _name, _parents, _reg, \ |
| _mshift, _mwidth, \ |
| _muxshift, _muxwidth, \ |
| _gate, _flags) \ |
| struct ccu_div _struct = { \ |
| .enable = _gate, \ |
| .div = _SUNXI_CCU_DIV(_mshift, _mwidth), \ |
| .mux = _SUNXI_CCU_MUX(_muxshift, _muxwidth), \ |
| .common = { \ |
| .reg = _reg, \ |
| .hw.init = CLK_HW_INIT_PARENTS_DATA(_name, \ |
| _parents, \ |
| &ccu_div_ops, \ |
| _flags), \ |
| }, \ |
| } |
| |
| #define SUNXI_CCU_M_DATA_WITH_MUX(_struct, _name, _parents, _reg, \ |
| _mshift, _mwidth, \ |
| _muxshift, _muxwidth, \ |
| _flags) \ |
| SUNXI_CCU_M_DATA_WITH_MUX_GATE(_struct, _name, _parents, _reg, \ |
| _mshift, _mwidth, \ |
| _muxshift, _muxwidth, \ |
| 0, _flags) |
| |
| #define SUNXI_CCU_M_HW_WITH_MUX_GATE(_struct, _name, _parents, _reg, \ |
| _mshift, _mwidth, _muxshift, _muxwidth, \ |
| _gate, _flags) \ |
| struct ccu_div _struct = { \ |
| .enable = _gate, \ |
| .div = _SUNXI_CCU_DIV(_mshift, _mwidth), \ |
| .mux = _SUNXI_CCU_MUX(_muxshift, _muxwidth), \ |
| .common = { \ |
| .reg = _reg, \ |
| .hw.init = CLK_HW_INIT_PARENTS_HW(_name, \ |
| _parents, \ |
| &ccu_div_ops, \ |
| _flags), \ |
| }, \ |
| } |
| |
| #define SUNXI_CCU_M_HWS_WITH_GATE(_struct, _name, _parent, _reg, \ |
| _mshift, _mwidth, _gate, \ |
| _flags) \ |
| struct ccu_div _struct = { \ |
| .enable = _gate, \ |
| .div = _SUNXI_CCU_DIV(_mshift, _mwidth), \ |
| .common = { \ |
| .reg = _reg, \ |
| .hw.init = CLK_HW_INIT_HWS(_name, \ |
| _parent, \ |
| &ccu_div_ops, \ |
| _flags), \ |
| }, \ |
| } |
| |
| #define SUNXI_CCU_M_HWS(_struct, _name, _parent, _reg, _mshift, \ |
| _mwidth, _flags) \ |
| SUNXI_CCU_M_HWS_WITH_GATE(_struct, _name, _parent, _reg, \ |
| _mshift, _mwidth, 0, _flags) |
| |
| static inline struct ccu_div *hw_to_ccu_div(struct clk_hw *hw) |
| { |
| struct ccu_common *common = hw_to_ccu_common(hw); |
| |
| return container_of(common, struct ccu_div, common); |
| } |
| |
| extern const struct clk_ops ccu_div_ops; |
| |
| #endif /* _CCU_DIV_H_ */ |