162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright (C) 2009 Nokia Corporation 462306a36Sopenharmony_ci * Author: Tomi Valkeinen <tomi.valkeinen@nokia.com> 562306a36Sopenharmony_ci * 662306a36Sopenharmony_ci * Some code and ideas taken from drivers/video/omap/ driver 762306a36Sopenharmony_ci * by Imre Deak. 862306a36Sopenharmony_ci */ 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ci#define DSS_SUBSYS_NAME "DISPLAY" 1162306a36Sopenharmony_ci 1262306a36Sopenharmony_ci#include <linux/kernel.h> 1362306a36Sopenharmony_ci#include <linux/kstrtox.h> 1462306a36Sopenharmony_ci#include <linux/module.h> 1562306a36Sopenharmony_ci#include <linux/platform_device.h> 1662306a36Sopenharmony_ci#include <linux/sysfs.h> 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_ci#include <video/omapfb_dss.h> 1962306a36Sopenharmony_ci#include "dss.h" 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_cistatic ssize_t display_name_show(struct omap_dss_device *dssdev, char *buf) 2262306a36Sopenharmony_ci{ 2362306a36Sopenharmony_ci return sysfs_emit(buf, "%s\n", 2462306a36Sopenharmony_ci dssdev->name ? 2562306a36Sopenharmony_ci dssdev->name : ""); 2662306a36Sopenharmony_ci} 2762306a36Sopenharmony_ci 2862306a36Sopenharmony_cistatic ssize_t display_enabled_show(struct omap_dss_device *dssdev, char *buf) 2962306a36Sopenharmony_ci{ 3062306a36Sopenharmony_ci return sysfs_emit(buf, "%d\n", 3162306a36Sopenharmony_ci omapdss_device_is_enabled(dssdev)); 3262306a36Sopenharmony_ci} 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_cistatic ssize_t display_enabled_store(struct omap_dss_device *dssdev, 3562306a36Sopenharmony_ci const char *buf, size_t size) 3662306a36Sopenharmony_ci{ 3762306a36Sopenharmony_ci int r; 3862306a36Sopenharmony_ci bool enable; 3962306a36Sopenharmony_ci 4062306a36Sopenharmony_ci r = kstrtobool(buf, &enable); 4162306a36Sopenharmony_ci if (r) 4262306a36Sopenharmony_ci return r; 4362306a36Sopenharmony_ci 4462306a36Sopenharmony_ci if (enable == omapdss_device_is_enabled(dssdev)) 4562306a36Sopenharmony_ci return size; 4662306a36Sopenharmony_ci 4762306a36Sopenharmony_ci if (omapdss_device_is_connected(dssdev) == false) 4862306a36Sopenharmony_ci return -ENODEV; 4962306a36Sopenharmony_ci 5062306a36Sopenharmony_ci if (enable) { 5162306a36Sopenharmony_ci r = dssdev->driver->enable(dssdev); 5262306a36Sopenharmony_ci if (r) 5362306a36Sopenharmony_ci return r; 5462306a36Sopenharmony_ci } else { 5562306a36Sopenharmony_ci dssdev->driver->disable(dssdev); 5662306a36Sopenharmony_ci } 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_ci return size; 5962306a36Sopenharmony_ci} 6062306a36Sopenharmony_ci 6162306a36Sopenharmony_cistatic ssize_t display_tear_show(struct omap_dss_device *dssdev, char *buf) 6262306a36Sopenharmony_ci{ 6362306a36Sopenharmony_ci return sysfs_emit(buf, "%d\n", 6462306a36Sopenharmony_ci dssdev->driver->get_te ? 6562306a36Sopenharmony_ci dssdev->driver->get_te(dssdev) : 0); 6662306a36Sopenharmony_ci} 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_cistatic ssize_t display_tear_store(struct omap_dss_device *dssdev, 6962306a36Sopenharmony_ci const char *buf, size_t size) 7062306a36Sopenharmony_ci{ 7162306a36Sopenharmony_ci int r; 7262306a36Sopenharmony_ci bool te; 7362306a36Sopenharmony_ci 7462306a36Sopenharmony_ci if (!dssdev->driver->enable_te || !dssdev->driver->get_te) 7562306a36Sopenharmony_ci return -ENOENT; 7662306a36Sopenharmony_ci 7762306a36Sopenharmony_ci r = kstrtobool(buf, &te); 7862306a36Sopenharmony_ci if (r) 7962306a36Sopenharmony_ci return r; 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ci r = dssdev->driver->enable_te(dssdev, te); 8262306a36Sopenharmony_ci if (r) 8362306a36Sopenharmony_ci return r; 8462306a36Sopenharmony_ci 8562306a36Sopenharmony_ci return size; 8662306a36Sopenharmony_ci} 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_cistatic ssize_t display_timings_show(struct omap_dss_device *dssdev, char *buf) 8962306a36Sopenharmony_ci{ 9062306a36Sopenharmony_ci struct omap_video_timings t; 9162306a36Sopenharmony_ci 9262306a36Sopenharmony_ci if (!dssdev->driver->get_timings) 9362306a36Sopenharmony_ci return -ENOENT; 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ci dssdev->driver->get_timings(dssdev, &t); 9662306a36Sopenharmony_ci 9762306a36Sopenharmony_ci return sysfs_emit(buf, "%u,%u/%u/%u/%u,%u/%u/%u/%u\n", 9862306a36Sopenharmony_ci t.pixelclock, 9962306a36Sopenharmony_ci t.x_res, t.hfp, t.hbp, t.hsw, 10062306a36Sopenharmony_ci t.y_res, t.vfp, t.vbp, t.vsw); 10162306a36Sopenharmony_ci} 10262306a36Sopenharmony_ci 10362306a36Sopenharmony_cistatic ssize_t display_timings_store(struct omap_dss_device *dssdev, 10462306a36Sopenharmony_ci const char *buf, size_t size) 10562306a36Sopenharmony_ci{ 10662306a36Sopenharmony_ci struct omap_video_timings t = dssdev->panel.timings; 10762306a36Sopenharmony_ci int r, found; 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_ci if (!dssdev->driver->set_timings || !dssdev->driver->check_timings) 11062306a36Sopenharmony_ci return -ENOENT; 11162306a36Sopenharmony_ci 11262306a36Sopenharmony_ci found = 0; 11362306a36Sopenharmony_ci#ifdef CONFIG_FB_OMAP2_DSS_VENC 11462306a36Sopenharmony_ci if (strncmp("pal", buf, 3) == 0) { 11562306a36Sopenharmony_ci t = omap_dss_pal_timings; 11662306a36Sopenharmony_ci found = 1; 11762306a36Sopenharmony_ci } else if (strncmp("ntsc", buf, 4) == 0) { 11862306a36Sopenharmony_ci t = omap_dss_ntsc_timings; 11962306a36Sopenharmony_ci found = 1; 12062306a36Sopenharmony_ci } 12162306a36Sopenharmony_ci#endif 12262306a36Sopenharmony_ci if (!found && sscanf(buf, "%u,%hu/%hu/%hu/%hu,%hu/%hu/%hu/%hu", 12362306a36Sopenharmony_ci &t.pixelclock, 12462306a36Sopenharmony_ci &t.x_res, &t.hfp, &t.hbp, &t.hsw, 12562306a36Sopenharmony_ci &t.y_res, &t.vfp, &t.vbp, &t.vsw) != 9) 12662306a36Sopenharmony_ci return -EINVAL; 12762306a36Sopenharmony_ci 12862306a36Sopenharmony_ci r = dssdev->driver->check_timings(dssdev, &t); 12962306a36Sopenharmony_ci if (r) 13062306a36Sopenharmony_ci return r; 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci dssdev->driver->disable(dssdev); 13362306a36Sopenharmony_ci dssdev->driver->set_timings(dssdev, &t); 13462306a36Sopenharmony_ci r = dssdev->driver->enable(dssdev); 13562306a36Sopenharmony_ci if (r) 13662306a36Sopenharmony_ci return r; 13762306a36Sopenharmony_ci 13862306a36Sopenharmony_ci return size; 13962306a36Sopenharmony_ci} 14062306a36Sopenharmony_ci 14162306a36Sopenharmony_cistatic ssize_t display_rotate_show(struct omap_dss_device *dssdev, char *buf) 14262306a36Sopenharmony_ci{ 14362306a36Sopenharmony_ci int rotate; 14462306a36Sopenharmony_ci if (!dssdev->driver->get_rotate) 14562306a36Sopenharmony_ci return -ENOENT; 14662306a36Sopenharmony_ci rotate = dssdev->driver->get_rotate(dssdev); 14762306a36Sopenharmony_ci return sysfs_emit(buf, "%u\n", rotate); 14862306a36Sopenharmony_ci} 14962306a36Sopenharmony_ci 15062306a36Sopenharmony_cistatic ssize_t display_rotate_store(struct omap_dss_device *dssdev, 15162306a36Sopenharmony_ci const char *buf, size_t size) 15262306a36Sopenharmony_ci{ 15362306a36Sopenharmony_ci int rot, r; 15462306a36Sopenharmony_ci 15562306a36Sopenharmony_ci if (!dssdev->driver->set_rotate || !dssdev->driver->get_rotate) 15662306a36Sopenharmony_ci return -ENOENT; 15762306a36Sopenharmony_ci 15862306a36Sopenharmony_ci r = kstrtoint(buf, 0, &rot); 15962306a36Sopenharmony_ci if (r) 16062306a36Sopenharmony_ci return r; 16162306a36Sopenharmony_ci 16262306a36Sopenharmony_ci r = dssdev->driver->set_rotate(dssdev, rot); 16362306a36Sopenharmony_ci if (r) 16462306a36Sopenharmony_ci return r; 16562306a36Sopenharmony_ci 16662306a36Sopenharmony_ci return size; 16762306a36Sopenharmony_ci} 16862306a36Sopenharmony_ci 16962306a36Sopenharmony_cistatic ssize_t display_mirror_show(struct omap_dss_device *dssdev, char *buf) 17062306a36Sopenharmony_ci{ 17162306a36Sopenharmony_ci int mirror; 17262306a36Sopenharmony_ci if (!dssdev->driver->get_mirror) 17362306a36Sopenharmony_ci return -ENOENT; 17462306a36Sopenharmony_ci mirror = dssdev->driver->get_mirror(dssdev); 17562306a36Sopenharmony_ci return sysfs_emit(buf, "%u\n", mirror); 17662306a36Sopenharmony_ci} 17762306a36Sopenharmony_ci 17862306a36Sopenharmony_cistatic ssize_t display_mirror_store(struct omap_dss_device *dssdev, 17962306a36Sopenharmony_ci const char *buf, size_t size) 18062306a36Sopenharmony_ci{ 18162306a36Sopenharmony_ci int r; 18262306a36Sopenharmony_ci bool mirror; 18362306a36Sopenharmony_ci 18462306a36Sopenharmony_ci if (!dssdev->driver->set_mirror || !dssdev->driver->get_mirror) 18562306a36Sopenharmony_ci return -ENOENT; 18662306a36Sopenharmony_ci 18762306a36Sopenharmony_ci r = kstrtobool(buf, &mirror); 18862306a36Sopenharmony_ci if (r) 18962306a36Sopenharmony_ci return r; 19062306a36Sopenharmony_ci 19162306a36Sopenharmony_ci r = dssdev->driver->set_mirror(dssdev, mirror); 19262306a36Sopenharmony_ci if (r) 19362306a36Sopenharmony_ci return r; 19462306a36Sopenharmony_ci 19562306a36Sopenharmony_ci return size; 19662306a36Sopenharmony_ci} 19762306a36Sopenharmony_ci 19862306a36Sopenharmony_cistatic ssize_t display_wss_show(struct omap_dss_device *dssdev, char *buf) 19962306a36Sopenharmony_ci{ 20062306a36Sopenharmony_ci unsigned int wss; 20162306a36Sopenharmony_ci 20262306a36Sopenharmony_ci if (!dssdev->driver->get_wss) 20362306a36Sopenharmony_ci return -ENOENT; 20462306a36Sopenharmony_ci 20562306a36Sopenharmony_ci wss = dssdev->driver->get_wss(dssdev); 20662306a36Sopenharmony_ci 20762306a36Sopenharmony_ci return sysfs_emit(buf, "0x%05x\n", wss); 20862306a36Sopenharmony_ci} 20962306a36Sopenharmony_ci 21062306a36Sopenharmony_cistatic ssize_t display_wss_store(struct omap_dss_device *dssdev, 21162306a36Sopenharmony_ci const char *buf, size_t size) 21262306a36Sopenharmony_ci{ 21362306a36Sopenharmony_ci u32 wss; 21462306a36Sopenharmony_ci int r; 21562306a36Sopenharmony_ci 21662306a36Sopenharmony_ci if (!dssdev->driver->get_wss || !dssdev->driver->set_wss) 21762306a36Sopenharmony_ci return -ENOENT; 21862306a36Sopenharmony_ci 21962306a36Sopenharmony_ci r = kstrtou32(buf, 0, &wss); 22062306a36Sopenharmony_ci if (r) 22162306a36Sopenharmony_ci return r; 22262306a36Sopenharmony_ci 22362306a36Sopenharmony_ci if (wss > 0xfffff) 22462306a36Sopenharmony_ci return -EINVAL; 22562306a36Sopenharmony_ci 22662306a36Sopenharmony_ci r = dssdev->driver->set_wss(dssdev, wss); 22762306a36Sopenharmony_ci if (r) 22862306a36Sopenharmony_ci return r; 22962306a36Sopenharmony_ci 23062306a36Sopenharmony_ci return size; 23162306a36Sopenharmony_ci} 23262306a36Sopenharmony_ci 23362306a36Sopenharmony_cistruct display_attribute { 23462306a36Sopenharmony_ci struct attribute attr; 23562306a36Sopenharmony_ci ssize_t (*show)(struct omap_dss_device *, char *); 23662306a36Sopenharmony_ci ssize_t (*store)(struct omap_dss_device *, const char *, size_t); 23762306a36Sopenharmony_ci}; 23862306a36Sopenharmony_ci 23962306a36Sopenharmony_ci#define DISPLAY_ATTR(_name, _mode, _show, _store) \ 24062306a36Sopenharmony_ci struct display_attribute display_attr_##_name = \ 24162306a36Sopenharmony_ci __ATTR(_name, _mode, _show, _store) 24262306a36Sopenharmony_ci 24362306a36Sopenharmony_cistatic DISPLAY_ATTR(name, S_IRUGO, display_name_show, NULL); 24462306a36Sopenharmony_cistatic DISPLAY_ATTR(display_name, S_IRUGO, display_name_show, NULL); 24562306a36Sopenharmony_cistatic DISPLAY_ATTR(enabled, S_IRUGO|S_IWUSR, 24662306a36Sopenharmony_ci display_enabled_show, display_enabled_store); 24762306a36Sopenharmony_cistatic DISPLAY_ATTR(tear_elim, S_IRUGO|S_IWUSR, 24862306a36Sopenharmony_ci display_tear_show, display_tear_store); 24962306a36Sopenharmony_cistatic DISPLAY_ATTR(timings, S_IRUGO|S_IWUSR, 25062306a36Sopenharmony_ci display_timings_show, display_timings_store); 25162306a36Sopenharmony_cistatic DISPLAY_ATTR(rotate, S_IRUGO|S_IWUSR, 25262306a36Sopenharmony_ci display_rotate_show, display_rotate_store); 25362306a36Sopenharmony_cistatic DISPLAY_ATTR(mirror, S_IRUGO|S_IWUSR, 25462306a36Sopenharmony_ci display_mirror_show, display_mirror_store); 25562306a36Sopenharmony_cistatic DISPLAY_ATTR(wss, S_IRUGO|S_IWUSR, 25662306a36Sopenharmony_ci display_wss_show, display_wss_store); 25762306a36Sopenharmony_ci 25862306a36Sopenharmony_cistatic struct attribute *display_sysfs_attrs[] = { 25962306a36Sopenharmony_ci &display_attr_name.attr, 26062306a36Sopenharmony_ci &display_attr_display_name.attr, 26162306a36Sopenharmony_ci &display_attr_enabled.attr, 26262306a36Sopenharmony_ci &display_attr_tear_elim.attr, 26362306a36Sopenharmony_ci &display_attr_timings.attr, 26462306a36Sopenharmony_ci &display_attr_rotate.attr, 26562306a36Sopenharmony_ci &display_attr_mirror.attr, 26662306a36Sopenharmony_ci &display_attr_wss.attr, 26762306a36Sopenharmony_ci NULL 26862306a36Sopenharmony_ci}; 26962306a36Sopenharmony_ciATTRIBUTE_GROUPS(display_sysfs); 27062306a36Sopenharmony_ci 27162306a36Sopenharmony_cistatic ssize_t display_attr_show(struct kobject *kobj, struct attribute *attr, 27262306a36Sopenharmony_ci char *buf) 27362306a36Sopenharmony_ci{ 27462306a36Sopenharmony_ci struct omap_dss_device *dssdev; 27562306a36Sopenharmony_ci struct display_attribute *display_attr; 27662306a36Sopenharmony_ci 27762306a36Sopenharmony_ci dssdev = container_of(kobj, struct omap_dss_device, kobj); 27862306a36Sopenharmony_ci display_attr = container_of(attr, struct display_attribute, attr); 27962306a36Sopenharmony_ci 28062306a36Sopenharmony_ci if (!display_attr->show) 28162306a36Sopenharmony_ci return -ENOENT; 28262306a36Sopenharmony_ci 28362306a36Sopenharmony_ci return display_attr->show(dssdev, buf); 28462306a36Sopenharmony_ci} 28562306a36Sopenharmony_ci 28662306a36Sopenharmony_cistatic ssize_t display_attr_store(struct kobject *kobj, struct attribute *attr, 28762306a36Sopenharmony_ci const char *buf, size_t size) 28862306a36Sopenharmony_ci{ 28962306a36Sopenharmony_ci struct omap_dss_device *dssdev; 29062306a36Sopenharmony_ci struct display_attribute *display_attr; 29162306a36Sopenharmony_ci 29262306a36Sopenharmony_ci dssdev = container_of(kobj, struct omap_dss_device, kobj); 29362306a36Sopenharmony_ci display_attr = container_of(attr, struct display_attribute, attr); 29462306a36Sopenharmony_ci 29562306a36Sopenharmony_ci if (!display_attr->store) 29662306a36Sopenharmony_ci return -ENOENT; 29762306a36Sopenharmony_ci 29862306a36Sopenharmony_ci return display_attr->store(dssdev, buf, size); 29962306a36Sopenharmony_ci} 30062306a36Sopenharmony_ci 30162306a36Sopenharmony_cistatic const struct sysfs_ops display_sysfs_ops = { 30262306a36Sopenharmony_ci .show = display_attr_show, 30362306a36Sopenharmony_ci .store = display_attr_store, 30462306a36Sopenharmony_ci}; 30562306a36Sopenharmony_ci 30662306a36Sopenharmony_cistatic struct kobj_type display_ktype = { 30762306a36Sopenharmony_ci .sysfs_ops = &display_sysfs_ops, 30862306a36Sopenharmony_ci .default_groups = display_sysfs_groups, 30962306a36Sopenharmony_ci}; 31062306a36Sopenharmony_ci 31162306a36Sopenharmony_ciint display_init_sysfs(struct platform_device *pdev) 31262306a36Sopenharmony_ci{ 31362306a36Sopenharmony_ci struct omap_dss_device *dssdev = NULL; 31462306a36Sopenharmony_ci int r; 31562306a36Sopenharmony_ci 31662306a36Sopenharmony_ci for_each_dss_dev(dssdev) { 31762306a36Sopenharmony_ci r = kobject_init_and_add(&dssdev->kobj, &display_ktype, 31862306a36Sopenharmony_ci &pdev->dev.kobj, "%s", dssdev->alias); 31962306a36Sopenharmony_ci if (r) { 32062306a36Sopenharmony_ci DSSERR("failed to create sysfs files\n"); 32162306a36Sopenharmony_ci omap_dss_put_device(dssdev); 32262306a36Sopenharmony_ci goto err; 32362306a36Sopenharmony_ci } 32462306a36Sopenharmony_ci } 32562306a36Sopenharmony_ci 32662306a36Sopenharmony_ci return 0; 32762306a36Sopenharmony_ci 32862306a36Sopenharmony_cierr: 32962306a36Sopenharmony_ci display_uninit_sysfs(pdev); 33062306a36Sopenharmony_ci 33162306a36Sopenharmony_ci return r; 33262306a36Sopenharmony_ci} 33362306a36Sopenharmony_ci 33462306a36Sopenharmony_civoid display_uninit_sysfs(struct platform_device *pdev) 33562306a36Sopenharmony_ci{ 33662306a36Sopenharmony_ci struct omap_dss_device *dssdev = NULL; 33762306a36Sopenharmony_ci 33862306a36Sopenharmony_ci for_each_dss_dev(dssdev) { 33962306a36Sopenharmony_ci if (kobject_name(&dssdev->kobj) == NULL) 34062306a36Sopenharmony_ci continue; 34162306a36Sopenharmony_ci 34262306a36Sopenharmony_ci kobject_del(&dssdev->kobj); 34362306a36Sopenharmony_ci kobject_put(&dssdev->kobj); 34462306a36Sopenharmony_ci 34562306a36Sopenharmony_ci memset(&dssdev->kobj, 0, sizeof(dssdev->kobj)); 34662306a36Sopenharmony_ci } 34762306a36Sopenharmony_ci} 348