18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0+ 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * rcar_cmm.c -- R-Car Display Unit Color Management Module 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright (C) 2019 Jacopo Mondi <jacopo+renesas@jmondi.org> 68c2ecf20Sopenharmony_ci */ 78c2ecf20Sopenharmony_ci 88c2ecf20Sopenharmony_ci#include <linux/io.h> 98c2ecf20Sopenharmony_ci#include <linux/module.h> 108c2ecf20Sopenharmony_ci#include <linux/of.h> 118c2ecf20Sopenharmony_ci#include <linux/platform_device.h> 128c2ecf20Sopenharmony_ci#include <linux/pm_runtime.h> 138c2ecf20Sopenharmony_ci 148c2ecf20Sopenharmony_ci#include <drm/drm_color_mgmt.h> 158c2ecf20Sopenharmony_ci 168c2ecf20Sopenharmony_ci#include "rcar_cmm.h" 178c2ecf20Sopenharmony_ci 188c2ecf20Sopenharmony_ci#define CM2_LUT_CTRL 0x0000 198c2ecf20Sopenharmony_ci#define CM2_LUT_CTRL_LUT_EN BIT(0) 208c2ecf20Sopenharmony_ci#define CM2_LUT_TBL_BASE 0x0600 218c2ecf20Sopenharmony_ci#define CM2_LUT_TBL(__i) (CM2_LUT_TBL_BASE + (__i) * 4) 228c2ecf20Sopenharmony_ci 238c2ecf20Sopenharmony_cistruct rcar_cmm { 248c2ecf20Sopenharmony_ci void __iomem *base; 258c2ecf20Sopenharmony_ci 268c2ecf20Sopenharmony_ci /* 278c2ecf20Sopenharmony_ci * @lut: 1D-LUT state 288c2ecf20Sopenharmony_ci * @lut.enabled: 1D-LUT enabled flag 298c2ecf20Sopenharmony_ci */ 308c2ecf20Sopenharmony_ci struct { 318c2ecf20Sopenharmony_ci bool enabled; 328c2ecf20Sopenharmony_ci } lut; 338c2ecf20Sopenharmony_ci}; 348c2ecf20Sopenharmony_ci 358c2ecf20Sopenharmony_cistatic inline int rcar_cmm_read(struct rcar_cmm *rcmm, u32 reg) 368c2ecf20Sopenharmony_ci{ 378c2ecf20Sopenharmony_ci return ioread32(rcmm->base + reg); 388c2ecf20Sopenharmony_ci} 398c2ecf20Sopenharmony_ci 408c2ecf20Sopenharmony_cistatic inline void rcar_cmm_write(struct rcar_cmm *rcmm, u32 reg, u32 data) 418c2ecf20Sopenharmony_ci{ 428c2ecf20Sopenharmony_ci iowrite32(data, rcmm->base + reg); 438c2ecf20Sopenharmony_ci} 448c2ecf20Sopenharmony_ci 458c2ecf20Sopenharmony_ci/* 468c2ecf20Sopenharmony_ci * rcar_cmm_lut_write() - Scale the DRM LUT table entries to hardware precision 478c2ecf20Sopenharmony_ci * and write to the CMM registers 488c2ecf20Sopenharmony_ci * @rcmm: Pointer to the CMM device 498c2ecf20Sopenharmony_ci * @drm_lut: Pointer to the DRM LUT table 508c2ecf20Sopenharmony_ci */ 518c2ecf20Sopenharmony_cistatic void rcar_cmm_lut_write(struct rcar_cmm *rcmm, 528c2ecf20Sopenharmony_ci const struct drm_color_lut *drm_lut) 538c2ecf20Sopenharmony_ci{ 548c2ecf20Sopenharmony_ci unsigned int i; 558c2ecf20Sopenharmony_ci 568c2ecf20Sopenharmony_ci for (i = 0; i < CM2_LUT_SIZE; ++i) { 578c2ecf20Sopenharmony_ci u32 entry = drm_color_lut_extract(drm_lut[i].red, 8) << 16 588c2ecf20Sopenharmony_ci | drm_color_lut_extract(drm_lut[i].green, 8) << 8 598c2ecf20Sopenharmony_ci | drm_color_lut_extract(drm_lut[i].blue, 8); 608c2ecf20Sopenharmony_ci 618c2ecf20Sopenharmony_ci rcar_cmm_write(rcmm, CM2_LUT_TBL(i), entry); 628c2ecf20Sopenharmony_ci } 638c2ecf20Sopenharmony_ci} 648c2ecf20Sopenharmony_ci 658c2ecf20Sopenharmony_ci/* 668c2ecf20Sopenharmony_ci * rcar_cmm_setup() - Configure the CMM unit 678c2ecf20Sopenharmony_ci * @pdev: The platform device associated with the CMM instance 688c2ecf20Sopenharmony_ci * @config: The CMM unit configuration 698c2ecf20Sopenharmony_ci * 708c2ecf20Sopenharmony_ci * Configure the CMM unit with the given configuration. Currently enabling, 718c2ecf20Sopenharmony_ci * disabling and programming of the 1-D LUT unit is supported. 728c2ecf20Sopenharmony_ci * 738c2ecf20Sopenharmony_ci * As rcar_cmm_setup() accesses the CMM registers the unit should be powered 748c2ecf20Sopenharmony_ci * and its functional clock enabled. To guarantee this, before any call to 758c2ecf20Sopenharmony_ci * this function is made, the CMM unit has to be enabled by calling 768c2ecf20Sopenharmony_ci * rcar_cmm_enable() first. 778c2ecf20Sopenharmony_ci * 788c2ecf20Sopenharmony_ci * TODO: Add support for LUT double buffer operations to avoid updating the 798c2ecf20Sopenharmony_ci * LUT table entries while a frame is being displayed. 808c2ecf20Sopenharmony_ci */ 818c2ecf20Sopenharmony_ciint rcar_cmm_setup(struct platform_device *pdev, 828c2ecf20Sopenharmony_ci const struct rcar_cmm_config *config) 838c2ecf20Sopenharmony_ci{ 848c2ecf20Sopenharmony_ci struct rcar_cmm *rcmm = platform_get_drvdata(pdev); 858c2ecf20Sopenharmony_ci 868c2ecf20Sopenharmony_ci /* Disable LUT if no table is provided. */ 878c2ecf20Sopenharmony_ci if (!config->lut.table) { 888c2ecf20Sopenharmony_ci if (rcmm->lut.enabled) { 898c2ecf20Sopenharmony_ci rcar_cmm_write(rcmm, CM2_LUT_CTRL, 0); 908c2ecf20Sopenharmony_ci rcmm->lut.enabled = false; 918c2ecf20Sopenharmony_ci } 928c2ecf20Sopenharmony_ci 938c2ecf20Sopenharmony_ci return 0; 948c2ecf20Sopenharmony_ci } 958c2ecf20Sopenharmony_ci 968c2ecf20Sopenharmony_ci /* Enable LUT and program the new gamma table values. */ 978c2ecf20Sopenharmony_ci if (!rcmm->lut.enabled) { 988c2ecf20Sopenharmony_ci rcar_cmm_write(rcmm, CM2_LUT_CTRL, CM2_LUT_CTRL_LUT_EN); 998c2ecf20Sopenharmony_ci rcmm->lut.enabled = true; 1008c2ecf20Sopenharmony_ci } 1018c2ecf20Sopenharmony_ci 1028c2ecf20Sopenharmony_ci rcar_cmm_lut_write(rcmm, config->lut.table); 1038c2ecf20Sopenharmony_ci 1048c2ecf20Sopenharmony_ci return 0; 1058c2ecf20Sopenharmony_ci} 1068c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(rcar_cmm_setup); 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_ci/* 1098c2ecf20Sopenharmony_ci * rcar_cmm_enable() - Enable the CMM unit 1108c2ecf20Sopenharmony_ci * @pdev: The platform device associated with the CMM instance 1118c2ecf20Sopenharmony_ci * 1128c2ecf20Sopenharmony_ci * When the output of the corresponding DU channel is routed to the CMM unit, 1138c2ecf20Sopenharmony_ci * the unit shall be enabled before the DU channel is started, and remain 1148c2ecf20Sopenharmony_ci * enabled until the channel is stopped. The CMM unit shall be disabled with 1158c2ecf20Sopenharmony_ci * rcar_cmm_disable(). 1168c2ecf20Sopenharmony_ci * 1178c2ecf20Sopenharmony_ci * Calls to rcar_cmm_enable() and rcar_cmm_disable() are not reference-counted. 1188c2ecf20Sopenharmony_ci * It is an error to attempt to enable an already enabled CMM unit, or to 1198c2ecf20Sopenharmony_ci * attempt to disable a disabled unit. 1208c2ecf20Sopenharmony_ci */ 1218c2ecf20Sopenharmony_ciint rcar_cmm_enable(struct platform_device *pdev) 1228c2ecf20Sopenharmony_ci{ 1238c2ecf20Sopenharmony_ci int ret; 1248c2ecf20Sopenharmony_ci 1258c2ecf20Sopenharmony_ci ret = pm_runtime_resume_and_get(&pdev->dev); 1268c2ecf20Sopenharmony_ci if (ret < 0) 1278c2ecf20Sopenharmony_ci return ret; 1288c2ecf20Sopenharmony_ci 1298c2ecf20Sopenharmony_ci return 0; 1308c2ecf20Sopenharmony_ci} 1318c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(rcar_cmm_enable); 1328c2ecf20Sopenharmony_ci 1338c2ecf20Sopenharmony_ci/* 1348c2ecf20Sopenharmony_ci * rcar_cmm_disable() - Disable the CMM unit 1358c2ecf20Sopenharmony_ci * @pdev: The platform device associated with the CMM instance 1368c2ecf20Sopenharmony_ci * 1378c2ecf20Sopenharmony_ci * See rcar_cmm_enable() for usage information. 1388c2ecf20Sopenharmony_ci * 1398c2ecf20Sopenharmony_ci * Disabling the CMM unit disable all the internal processing blocks. The CMM 1408c2ecf20Sopenharmony_ci * state shall thus be restored with rcar_cmm_setup() when re-enabling the CMM 1418c2ecf20Sopenharmony_ci * unit after the next rcar_cmm_enable() call. 1428c2ecf20Sopenharmony_ci */ 1438c2ecf20Sopenharmony_civoid rcar_cmm_disable(struct platform_device *pdev) 1448c2ecf20Sopenharmony_ci{ 1458c2ecf20Sopenharmony_ci struct rcar_cmm *rcmm = platform_get_drvdata(pdev); 1468c2ecf20Sopenharmony_ci 1478c2ecf20Sopenharmony_ci rcar_cmm_write(rcmm, CM2_LUT_CTRL, 0); 1488c2ecf20Sopenharmony_ci rcmm->lut.enabled = false; 1498c2ecf20Sopenharmony_ci 1508c2ecf20Sopenharmony_ci pm_runtime_put(&pdev->dev); 1518c2ecf20Sopenharmony_ci} 1528c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(rcar_cmm_disable); 1538c2ecf20Sopenharmony_ci 1548c2ecf20Sopenharmony_ci/* 1558c2ecf20Sopenharmony_ci * rcar_cmm_init() - Initialize the CMM unit 1568c2ecf20Sopenharmony_ci * @pdev: The platform device associated with the CMM instance 1578c2ecf20Sopenharmony_ci * 1588c2ecf20Sopenharmony_ci * Return: 0 on success, -EPROBE_DEFER if the CMM is not available yet, 1598c2ecf20Sopenharmony_ci * -ENODEV if the DRM_RCAR_CMM config option is disabled 1608c2ecf20Sopenharmony_ci */ 1618c2ecf20Sopenharmony_ciint rcar_cmm_init(struct platform_device *pdev) 1628c2ecf20Sopenharmony_ci{ 1638c2ecf20Sopenharmony_ci struct rcar_cmm *rcmm = platform_get_drvdata(pdev); 1648c2ecf20Sopenharmony_ci 1658c2ecf20Sopenharmony_ci if (!rcmm) 1668c2ecf20Sopenharmony_ci return -EPROBE_DEFER; 1678c2ecf20Sopenharmony_ci 1688c2ecf20Sopenharmony_ci return 0; 1698c2ecf20Sopenharmony_ci} 1708c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(rcar_cmm_init); 1718c2ecf20Sopenharmony_ci 1728c2ecf20Sopenharmony_cistatic int rcar_cmm_probe(struct platform_device *pdev) 1738c2ecf20Sopenharmony_ci{ 1748c2ecf20Sopenharmony_ci struct rcar_cmm *rcmm; 1758c2ecf20Sopenharmony_ci 1768c2ecf20Sopenharmony_ci rcmm = devm_kzalloc(&pdev->dev, sizeof(*rcmm), GFP_KERNEL); 1778c2ecf20Sopenharmony_ci if (!rcmm) 1788c2ecf20Sopenharmony_ci return -ENOMEM; 1798c2ecf20Sopenharmony_ci platform_set_drvdata(pdev, rcmm); 1808c2ecf20Sopenharmony_ci 1818c2ecf20Sopenharmony_ci rcmm->base = devm_platform_ioremap_resource(pdev, 0); 1828c2ecf20Sopenharmony_ci if (IS_ERR(rcmm->base)) 1838c2ecf20Sopenharmony_ci return PTR_ERR(rcmm->base); 1848c2ecf20Sopenharmony_ci 1858c2ecf20Sopenharmony_ci pm_runtime_enable(&pdev->dev); 1868c2ecf20Sopenharmony_ci 1878c2ecf20Sopenharmony_ci return 0; 1888c2ecf20Sopenharmony_ci} 1898c2ecf20Sopenharmony_ci 1908c2ecf20Sopenharmony_cistatic int rcar_cmm_remove(struct platform_device *pdev) 1918c2ecf20Sopenharmony_ci{ 1928c2ecf20Sopenharmony_ci pm_runtime_disable(&pdev->dev); 1938c2ecf20Sopenharmony_ci 1948c2ecf20Sopenharmony_ci return 0; 1958c2ecf20Sopenharmony_ci} 1968c2ecf20Sopenharmony_ci 1978c2ecf20Sopenharmony_cistatic const struct of_device_id rcar_cmm_of_table[] = { 1988c2ecf20Sopenharmony_ci { .compatible = "renesas,rcar-gen3-cmm", }, 1998c2ecf20Sopenharmony_ci { .compatible = "renesas,rcar-gen2-cmm", }, 2008c2ecf20Sopenharmony_ci { }, 2018c2ecf20Sopenharmony_ci}; 2028c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(of, rcar_cmm_of_table); 2038c2ecf20Sopenharmony_ci 2048c2ecf20Sopenharmony_cistatic struct platform_driver rcar_cmm_platform_driver = { 2058c2ecf20Sopenharmony_ci .probe = rcar_cmm_probe, 2068c2ecf20Sopenharmony_ci .remove = rcar_cmm_remove, 2078c2ecf20Sopenharmony_ci .driver = { 2088c2ecf20Sopenharmony_ci .name = "rcar-cmm", 2098c2ecf20Sopenharmony_ci .of_match_table = rcar_cmm_of_table, 2108c2ecf20Sopenharmony_ci }, 2118c2ecf20Sopenharmony_ci}; 2128c2ecf20Sopenharmony_ci 2138c2ecf20Sopenharmony_cimodule_platform_driver(rcar_cmm_platform_driver); 2148c2ecf20Sopenharmony_ci 2158c2ecf20Sopenharmony_ciMODULE_AUTHOR("Jacopo Mondi <jacopo+renesas@jmondi.org>"); 2168c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Renesas R-Car CMM Driver"); 2178c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2"); 218