On Mon, Dec 24, 2018 at 07:57:41PM +0800, Liviu Dudau wrote: > On Fri, Dec 21, 2018 at 09:58:55AM +0000, james qian wang (Arm Technology China) wrote: > > 1. Added a brief definition of komeda_dev/pipeline/component, this change > > didn't add the detailed component features and capabilities, which will > > be added in the following changes. > > 2. Corresponding resources discovery and initialzation functions. > > > > Signed-off-by: James (Qian) Wang <james.qian.wang@xxxxxxx> > > > > Changes in v3: > > - Fixed style problem found by checkpatch.pl --strict. > > > > Changes in v2: > > - Unified abbreviation of "pipeline" to "pipe". > > --- > > drivers/gpu/drm/arm/Kconfig | 2 + > > drivers/gpu/drm/arm/Makefile | 1 + > > drivers/gpu/drm/arm/display/Kbuild | 3 + > > drivers/gpu/drm/arm/display/Kconfig | 14 + > > .../drm/arm/display/include/malidp_product.h | 23 ++ > > .../drm/arm/display/include/malidp_utils.h | 16 + > > drivers/gpu/drm/arm/display/komeda/Makefile | 11 + > > .../gpu/drm/arm/display/komeda/komeda_dev.c | 117 ++++++ > > .../gpu/drm/arm/display/komeda/komeda_dev.h | 98 +++++ > > .../drm/arm/display/komeda/komeda_pipeline.c | 198 ++++++++++ > > .../drm/arm/display/komeda/komeda_pipeline.h | 350 ++++++++++++++++++ > > 11 files changed, 833 insertions(+) > > create mode 100644 drivers/gpu/drm/arm/display/Kbuild > > create mode 100644 drivers/gpu/drm/arm/display/Kconfig > > create mode 100644 drivers/gpu/drm/arm/display/include/malidp_product.h > > create mode 100644 drivers/gpu/drm/arm/display/include/malidp_utils.h > > create mode 100644 drivers/gpu/drm/arm/display/komeda/Makefile > > create mode 100644 drivers/gpu/drm/arm/display/komeda/komeda_dev.c > > create mode 100644 drivers/gpu/drm/arm/display/komeda/komeda_dev.h > > create mode 100644 drivers/gpu/drm/arm/display/komeda/komeda_pipeline.c > > create mode 100644 drivers/gpu/drm/arm/display/komeda/komeda_pipeline.h > > > > diff --git a/drivers/gpu/drm/arm/Kconfig b/drivers/gpu/drm/arm/Kconfig > > index f9f7761cb2f4..a204103b3efb 100644 > > --- a/drivers/gpu/drm/arm/Kconfig > > +++ b/drivers/gpu/drm/arm/Kconfig > > @@ -37,4 +37,6 @@ config DRM_MALI_DISPLAY > > > > If compiled as a module it will be called mali-dp. > > > > +source "drivers/gpu/drm/arm/display/Kconfig" > > + > > endmenu > > diff --git a/drivers/gpu/drm/arm/Makefile b/drivers/gpu/drm/arm/Makefile > > index 3bf31d1a4722..120bef801fcf 100644 > > --- a/drivers/gpu/drm/arm/Makefile > > +++ b/drivers/gpu/drm/arm/Makefile > > @@ -3,3 +3,4 @@ obj-$(CONFIG_DRM_HDLCD) += hdlcd.o > > mali-dp-y := malidp_drv.o malidp_hw.o malidp_planes.o malidp_crtc.o > > mali-dp-y += malidp_mw.o > > obj-$(CONFIG_DRM_MALI_DISPLAY) += mali-dp.o > > +obj-$(CONFIG_DRM_KOMEDA) += display/ > > diff --git a/drivers/gpu/drm/arm/display/Kbuild b/drivers/gpu/drm/arm/display/Kbuild > > new file mode 100644 > > index 000000000000..382f1ca831e4 > > --- /dev/null > > +++ b/drivers/gpu/drm/arm/display/Kbuild > > @@ -0,0 +1,3 @@ > > +# SPDX-License-Identifier: GPL-2.0 > > + > > +obj-$(CONFIG_DRM_KOMEDA) += komeda/ > > diff --git a/drivers/gpu/drm/arm/display/Kconfig b/drivers/gpu/drm/arm/display/Kconfig > > new file mode 100644 > > index 000000000000..cec0639e3aa1 > > --- /dev/null > > +++ b/drivers/gpu/drm/arm/display/Kconfig > > @@ -0,0 +1,14 @@ > > +# SPDX-License-Identifier: GPL-2.0 > > +config DRM_KOMEDA > > + tristate "ARM Komeda display driver" > > + depends on DRM && OF > > + depends on COMMON_CLK > > + select DRM_KMS_HELPER > > + select DRM_KMS_CMA_HELPER > > + select DRM_GEM_CMA_HELPER > > + select VIDEOMODE_HELPERS > > + help > > + Choose this option if you want to compile the ARM Komeda display > > + Processor driver. It supports the D71 variants of the hardware. > > + > > + If compiled as a module it will be called komeda. > > diff --git a/drivers/gpu/drm/arm/display/include/malidp_product.h b/drivers/gpu/drm/arm/display/include/malidp_product.h > > new file mode 100644 > > index 000000000000..b35fc5db866b > > --- /dev/null > > +++ b/drivers/gpu/drm/arm/display/include/malidp_product.h > > @@ -0,0 +1,23 @@ > > +/* SPDX-License-Identifier: GPL-2.0 */ > > +/* > > + * (C) COPYRIGHT 2018 ARM Limited. All rights reserved. > > + * Author: James.Qian.Wang <james.qian.wang@xxxxxxx> > > + * > > + */ > > +#ifndef _MALIDP_PRODUCT_H_ > > +#define _MALIDP_PRODUCT_H_ > > + > > +/* Product identification */ > > +#define MALIDP_CORE_ID(__product, __major, __minor, __status) \ > > + ((((__product) & 0xFFFF) << 16) | (((__major) & 0xF) << 12) | \ > > + (((__minor) & 0xF) << 8) | ((__status) & 0xFF)) > > + > > +#define MALIDP_CORE_ID_PRODUCT_ID(__core_id) ((__u32)(__core_id) >> 16) > > +#define MALIDP_CORE_ID_MAJOR(__core_id) (((__u32)(__core_id) >> 12) & 0xF) > > +#define MALIDP_CORE_ID_MINOR(__core_id) (((__u32)(__core_id) >> 8) & 0xF) > > +#define MALIDP_CORE_ID_STATUS(__core_id) (((__u32)(__core_id)) & 0xFF) > > + > > +/* Mali-display product IDs */ > > +#define MALIDP_D71_PRODUCT_ID 0x0071 > > + > > +#endif /* _MALIDP_PRODUCT_H_ */ > > diff --git a/drivers/gpu/drm/arm/display/include/malidp_utils.h b/drivers/gpu/drm/arm/display/include/malidp_utils.h > > new file mode 100644 > > index 000000000000..63cc47cefcf8 > > --- /dev/null > > +++ b/drivers/gpu/drm/arm/display/include/malidp_utils.h > > @@ -0,0 +1,16 @@ > > +/* SPDX-License-Identifier: GPL-2.0 */ > > +/* > > + * (C) COPYRIGHT 2018 ARM Limited. All rights reserved. > > + * Author: James.Qian.Wang <james.qian.wang@xxxxxxx> > > + * > > + */ > > +#ifndef _MALIDP_UTILS_ > > +#define _MALIDP_UTILS_ > > + > > +#define has_bit(nr, mask) (BIT(nr) & (mask)) > > +#define has_bits(bits, mask) (((bits) & (mask)) == (bits)) > > + > > +#define dp_for_each_set_bit(bit, mask) \ > > + for_each_set_bit((bit), ((unsigned long *)&(mask)), sizeof(mask) * 8) > > Given that most of our registers (and masks, by extension) are 32bit, I > think it might be better to use 32 instead of sizeof(mask) * 8 as we > don't want to introduce subtle bugs in the future. And I don't think you > need the (unsigned long *) cast either. > Sorry, This MACRO is not only for the registers, but which is mainly used by the Komeda-CORE for iterating pipeline components by mask (32bit), or iterating active_inputs (16bit) of a component, maybe also will be used by the event/irq handling, komeda event will be 64bit value every bit will indicate a seperated event or error. > > + > > +#endif /* _MALIDP_UTILS_ */ > > diff --git a/drivers/gpu/drm/arm/display/komeda/Makefile b/drivers/gpu/drm/arm/display/komeda/Makefile > > new file mode 100644 > > index 000000000000..5b44e36509b1 > > --- /dev/null > > +++ b/drivers/gpu/drm/arm/display/komeda/Makefile > > @@ -0,0 +1,11 @@ > > +# SPDX-License-Identifier: GPL-2.0 > > + > > +ccflags-y := \ > > + -I$(src)/../include \ > > + -I$(src) > > + > > +komeda-y := \ > > + komeda_dev.o \ > > + komeda_pipeline.o \ > > + > > +obj-$(CONFIG_DRM_KOMEDA) += komeda.o > > diff --git a/drivers/gpu/drm/arm/display/komeda/komeda_dev.c b/drivers/gpu/drm/arm/display/komeda/komeda_dev.c > > new file mode 100644 > > index 000000000000..887a17005367 > > --- /dev/null > > +++ b/drivers/gpu/drm/arm/display/komeda/komeda_dev.c > > @@ -0,0 +1,117 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * (C) COPYRIGHT 2018 ARM Limited. All rights reserved. > > + * Author: James.Qian.Wang <james.qian.wang@xxxxxxx> > > + * > > + */ > > +#include <linux/platform_device.h> > > +#include <linux/clk.h> > > +#include <linux/io.h> > > +#include <linux/of_device.h> > > +#include <linux/of_graph.h> > > +#include <linux/version.h> > > +#include "komeda_dev.h" > > + > > +struct komeda_dev *komeda_dev_create(struct device *dev) > > +{ > > + struct platform_device *pdev = to_platform_device(dev); > > + const struct komeda_product_data *product; > > + struct komeda_dev *mdev; > > + struct resource *io_res; > > + int err = 0; > > + > > + product = of_device_get_match_data(dev); > > + if (!product) > > + return ERR_PTR(-ENODEV); > > + > > + io_res = platform_get_resource(pdev, IORESOURCE_MEM, 0); > > + if (!io_res) { > > + DRM_ERROR("No registers defined.\n"); > > + return ERR_PTR(-ENODEV); > > + } > > + > > + mdev = devm_kzalloc(dev, sizeof(*mdev), GFP_KERNEL); > > + if (!mdev) > > + return ERR_PTR(-ENOMEM); > > + > > + mdev->dev = dev; > > + mdev->reg_base = devm_ioremap_resource(dev, io_res); > > + if (IS_ERR(mdev->reg_base)) { > > + DRM_ERROR("Map register space failed.\n"); > > + err = PTR_ERR(mdev->reg_base); > > + mdev->reg_base = NULL; > > + goto err_cleanup; > > + } > > + > > + mdev->pclk = devm_clk_get(dev, "pclk"); > > + if (IS_ERR(mdev->pclk)) { > > + DRM_ERROR("Get APB clk failed.\n"); > > + err = PTR_ERR(mdev->pclk); > > + mdev->pclk = NULL; > > + goto err_cleanup; > > + } > > + > > + /* Enable APB clock to access the registers */ > > + clk_prepare_enable(mdev->pclk); > > + > > + mdev->funcs = product->identify(mdev->reg_base, &mdev->chip); > > + if (!komeda_product_match(mdev, product->product_id)) { > > + DRM_ERROR("DT configured %x mismatch with real HW %x.\n", > > + product->product_id, > > + MALIDP_CORE_ID_PRODUCT_ID(mdev->chip.core_id)); > > + err = -ENODEV; > > + goto err_cleanup; > > + } > > + > > + DRM_INFO("Found ARM Mali-D%x version r%dp%d\n", > > + MALIDP_CORE_ID_PRODUCT_ID(mdev->chip.core_id), > > + MALIDP_CORE_ID_MAJOR(mdev->chip.core_id), > > + MALIDP_CORE_ID_MINOR(mdev->chip.core_id)); > > + > > + err = mdev->funcs->enum_resources(mdev); > > + if (err) { > > + DRM_ERROR("enumerate display resource failed.\n"); > > + goto err_cleanup; > > + } > > + > > + return mdev; > > + > > +err_cleanup: > > + komeda_dev_destroy(mdev); > > + return ERR_PTR(err); > > +} > > + > > +void komeda_dev_destroy(struct komeda_dev *mdev) > > +{ > > + struct device *dev = mdev->dev; > > + struct komeda_dev_funcs *funcs = mdev->funcs; > > + int i; > > + > > + for (i = 0; i < mdev->n_pipelines; i++) { > > + komeda_pipeline_destroy(mdev, mdev->pipelines[i]); > > + mdev->pipelines[i] = NULL; > > + } > > + > > + mdev->n_pipelines = 0; > > + > > + if (funcs && funcs->cleanup) > > + funcs->cleanup(mdev); > > + > > + if (mdev->reg_base) { > > + devm_iounmap(dev, mdev->reg_base); > > + mdev->reg_base = NULL; > > + } > > + > > + if (mdev->mclk) { > > + devm_clk_put(dev, mdev->mclk); > > + mdev->mclk = NULL; > > + } > > + > > + if (mdev->pclk) { > > + clk_disable_unprepare(mdev->pclk); > > + devm_clk_put(dev, mdev->pclk); > > + mdev->pclk = NULL; > > + } > > + > > + devm_kfree(dev, mdev); > > +} > > diff --git a/drivers/gpu/drm/arm/display/komeda/komeda_dev.h b/drivers/gpu/drm/arm/display/komeda/komeda_dev.h > > new file mode 100644 > > index 000000000000..ad8fa160eff9 > > --- /dev/null > > +++ b/drivers/gpu/drm/arm/display/komeda/komeda_dev.h > > @@ -0,0 +1,98 @@ > > +/* SPDX-License-Identifier: GPL-2.0 */ > > +/* > > + * (C) COPYRIGHT 2018 ARM Limited. All rights reserved. > > + * Author: James.Qian.Wang <james.qian.wang@xxxxxxx> > > + * > > + */ > > +#ifndef _KOMEDA_DEV_H_ > > +#define _KOMEDA_DEV_H_ > > + > > +#include <linux/device.h> > > +#include <linux/interrupt.h> > > You don't need this header to be included here. OK. > > > +#include "komeda_pipeline.h" > > +#include "malidp_product.h" > > + > > +/* malidp device id */ > > +enum { > > + MALI_D71 = 0, > > +}; > > + > > +/* pipeline DT ports */ > > +enum { > > + KOMEDA_OF_PORT_OUTPUT = 0, > > + KOMEDA_OF_PORT_COPROC = 1, > > +}; > > + > > +struct komeda_chip_info { > > + u32 arch_id; > > + u32 core_id; > > + u32 core_info; > > + u32 bus_width; > > +}; > > + > > +struct komeda_product_data { > > + u32 product_id; > > + struct komeda_dev_funcs *(*identify)(u32 __iomem *reg, > > + struct komeda_chip_info *info); > > +}; > > + > > +struct komeda_dev; > > + > > +/** > > + * struct komeda_dev_funcs > > + * > > + * Supplied by chip level and returned by the chip entry function xxx_identify, > > + */ > > +struct komeda_dev_funcs { > > + /** > > + * @enum_resources: > > + * > > + * for CHIP to report or add pipeline and component resources to CORE > > + */ > > + int (*enum_resources)(struct komeda_dev *mdev); > > + /** @cleanup: call to chip to cleanup komeda_dev->chip data */ > > + void (*cleanup)(struct komeda_dev *mdev); > > +}; > > + > > +/** > > + * struct komeda_dev > > + * > > + * Pipeline and component are used to describe how to handle the pixel data. > > + * komeda_device is for describing the whole view of the device, and the > > + * control-abilites of device. > > + */ > > +struct komeda_dev { > > + struct device *dev; > > + u32 __iomem *reg_base; > > + > > + struct komeda_chip_info chip; > > + > > + /** @pclk: APB clock for register access */ > > + struct clk *pclk; > > + /** @mck: HW main engine clk */ > > + struct clk *mclk; > > + > > + int n_pipelines; > > + struct komeda_pipeline *pipelines[KOMEDA_MAX_PIPELINES]; > > + > > + /** @funcs: chip funcs to access to HW */ > > + struct komeda_dev_funcs *funcs; > > + /** > > + * @chip_data: > > + * > > + * chip data will be added by &komeda_dev_funcs.enum_resources() and > > + * destroyed by &komeda_dev_funcs.cleanup() > > + */ > > + void *chip_data; > > +}; > > + > > +static inline bool > > +komeda_product_match(struct komeda_dev *mdev, u32 target) > > +{ > > + return MALIDP_CORE_ID_PRODUCT_ID(mdev->chip.core_id) == target; > > +} > > + > > +struct komeda_dev *komeda_dev_create(struct device *dev); > > +void komeda_dev_destroy(struct komeda_dev *mdev); > > + > > +#endif /*_KOMEDA_DEV_H_*/ > > diff --git a/drivers/gpu/drm/arm/display/komeda/komeda_pipeline.c b/drivers/gpu/drm/arm/display/komeda/komeda_pipeline.c > > new file mode 100644 > > index 000000000000..9293598b0533 > > --- /dev/null > > +++ b/drivers/gpu/drm/arm/display/komeda/komeda_pipeline.c > > @@ -0,0 +1,198 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * (C) COPYRIGHT 2018 ARM Limited. All rights reserved. > > + * Author: James.Qian.Wang <james.qian.wang@xxxxxxx> > > + * > > + */ > > +#include <linux/clk.h> > > +#include "komeda_dev.h" > > +#include "komeda_pipeline.h" > > + > > +/** komeda_pipeline_add - Add a pipeline to &komeda_dev */ > > +struct komeda_pipeline * > > +komeda_pipeline_add(struct komeda_dev *mdev, size_t size, > > + struct komeda_pipeline_funcs *funcs) > > +{ > > + struct komeda_pipeline *pipe; > > + > > + if (mdev->n_pipelines + 1 > KOMEDA_MAX_PIPELINES) { > > + DRM_ERROR("Exceed max support %d pipelines.\n", > > + KOMEDA_MAX_PIPELINES); > > + return NULL; > > + } > > + > > + if (size < sizeof(*pipe)) { > > + DRM_ERROR("Request pipeline size too small.\n"); > > + return NULL; > > + } > > + > > + pipe = devm_kzalloc(mdev->dev, size, GFP_KERNEL); > > + if (!pipe) > > + return NULL; > > + > > + pipe->mdev = mdev; > > + pipe->id = mdev->n_pipelines; > > + pipe->funcs = funcs; > > + > > + mdev->pipelines[mdev->n_pipelines] = pipe; > > + mdev->n_pipelines++; > > + > > + return pipe; > > +} > > + > > +void komeda_pipeline_destroy(struct komeda_dev *mdev, > > + struct komeda_pipeline *pipe) > > +{ > > + struct komeda_component *c; > > + int i; > > + > > + dp_for_each_set_bit(i, pipe->avail_comps) { > > + c = komeda_pipeline_get_component(pipe, i); > > + > > Unnecessary empty line > OK, will del it. > > + komeda_component_destroy(mdev, c); > > + } > > + > > + clk_put(pipe->pxlclk); > > + clk_put(pipe->aclk); > > + > > + devm_kfree(mdev->dev, pipe); > > +} > > + > > +struct komeda_component ** > > +komeda_pipeline_get_component_pos(struct komeda_pipeline *pipe, int id) > > +{ > > + struct komeda_dev *mdev = pipe->mdev; > > + struct komeda_pipeline *temp = NULL; > > + struct komeda_component **pos = NULL; > > + > > + switch (id) { > > + case KOMEDA_COMPONENT_LAYER0: > > + case KOMEDA_COMPONENT_LAYER1: > > + case KOMEDA_COMPONENT_LAYER2: > > + case KOMEDA_COMPONENT_LAYER3: > > + pos = to_cpos(pipe->layers[id - KOMEDA_COMPONENT_LAYER0]); > > + break; > > + case KOMEDA_COMPONENT_WB_LAYER: > > + pos = to_cpos(pipe->wb_layer); > > + break; > > + case KOMEDA_COMPONENT_COMPIZ0: > > + case KOMEDA_COMPONENT_COMPIZ1: > > + temp = mdev->pipelines[id - KOMEDA_COMPONENT_COMPIZ0]; > > + if (!temp) { > > + DRM_ERROR("compiz-%d doesn't exist.\n", id); > > + return NULL; > > + } > > + pos = to_cpos(temp->compiz); > > + break; > > + case KOMEDA_COMPONENT_SCALER0: > > + case KOMEDA_COMPONENT_SCALER1: > > + pos = to_cpos(pipe->scalers[id - KOMEDA_COMPONENT_SCALER0]); > > + break; > > + case KOMEDA_COMPONENT_IPS0: > > + case KOMEDA_COMPONENT_IPS1: > > + temp = mdev->pipelines[id - KOMEDA_COMPONENT_IPS0]; > > + if (!temp) { > > + DRM_ERROR("ips-%d doesn't exist.\n", id); > > + return NULL; > > + } > > + pos = to_cpos(temp->improc); > > + break; > > + case KOMEDA_COMPONENT_TIMING_CTRLR: > > + pos = to_cpos(pipe->ctrlr); > > + break; > > + default: > > + pos = NULL; > > + DRM_ERROR("Unknown pipeline resource ID: %d.\n", id); > > + break; > > + } > > + > > + return pos; > > +} > > + > > +struct komeda_component * > > +komeda_pipeline_get_component(struct komeda_pipeline *pipe, int id) > > +{ > > + struct komeda_component **pos = NULL; > > + struct komeda_component *c = NULL; > > + > > + pos = komeda_pipeline_get_component_pos(pipe, id); > > + if (pos) > > + c = *pos; > > + > > + return c; > > +} > > + > > +/** komeda_component_add - Add a component to &komeda_pipeline */ > > +struct komeda_component * > > +komeda_component_add(struct komeda_pipeline *pipe, > > + size_t comp_sz, u32 id, u32 hw_id, > > + struct komeda_component_funcs *funcs, > > + u8 max_active_inputs, u32 supported_inputs, > > + u8 max_active_outputs, u32 __iomem *reg, > > + const char *name_fmt, ...) > > +{ > > + struct komeda_component **pos; > > + struct komeda_component *c; > > + int idx, *num = NULL; > > + > > + if (max_active_inputs > KOMEDA_COMPONENT_N_INPUTS) { > > + WARN(1, "please large KOMEDA_COMPONENT_N_INPUTS to %d.\n", > > + max_active_inputs); > > + return NULL; > > + } > > + > > + pos = komeda_pipeline_get_component_pos(pipe, id); > > + if (!pos || !(*pos)) > > + return NULL; > > + > > + if (has_bit(id, KOMEDA_PIPELINE_LAYERS)) { > > + idx = id - KOMEDA_COMPONENT_LAYER0; > > + num = &pipe->n_layers; > > + if (idx != pipe->n_layers) { > > + DRM_ERROR("please add Layer by id sequence.\n"); > > + return NULL; > > + } > > + } else if (has_bit(id, KOMEDA_PIPELINE_SCALERS)) { > > + idx = id - KOMEDA_COMPONENT_SCALER0; > > + num = &pipe->n_scalers; > > + if (idx != pipe->n_scalers) { > > + DRM_ERROR("please add Scaler by id sequence.\n"); > > + return NULL; > > + } > > + } > > + > > + c = devm_kzalloc(pipe->mdev->dev, comp_sz, GFP_KERNEL); > > + if (!c) > > + return NULL; > > + > > + c->id = id; > > + c->hw_id = hw_id; > > + c->reg = reg; > > + c->pipeline = pipe; > > + c->max_active_inputs = max_active_inputs; > > + c->max_active_outputs = max_active_outputs; > > + c->supported_inputs = supported_inputs; > > + c->funcs = funcs; > > + > > + if (name_fmt) { > > + va_list args; > > + > > + va_start(args, name_fmt); > > + vsnprintf(c->name, sizeof(c->name), name_fmt, args); > > + va_end(args); > > + } > > + > > + if (num) > > + *num = *num + 1; > > + > > + pipe->avail_comps |= BIT(c->id); > > + *pos = c; > > + > > + return c; > > +} > > + > > +void komeda_component_destroy(struct komeda_dev *mdev, > > + struct komeda_component *c) > > +{ > > + devm_kfree(mdev->dev, c); > > +} > > diff --git a/drivers/gpu/drm/arm/display/komeda/komeda_pipeline.h b/drivers/gpu/drm/arm/display/komeda/komeda_pipeline.h > > new file mode 100644 > > index 000000000000..2174796d47c5 > > --- /dev/null > > +++ b/drivers/gpu/drm/arm/display/komeda/komeda_pipeline.h > > @@ -0,0 +1,350 @@ > > +/* SPDX-License-Identifier: GPL-2.0 */ > > +/* > > + * (C) COPYRIGHT 2018 ARM Limited. All rights reserved. > > + * Author: James.Qian.Wang <james.qian.wang@xxxxxxx> > > + * > > + */ > > +#ifndef _KOMEDA_PIPELINE_H_ > > +#define _KOMEDA_PIPELINE_H_ > > + > > +#include <linux/types.h> > > +#include <linux/of.h> > > +#include <linux/bitops.h> > > of.h and bitops.h are unnecessary in this header file. OK, will del these unnecessary headers. > > > +#include <drm/drm_atomic.h> > > +#include <drm/drm_atomic_helper.h> > > +#include "malidp_utils.h" > > + > > +#define KOMEDA_MAX_PIPELINES 2 > > +#define KOMEDA_PIPELINE_MAX_LAYERS 4 > > +#define KOMEDA_PIPELINE_MAX_SCALERS 2 > > +#define KOMEDA_COMPONENT_N_INPUTS 5 > > + > > +/* pipeline component IDs */ > > +enum { > > + KOMEDA_COMPONENT_LAYER0 = 0, > > + KOMEDA_COMPONENT_LAYER1 = 1, > > + KOMEDA_COMPONENT_LAYER2 = 2, > > + KOMEDA_COMPONENT_LAYER3 = 3, > > + KOMEDA_COMPONENT_WB_LAYER = 7, /* write back layer */ > > + KOMEDA_COMPONENT_SCALER0 = 8, > > + KOMEDA_COMPONENT_SCALER1 = 9, > > + KOMEDA_COMPONENT_SPLITTER = 12, > > + KOMEDA_COMPONENT_MERGER = 14, > > + KOMEDA_COMPONENT_COMPIZ0 = 16, /* compositor */ > > + KOMEDA_COMPONENT_COMPIZ1 = 17, > > + KOMEDA_COMPONENT_IPS0 = 20, /* post image processor */ > > + KOMEDA_COMPONENT_IPS1 = 21, > > + KOMEDA_COMPONENT_TIMING_CTRLR = 22, /* timing controller */ > > +}; > > + > > +#define KOMEDA_PIPELINE_LAYERS (BIT(KOMEDA_COMPONENT_LAYER0) |\ > > + BIT(KOMEDA_COMPONENT_LAYER1) |\ > > + BIT(KOMEDA_COMPONENT_LAYER2) |\ > > + BIT(KOMEDA_COMPONENT_LAYER3)) > > + > > +#define KOMEDA_PIPELINE_SCALERS (BIT(KOMEDA_COMPONENT_SCALER0) |\ > > + BIT(KOMEDA_COMPONENT_SCALER1)) > > + > > +#define KOMEDA_PIPELINE_COMPIZS (BIT(KOMEDA_COMPONENT_COMPIZ0) |\ > > + BIT(KOMEDA_COMPONENT_COMPIZ1)) > > + > > +#define KOMEDA_PIPELINE_IMPROCS (BIT(KOMEDA_COMPONENT_IPS0) |\ > > + BIT(KOMEDA_COMPONENT_IPS1)) > > +struct komeda_component; > > +struct komeda_component_state; > > + > > +/** komeda_component_funcs - component control functions */ > > +struct komeda_component_funcs { > > + /** @validate: optional, > > + * component may has special requirements or limitations, this function > > + * supply HW the ability to do the further HW specific check. > > + */ > > + int (*validate)(struct komeda_component *c, > > + struct komeda_component_state *state); > > + /** @update: update is a active update */ > > + void (*update)(struct komeda_component *c, > > + struct komeda_component_state *state); > > + /** @disable: disable component */ > > + void (*disable)(struct komeda_component *c); > > + /** @dump_register: Optional, dump registers to seq_file */ > > + void (*dump_register)(struct komeda_component *c, struct seq_file *seq); > > +}; > > + > > +/** > > + * struct komeda_component > > + * > > + * struct komeda_component describe the data flow capabilities for how to link a > > + * component into the display pipeline. > > + * all specified components are subclass of this structure. > > + */ > > +struct komeda_component { > > + /** @obj: treat component as private obj */ > > + struct drm_private_obj obj; > > + /** @pipeline: the komeda pipeline this component belongs to */ > > + struct komeda_pipeline *pipeline; > > + /** @name: component name */ > > + char name[32]; > > + /** > > + * @reg: > > + * component register base, > > + * which is initialized by chip and used by chip only > > + */ > > + u32 __iomem *reg; > > + /** @id: component id */ > > + u32 id; > > + /** @hw_ic: component hw id, > > + * which is initialized by chip and used by chip only > > + */ > > + u32 hw_id; > > + > > + /** > > + * @max_active_inputs: > > + * @max_active_outpus: > > + * > > + * maximum number of inputs/outputs that can be active in the same time > > + * Note: > > + * the number isn't the bit number of @supported_inputs or > > + * @supported_outputs, but may be less than it, since component may not > > + * support enabling all @supported_inputs/outputs at the same time. > > + */ > > + u8 max_active_inputs; > > + u8 max_active_outputs; > > + /** > > + * @supported_inputs: > > + * @supported_outputs: > > + * > > + * bitmask of BIT(component->id) for the supported inputs/outputs > > + * describes the possibilities of how a component is linked into a > > + * pipeline. > > + */ > > + u32 supported_inputs; > > + u32 supported_outputs; > > + > > + /** > > + * @funcs: chip functions to access HW > > + */ > > + struct komeda_component_funcs *funcs; > > +}; > > + > > +/** > > + * struct komeda_component_output > > + * > > + * a component has multiple outputs, if want to know where the data > > + * comes from, only know the component is not enough, we still need to know > > + * its output port > > + */ > > +struct komeda_component_output { > > + /** @component: indicate which component the data comes from */ > > + struct komeda_component *component; > > + /** @output_port: > > + * the output port of the &komeda_component_output.component > > + */ > > + u8 output_port; > > +}; > > + > > +/** > > + * struct komeda_component_state > > + * > > + * component_state is the data flow configuration of the component, and it's > > + * the superclass of all specific component_state like @komeda_layer_state, > > + * @komeda_scaler_state > > + */ > > +struct komeda_component_state { > > + /** @obj: tracking component_state by drm_atomic_state */ > > + struct drm_private_state obj; > > + struct komeda_component *component; > > + /** > > + * @binding_user: > > + * currently bound user, the user can be crtc/plane/wb_conn, which is > > + * valid decided by @component and @inputs > > + * > > + * - Layer: its user always is plane. > > + * - compiz/improc/timing_ctrlr: the user is crtc. > > + * - wb_layer: wb_conn; > > + * - scaler: plane when input is layer, wb_conn if input is compiz. > > + */ > > + union { > > + struct drm_crtc *crtc; > > + struct drm_plane *plane; > > + struct drm_connector *wb_conn; > > + void *binding_user; > > + }; > > + /** > > + * @active_inputs: > > + * > > + * active_inputs is bitmask of @inputs index > > + * > > + * - active_inputs = changed_active_inputs + unchanged_active_inputs > > + * - affected_inputs = old->active_inputs + new->active_inputs; > > + * - disabling_inputs = affected_inputs ^ active_inputs; > > + * - changed_inputs = disabling_inputs + changed_active_inputs; > > + * > > + * NOTE: > > + * changed_inputs doesn't include all active_input but only > > + * @changed_active_inputs, and this bitmask can be used in chip > > + * level for dirty update. > > + */ > > + u16 active_inputs; > > + u16 changed_active_inputs; > > + u16 affected_inputs; > > + /** > > + * @inputs: > > + * > > + * the specific inputs[i] only valid on BIT(i) has been set in > > + * @active_inputs, if not the inputs[i] is undefined. > > + */ > > + struct komeda_component_output inputs[KOMEDA_COMPONENT_N_INPUTS]; > > +}; > > + > > +static inline u16 component_disabling_inputs(struct komeda_component_state *st) > > +{ > > + return st->affected_inputs ^ st->active_inputs; > > +} > > + > > +static inline u16 component_changed_inputs(struct komeda_component_state *st) > > +{ > > + return component_disabling_inputs(st) | st->changed_active_inputs; > > +} > > + > > +#define to_comp(__c) (((__c) == NULL) ? NULL : &((__c)->base)) > > +#define to_cpos(__c) ((struct komeda_component **)&(__c)) > > + > > +/* these structures are going to be filled in in uture patches */ > > +struct komeda_layer { > > + struct komeda_component base; > > + /* layer specific features and caps */ > > +}; > > + > > +struct komeda_layer_state { > > + struct komeda_component_state base; > > + /* layer specific configuration state */ > > +}; > > + > > +struct komeda_compiz { > > + struct komeda_component base; > > + /* compiz specific features and caps */ > > +}; > > + > > +struct komeda_compiz_state { > > + struct komeda_component_state base; > > + /* compiz specific configuration state */ > > +}; > > + > > +struct komeda_scaler { > > + struct komeda_component base; > > + /* scaler features and caps */ > > +}; > > + > > +struct komeda_scaler_state { > > + struct komeda_component_state base; > > +}; > > + > > +struct komeda_improc { > > + struct komeda_component base; > > +}; > > + > > +struct komeda_improc_state { > > + struct komeda_component_state base; > > +}; > > + > > +/* display timing controller */ > > +struct komeda_timing_ctrlr { > > + struct komeda_component base; > > +}; > > + > > +struct komeda_timing_ctrlr_state { > > + struct komeda_component_state base; > > +}; > > + > > +/** struct komeda_pipeline_funcs */ > > +struct komeda_pipeline_funcs { > > + /* dump_register: Optional, dump registers to seq_file */ > > + void (*dump_register)(struct komeda_pipeline *pipe, > > + struct seq_file *sf); > > +}; > > + > > +/** > > + * struct komeda_pipeline > > + * > > + * Represent a complete display pipeline and hold all functional components. > > + */ > > +struct komeda_pipeline { > > + /** @obj: link pipeline as private obj of drm_atomic_state */ > > + struct drm_private_obj obj; > > + /** @mdev: the parent komeda_dev */ > > + struct komeda_dev *mdev; > > + /** @pxlclk: pixel clock */ > > + struct clk *pxlclk; > > + /** @aclk: AXI clock */ > > + struct clk *aclk; > > + /** @id: pipeline id */ > > + int id; > > + /** @avail_comps: available components mask of pipeline */ > > + u32 avail_comps; > > + int n_layers; > > + struct komeda_layer *layers[KOMEDA_PIPELINE_MAX_LAYERS]; > > + int n_scalers; > > + struct komeda_scaler *scalers[KOMEDA_PIPELINE_MAX_SCALERS]; > > + struct komeda_compiz *compiz; > > + struct komeda_layer *wb_layer; > > + struct komeda_improc *improc; > > + struct komeda_timing_ctrlr *ctrlr; > > + struct komeda_pipeline_funcs *funcs; /* private pipeline functions */ > > +}; > > + > > +/** > > + * struct komeda_pipeline_state > > + * > > + * NOTE: > > + * Unlike the pipeline, pipeline_state doesn’t gather any component_state > > + * into it. It because all component will be managed by drm_atomic_state. > > + */ > > +struct komeda_pipeline_state { > > + /** @obj: tracking pipeline_state by drm_atomic_state */ > > + struct drm_private_state obj; > > + struct komeda_pipeline *pipe; > > + /** @crtc: currently bound crtc */ > > + struct drm_crtc *crtc; > > + /** > > + * @active_comps: > > + * > > + * bitmask - BIT(component->id) of active components > > + */ > > + u32 active_comps; > > +}; > > + > > +#define to_layer(c) container_of(c, struct komeda_layer, base) > > +#define to_compiz(c) container_of(c, struct komeda_compiz, base) > > +#define to_scaler(c) container_of(c, struct komeda_scaler, base) > > +#define to_improc(c) container_of(c, struct komeda_improc, base) > > +#define to_ctrlr(c) container_of(c, struct komeda_timing_ctrlr, base) > > + > > +#define to_layer_st(c) container_of(c, struct komeda_layer_state, base) > > +#define to_compiz_st(c) container_of(c, struct komeda_compiz_state, base) > > +#define to_scaler_st(c) container_of(c, struct komeda_scaler_state, base) > > +#define to_improc_st(c) container_of(c, struct komeda_improc_state, base) > > +#define to_ctrlr_st(c) container_of(c, struct komeda_timing_ctrlr_state, base) > > + > > +/* pipeline APIs */ > > +struct komeda_pipeline * > > +komeda_pipeline_add(struct komeda_dev *mdev, size_t size, > > + struct komeda_pipeline_funcs *funcs); > > +void komeda_pipeline_destroy(struct komeda_dev *mdev, > > + struct komeda_pipeline *pipe); > > + > > +struct komeda_component * > > +komeda_pipeline_get_component(struct komeda_pipeline *pipe, int id); > > + > > +/* component APIs */ > > +struct komeda_component * > > +komeda_component_add(struct komeda_pipeline *pipe, > > + size_t comp_sz, u32 id, u32 hw_id, > > + struct komeda_component_funcs *funcs, > > + u8 max_active_inputs, u32 supported_inputs, > > + u8 max_active_outputs, u32 __iomem *reg, > > + const char *name_fmt, ...); > > + > > +void komeda_component_destroy(struct komeda_dev *mdev, > > + struct komeda_component *c); > > + > > +#endif /* _KOMEDA_PIPELINE_H_*/ > > -- > > 2.17.1 > > > > With these small changes: > > Reviewed-by: Liviu Dudau <liviu.dudau@xxxxxxx> > > Best regards, > Liviu > > > -- > ==================== > | I would like to | > | fix the world, | > | but they're not | > | giving me the | > \ source code! / > --------------- > ¯\_(ツ)_/¯