cregit-Linux how code gets into the kernel

Release 4.7 drivers/s390/cio/qdio_debug.c

Directory: drivers/s390/cio
/*
 *  Copyright IBM Corp. 2008, 2009
 *
 *  Author: Jan Glauber (jang@linux.vnet.ibm.com)
 */
#include <linux/seq_file.h>
#include <linux/debugfs.h>
#include <linux/uaccess.h>
#include <linux/export.h>
#include <linux/slab.h>
#include <asm/debug.h>
#include "qdio_debug.h"
#include "qdio.h"


debug_info_t *qdio_dbf_setup;

debug_info_t *qdio_dbf_error;


static struct dentry *debugfs_root;

#define QDIO_DEBUGFS_NAME_LEN	10

#define QDIO_DBF_NAME_LEN	20


struct qdio_dbf_entry {
	
char dbf_name[QDIO_DBF_NAME_LEN];
	
debug_info_t *dbf_info;
	
struct list_head dbf_list;
};

static LIST_HEAD(qdio_dbf_list);
static DEFINE_MUTEX(qdio_dbf_list_mutex);


static debug_info_t *qdio_get_dbf_entry(char *name) { struct qdio_dbf_entry *entry; debug_info_t *rc = NULL; mutex_lock(&qdio_dbf_list_mutex); list_for_each_entry(entry, &qdio_dbf_list, dbf_list) { if (strcmp(entry->dbf_name, name) == 0) { rc = entry->dbf_info; break; } } mutex_unlock(&qdio_dbf_list_mutex); return rc; }

Contributors

PersonTokensPropCommitsCommitProp
stefan rasplstefan raspl69100.00%1100.00%
Total69100.00%1100.00%


static void qdio_clear_dbf_list(void) { struct qdio_dbf_entry *entry, *tmp; mutex_lock(&qdio_dbf_list_mutex); list_for_each_entry_safe(entry, tmp, &qdio_dbf_list, dbf_list) { list_del(&entry->dbf_list); debug_unregister(entry->dbf_info); kfree(entry); } mutex_unlock(&qdio_dbf_list_mutex); }

Contributors

PersonTokensPropCommitsCommitProp
stefan rasplstefan raspl5998.33%150.00%
jan glauberjan glauber11.67%150.00%
Total60100.00%2100.00%


