162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright (C) 2012 Russell King 462306a36Sopenharmony_ci * Rewritten from the dovefb driver, and Armada510 manuals. 562306a36Sopenharmony_ci */ 662306a36Sopenharmony_ci 762306a36Sopenharmony_ci#include <linux/ctype.h> 862306a36Sopenharmony_ci#include <linux/module.h> 962306a36Sopenharmony_ci#include <linux/seq_file.h> 1062306a36Sopenharmony_ci#include <linux/uaccess.h> 1162306a36Sopenharmony_ci 1262306a36Sopenharmony_ci#include <drm/drm_debugfs.h> 1362306a36Sopenharmony_ci#include <drm/drm_file.h> 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_ci#include "armada_crtc.h" 1662306a36Sopenharmony_ci#include "armada_drm.h" 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_cistatic int armada_debugfs_gem_linear_show(struct seq_file *m, void *data) 1962306a36Sopenharmony_ci{ 2062306a36Sopenharmony_ci struct drm_info_node *node = m->private; 2162306a36Sopenharmony_ci struct drm_device *dev = node->minor->dev; 2262306a36Sopenharmony_ci struct armada_private *priv = drm_to_armada_dev(dev); 2362306a36Sopenharmony_ci struct drm_printer p = drm_seq_file_printer(m); 2462306a36Sopenharmony_ci 2562306a36Sopenharmony_ci mutex_lock(&priv->linear_lock); 2662306a36Sopenharmony_ci drm_mm_print(&priv->linear, &p); 2762306a36Sopenharmony_ci mutex_unlock(&priv->linear_lock); 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_ci return 0; 3062306a36Sopenharmony_ci} 3162306a36Sopenharmony_ci 3262306a36Sopenharmony_cistatic int armada_debugfs_crtc_reg_show(struct seq_file *m, void *data) 3362306a36Sopenharmony_ci{ 3462306a36Sopenharmony_ci struct armada_crtc *dcrtc = m->private; 3562306a36Sopenharmony_ci int i; 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_ci for (i = 0x84; i <= 0x1c4; i += 4) { 3862306a36Sopenharmony_ci u32 v = readl_relaxed(dcrtc->base + i); 3962306a36Sopenharmony_ci seq_printf(m, "0x%04x: 0x%08x\n", i, v); 4062306a36Sopenharmony_ci } 4162306a36Sopenharmony_ci 4262306a36Sopenharmony_ci return 0; 4362306a36Sopenharmony_ci} 4462306a36Sopenharmony_ci 4562306a36Sopenharmony_cistatic int armada_debugfs_crtc_reg_open(struct inode *inode, struct file *file) 4662306a36Sopenharmony_ci{ 4762306a36Sopenharmony_ci return single_open(file, armada_debugfs_crtc_reg_show, 4862306a36Sopenharmony_ci inode->i_private); 4962306a36Sopenharmony_ci} 5062306a36Sopenharmony_ci 5162306a36Sopenharmony_cistatic int armada_debugfs_crtc_reg_write(struct file *file, 5262306a36Sopenharmony_ci const char __user *ptr, size_t len, loff_t *off) 5362306a36Sopenharmony_ci{ 5462306a36Sopenharmony_ci struct armada_crtc *dcrtc; 5562306a36Sopenharmony_ci unsigned long reg, mask, val; 5662306a36Sopenharmony_ci char buf[32]; 5762306a36Sopenharmony_ci int ret; 5862306a36Sopenharmony_ci u32 v; 5962306a36Sopenharmony_ci 6062306a36Sopenharmony_ci if (*off != 0) 6162306a36Sopenharmony_ci return 0; 6262306a36Sopenharmony_ci 6362306a36Sopenharmony_ci if (len > sizeof(buf) - 1) 6462306a36Sopenharmony_ci len = sizeof(buf) - 1; 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_ci ret = strncpy_from_user(buf, ptr, len); 6762306a36Sopenharmony_ci if (ret < 0) 6862306a36Sopenharmony_ci return ret; 6962306a36Sopenharmony_ci buf[len] = '\0'; 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_ci if (sscanf(buf, "%lx %lx %lx", ®, &mask, &val) != 3) 7262306a36Sopenharmony_ci return -EINVAL; 7362306a36Sopenharmony_ci if (reg < 0x84 || reg > 0x1c4 || reg & 3) 7462306a36Sopenharmony_ci return -ERANGE; 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_ci dcrtc = ((struct seq_file *)file->private_data)->private; 7762306a36Sopenharmony_ci v = readl(dcrtc->base + reg); 7862306a36Sopenharmony_ci v &= ~mask; 7962306a36Sopenharmony_ci v |= val & mask; 8062306a36Sopenharmony_ci writel(v, dcrtc->base + reg); 8162306a36Sopenharmony_ci 8262306a36Sopenharmony_ci return len; 8362306a36Sopenharmony_ci} 8462306a36Sopenharmony_ci 8562306a36Sopenharmony_cistatic const struct file_operations armada_debugfs_crtc_reg_fops = { 8662306a36Sopenharmony_ci .owner = THIS_MODULE, 8762306a36Sopenharmony_ci .open = armada_debugfs_crtc_reg_open, 8862306a36Sopenharmony_ci .read = seq_read, 8962306a36Sopenharmony_ci .write = armada_debugfs_crtc_reg_write, 9062306a36Sopenharmony_ci .llseek = seq_lseek, 9162306a36Sopenharmony_ci .release = single_release, 9262306a36Sopenharmony_ci}; 9362306a36Sopenharmony_ci 9462306a36Sopenharmony_civoid armada_drm_crtc_debugfs_init(struct armada_crtc *dcrtc) 9562306a36Sopenharmony_ci{ 9662306a36Sopenharmony_ci debugfs_create_file("armada-regs", 0600, dcrtc->crtc.debugfs_entry, 9762306a36Sopenharmony_ci dcrtc, &armada_debugfs_crtc_reg_fops); 9862306a36Sopenharmony_ci} 9962306a36Sopenharmony_ci 10062306a36Sopenharmony_cistatic struct drm_info_list armada_debugfs_list[] = { 10162306a36Sopenharmony_ci { "gem_linear", armada_debugfs_gem_linear_show, 0 }, 10262306a36Sopenharmony_ci}; 10362306a36Sopenharmony_ci#define ARMADA_DEBUGFS_ENTRIES ARRAY_SIZE(armada_debugfs_list) 10462306a36Sopenharmony_ci 10562306a36Sopenharmony_ciint armada_drm_debugfs_init(struct drm_minor *minor) 10662306a36Sopenharmony_ci{ 10762306a36Sopenharmony_ci drm_debugfs_create_files(armada_debugfs_list, ARMADA_DEBUGFS_ENTRIES, 10862306a36Sopenharmony_ci minor->debugfs_root, minor); 10962306a36Sopenharmony_ci 11062306a36Sopenharmony_ci return 0; 11162306a36Sopenharmony_ci} 112