int qdio_allocate_dbf(struct qdio_initialize *init_data, struct qdio_irq *irq_ptr) { char text[QDIO_DBF_NAME_LEN]; struct qdio_dbf_entry *new_entry; DBF_EVENT("qfmt:%1d", init_data->q_format); DBF_HEX(init_data->adapter_name, 8); DBF_EVENT("qpff%4x", init_data->qib_param_field_format); DBF_HEX(&init_data->qib_param_field, sizeof(void *)); DBF_HEX(&init_data->input_slib_elements, sizeof(void *)); DBF_HEX(&init_data->output_slib_elements, sizeof(void *)); DBF_EVENT("niq:%1d noq:%1d", init_data->no_input_qs, init_data->no_output_qs); DBF_HEX(&init_data->input_handler, sizeof(void *)); DBF_HEX(&init_data->output_handler, sizeof(void *)); DBF_HEX(&init_data->int_parm, sizeof(long)); DBF_HEX(&init_data->input_sbal_addr_array, sizeof(void *)); DBF_HEX(&init_data->output_sbal_addr_array, sizeof(void *)); DBF_EVENT("irq:%8lx", (unsigned long)irq_ptr); /* allocate trace view for the interface */ snprintf(text, QDIO_DBF_NAME_LEN, "qdio_%s", dev_name(&init_data->cdev->dev)); irq_ptr->debug_area = qdio_get_dbf_entry(text); if (irq_ptr->debug_area) DBF_DEV_EVENT(DBF_ERR, irq_ptr, "dbf reused"); else { irq_ptr->debug_area = debug_register(text, 2, 1, 16); if (!irq_ptr->debug_area) return -ENOMEM; if (debug_register_view(irq_ptr->debug_area, &debug_hex_ascii_view)) { debug_unregister(irq_ptr->debug_area); return -ENOMEM; } debug_set_level(irq_ptr->debug_area, DBF_WARN); DBF_DEV_EVENT(DBF_ERR, irq_ptr, "dbf created"); new_entry = kzalloc(sizeof(struct qdio_dbf_entry), GFP_KERNEL); if (!new_entry) { debug_unregister(irq_ptr->debug_area); return -ENOMEM; } strlcpy(new_entry->dbf_name, text, QDIO_DBF_NAME_LEN); new_entry->dbf_info = irq_ptr->debug_area; mutex_lock(&qdio_dbf_list_mutex); list_add(&new_entry->dbf_list, &qdio_dbf_list); mutex_unlock(&qdio_dbf_list_mutex); } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
jan glauberjan glauber24363.95%266.67%
stefan rasplstefan raspl13736.05%133.33%
Total380100.00%3100.00%


static int qstat_show(struct seq_file *m, void *v) { unsigned char state; struct qdio_q *q = m->private; int i; if (!q) return 0; seq_printf(m, "Timestamp: %Lx Last AI: %Lx\n", q->timestamp, last_ai_time); seq_printf(m, "nr_used: %d ftc: %d last_move: %d\n", atomic_read(&q->nr_buf_used), q->first_to_check, q->last_move); if (q->is_input_q) { seq_printf(m, "polling: %d ack start: %d ack count: %d\n", q->u.in.polling, q->u.in.ack_start, q->u.in.ack_count); seq_printf(m, "DSCI: %d IRQs disabled: %u\n", *(u32 *)q->irq_ptr->dsci, test_bit(QDIO_QUEUE_IRQS_DISABLED, &q->u.in.queue_irq_state)); } seq_printf(m, "SBAL states:\n"); seq_printf(m, "|0 |8 |16 |24 |32 |40 |48 |56 63|\n"); for (i = 0; i < QDIO_MAX_BUFFERS_PER_Q; i++) { debug_get_buf_state(q, i, &state); switch (state) { case SLSB_P_INPUT_NOT_INIT: case SLSB_P_OUTPUT_NOT_INIT: seq_printf(m, "N"); break; case SLSB_P_OUTPUT_PENDING: seq_printf(m, "P"); break; case SLSB_P_INPUT_PRIMED: case SLSB_CU_OUTPUT_PRIMED: seq_printf(m, "+"); break; case SLSB_P_INPUT_ACK: seq_printf(m, "A"); break; case SLSB_P_INPUT_ERROR: case SLSB_P_OUTPUT_ERROR: seq_printf(m, "x"); break; case SLSB_CU_INPUT_EMPTY: case SLSB_P_OUTPUT_EMPTY: seq_printf(m, "-"); break; case SLSB_P_INPUT_HALTED: case SLSB_P_OUTPUT_HALTED: seq_printf(m, "."); break; default: seq_printf(m, "?"); } if (i == 63) seq_printf(m, "\n"); } seq_printf(m, "\n"); seq_printf(m, "|64 |72 |80 |88 |96 |104 |112 | 127|\n"); seq_printf(m, "\nSBAL statistics:"); if (!q->irq_ptr->perf_stat_enabled) { seq_printf(m, " disabled\n"); return 0; } seq_printf(m, "\n1 2.. 4.. 8.. " "16.. 32.. 64.. 127\n"); for (i = 0; i < ARRAY_SIZE(q->q_stats.nr_sbals); i++) seq_printf(m, "%-10u ", q->q_stats.nr_sbals[i]); seq_printf(m, "\nError NOP Total\n%-10u %-10u %-10u\n\n", q->q_stats.nr_sbal_error, q->q_stats.nr_sbal_nop, q->q_stats.nr_sbal_total); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
jan glauberjan glauber40797.37%888.89%
frank blaschkafrank blaschka112.63%111.11%
Total418100.00%9100.00%


static int qstat_seq_open(struct inode *inode, struct file *filp) { return single_open(filp, qstat_show, file_inode(filp)->i_private); }

Contributors

PersonTokensPropCommitsCommitProp
jan glauberjan glauber2890.32%150.00%
al viroal viro39.68%150.00%
Total31100.00%2100.00%

static const struct file_operations debugfs_fops = { .owner = THIS_MODULE, .open = qstat_seq_open, .read = seq_read, .llseek = seq_lseek, .release = single_release, }; static char *qperf_names[] = { "Assumed adapter interrupts", "QDIO interrupts", "Requested PCIs", "Inbound tasklet runs", "Inbound tasklet resched", "Inbound tasklet resched2", "Outbound tasklet runs", "SIGA read", "SIGA write", "SIGA sync", "Inbound calls", "Inbound handler", "Inbound stop_polling", "Inbound queue full", "Outbound calls", "Outbound handler", "Outbound queue full", "Outbound fast_requeue", "Outbound target_full", "QEBSM eqbs", "QEBSM eqbs partial", "QEBSM sqbs", "QEBSM sqbs partial", "Discarded interrupts" };
static int qperf_show(struct seq_file *m, void *v) { struct qdio_irq *irq_ptr = m->private; unsigned int *stat; int i; if (!irq_ptr) return 0; if (!irq_ptr->perf_stat_enabled) { seq_printf(m, "disabled\n"); return 0; } stat = (unsigned int *)&irq_ptr->perf_stat; for (i = 0; i < ARRAY_SIZE(qperf_names); i++) seq_printf(m, "%26s:\t%u\n", qperf_names[i], *(stat + i)); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
jan glauberjan glauber109100.00%1100.00%
Total109100.00%1100.00%


static ssize_t qperf_seq_write(struct file *file, const char __user *ubuf, size_t count, loff_t *off) { struct seq_file *seq = file->private_data; struct qdio_irq *irq_ptr = seq->private; struct qdio_q *q; unsigned long val; int ret, i; if (!irq_ptr) return 0; ret = kstrtoul_from_user(ubuf, count, 10, &val); if (ret) return ret; switch (val) { case 0: irq_ptr->perf_stat_enabled = 0; memset(&irq_ptr->perf_stat, 0, sizeof(irq_ptr->perf_stat)); for_each_input_queue(irq_ptr, q, i) memset(&q->q_stats, 0, sizeof(q->q_stats)); for_each_output_queue(irq_ptr, q, i) memset(&q->q_stats, 0, sizeof(q->q_stats)); break; case 1: irq_ptr->perf_stat_enabled = 1; break; } return count; }

Contributors

PersonTokensPropCommitsCommitProp
jan glauberjan glauber17898.34%266.67%
peter huewepeter huewe31.66%133.33%
Total181100.00%3100.00%


static int qperf_seq_open(struct inode *inode, struct file *filp) { return single_open(filp, qperf_show, file_inode(filp)->i_private); }

Contributors

PersonTokensPropCommitsCommitProp
jan glauberjan glauber2890.32%150.00%
al viroal viro39.68%150.00%
Total31100.00%2100.00%

static const struct file_operations debugfs_perf_fops = { .owner = THIS_MODULE, .open = qperf_seq_open, .read = seq_read, .write = qperf_seq_write, .llseek = seq_lseek, .release = single_release, };
static void setup_debugfs_entry(struct qdio_q *q) { char name[QDIO_DEBUGFS_NAME_LEN]; snprintf(name, QDIO_DEBUGFS_NAME_LEN, "%s_%d", q->is_input_q ? "input" : "output", q->nr); q->debugfs_q = debugfs_create_file(name, S_IFREG | S_IRUGO | S_IWUSR, q->irq_ptr->debugfs_dev, q, &debugfs_fops); if (IS_ERR(q->debugfs_q)) q->debugfs_q = NULL; }

Contributors

PersonTokensPropCommitsCommitProp
jan glauberjan glauber79100.00%4100.00%
Total79100.00%4100.00%


void qdio_setup_debug_entries(struct qdio_irq *irq_ptr, struct ccw_device *cdev) { struct qdio_q *q; int i; irq_ptr->debugfs_dev = debugfs_create_dir(dev_name(&cdev->dev), debugfs_root); if (IS_ERR(irq_ptr->debugfs_dev)) irq_ptr->debugfs_dev = NULL; irq_ptr->debugfs_perf = debugfs_create_file("statistics", S_IFREG | S_IRUGO | S_IWUSR, irq_ptr->debugfs_dev, irq_ptr, &debugfs_perf_fops); if (IS_ERR(irq_ptr->debugfs_perf)) irq_ptr->debugfs_perf = NULL; for_each_input_queue(irq_ptr, q, i) setup_debugfs_entry(q); for_each_output_queue(irq_ptr, q, i) setup_debugfs_entry(q); }

Contributors

PersonTokensPropCommitsCommitProp
jan glauberjan glauber120100.00%3100.00%
Total120100.00%3100.00%


void qdio_shutdown_debug_entries(struct qdio_irq *irq_ptr) { struct qdio_q *q; int i; for_each_input_queue(irq_ptr, q, i) debugfs_remove(q->debugfs_q); for_each_output_queue(irq_ptr, q, i) debugfs_remove(q->debugfs_q); debugfs_remove(irq_ptr->debugfs_perf); debugfs_remove(irq_ptr->debugfs_dev); }

Contributors

PersonTokensPropCommitsCommitProp
jan glauberjan glauber62100.00%3100.00%
Total62100.00%3100.00%


int __init qdio_debug_init(void) { debugfs_root = debugfs_create_dir("qdio", NULL); qdio_dbf_setup = debug_register("qdio_setup", 16, 1, 16); debug_register_view(qdio_dbf_setup, &debug_hex_ascii_view); debug_set_level(qdio_dbf_setup, DBF_INFO); DBF_EVENT("dbf created\n"); qdio_dbf_error = debug_register("qdio_error", 4, 1, 16); debug_register_view(qdio_dbf_error, &debug_hex_ascii_view); debug_set_level(qdio_dbf_error, DBF_INFO); DBF_ERROR("dbf created\n"); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
jan glauberjan glauber86100.00%3100.00%
Total86100.00%3100.00%


void qdio_debug_exit(void) { qdio_clear_dbf_list(); debugfs_remove(debugfs_root); debug_unregister(qdio_dbf_setup); debug_unregister(qdio_dbf_error); }

Contributors

PersonTokensPropCommitsCommitProp
jan glauberjan glauber2288.00%266.67%
stefan rasplstefan raspl312.00%133.33%
Total25100.00%3100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
jan glauberjan glauber152181.95%1361.90%
stefan rasplstefan raspl30616.49%14.76%
frank blaschkafrank blaschka110.59%14.76%
heiko carstensheiko carstens70.38%29.52%
al viroal viro70.38%29.52%
peter huewepeter huewe30.16%14.76%
alexey dobriyanalexey dobriyan10.05%14.76%
Total1856100.00%21100.00%
Directory: drivers/s390/cio
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
{% endraw %}