1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * QLogic FCoE Offload Driver
4*4882a593Smuzhiyun * Copyright (c) 2016-2018 Cavium Inc.
5*4882a593Smuzhiyun */
6*4882a593Smuzhiyun #include "qedf.h"
7*4882a593Smuzhiyun
qedf_is_vport(struct qedf_ctx * qedf)8*4882a593Smuzhiyun inline bool qedf_is_vport(struct qedf_ctx *qedf)
9*4882a593Smuzhiyun {
10*4882a593Smuzhiyun return qedf->lport->vport != NULL;
11*4882a593Smuzhiyun }
12*4882a593Smuzhiyun
13*4882a593Smuzhiyun /* Get base qedf for physical port from vport */
qedf_get_base_qedf(struct qedf_ctx * qedf)14*4882a593Smuzhiyun static struct qedf_ctx *qedf_get_base_qedf(struct qedf_ctx *qedf)
15*4882a593Smuzhiyun {
16*4882a593Smuzhiyun struct fc_lport *lport;
17*4882a593Smuzhiyun struct fc_lport *base_lport;
18*4882a593Smuzhiyun
19*4882a593Smuzhiyun if (!(qedf_is_vport(qedf)))
20*4882a593Smuzhiyun return NULL;
21*4882a593Smuzhiyun
22*4882a593Smuzhiyun lport = qedf->lport;
23*4882a593Smuzhiyun base_lport = shost_priv(vport_to_shost(lport->vport));
24*4882a593Smuzhiyun return lport_priv(base_lport);
25*4882a593Smuzhiyun }
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun static ssize_t
qedf_fcoe_mac_show(struct device * dev,struct device_attribute * attr,char * buf)28*4882a593Smuzhiyun qedf_fcoe_mac_show(struct device *dev,
29*4882a593Smuzhiyun struct device_attribute *attr, char *buf)
30*4882a593Smuzhiyun {
31*4882a593Smuzhiyun struct fc_lport *lport = shost_priv(class_to_shost(dev));
32*4882a593Smuzhiyun u32 port_id;
33*4882a593Smuzhiyun u8 lport_src_id[3];
34*4882a593Smuzhiyun u8 fcoe_mac[6];
35*4882a593Smuzhiyun
36*4882a593Smuzhiyun port_id = fc_host_port_id(lport->host);
37*4882a593Smuzhiyun lport_src_id[2] = (port_id & 0x000000FF);
38*4882a593Smuzhiyun lport_src_id[1] = (port_id & 0x0000FF00) >> 8;
39*4882a593Smuzhiyun lport_src_id[0] = (port_id & 0x00FF0000) >> 16;
40*4882a593Smuzhiyun fc_fcoe_set_mac(fcoe_mac, lport_src_id);
41*4882a593Smuzhiyun
42*4882a593Smuzhiyun return scnprintf(buf, PAGE_SIZE, "%pM\n", fcoe_mac);
43*4882a593Smuzhiyun }
44*4882a593Smuzhiyun
45*4882a593Smuzhiyun static ssize_t
qedf_fka_period_show(struct device * dev,struct device_attribute * attr,char * buf)46*4882a593Smuzhiyun qedf_fka_period_show(struct device *dev,
47*4882a593Smuzhiyun struct device_attribute *attr, char *buf)
48*4882a593Smuzhiyun {
49*4882a593Smuzhiyun struct fc_lport *lport = shost_priv(class_to_shost(dev));
50*4882a593Smuzhiyun struct qedf_ctx *qedf = lport_priv(lport);
51*4882a593Smuzhiyun int fka_period = -1;
52*4882a593Smuzhiyun
53*4882a593Smuzhiyun if (qedf_is_vport(qedf))
54*4882a593Smuzhiyun qedf = qedf_get_base_qedf(qedf);
55*4882a593Smuzhiyun
56*4882a593Smuzhiyun if (qedf->ctlr.sel_fcf)
57*4882a593Smuzhiyun fka_period = qedf->ctlr.sel_fcf->fka_period;
58*4882a593Smuzhiyun
59*4882a593Smuzhiyun return scnprintf(buf, PAGE_SIZE, "%d\n", fka_period);
60*4882a593Smuzhiyun }
61*4882a593Smuzhiyun
62*4882a593Smuzhiyun static DEVICE_ATTR(fcoe_mac, S_IRUGO, qedf_fcoe_mac_show, NULL);
63*4882a593Smuzhiyun static DEVICE_ATTR(fka_period, S_IRUGO, qedf_fka_period_show, NULL);
64*4882a593Smuzhiyun
65*4882a593Smuzhiyun struct device_attribute *qedf_host_attrs[] = {
66*4882a593Smuzhiyun &dev_attr_fcoe_mac,
67*4882a593Smuzhiyun &dev_attr_fka_period,
68*4882a593Smuzhiyun NULL,
69*4882a593Smuzhiyun };
70*4882a593Smuzhiyun
71*4882a593Smuzhiyun extern const struct qed_fcoe_ops *qed_ops;
72*4882a593Smuzhiyun
qedf_capture_grc_dump(struct qedf_ctx * qedf)73*4882a593Smuzhiyun void qedf_capture_grc_dump(struct qedf_ctx *qedf)
74*4882a593Smuzhiyun {
75*4882a593Smuzhiyun struct qedf_ctx *base_qedf;
76*4882a593Smuzhiyun
77*4882a593Smuzhiyun /* Make sure we use the base qedf to take the GRC dump */
78*4882a593Smuzhiyun if (qedf_is_vport(qedf))
79*4882a593Smuzhiyun base_qedf = qedf_get_base_qedf(qedf);
80*4882a593Smuzhiyun else
81*4882a593Smuzhiyun base_qedf = qedf;
82*4882a593Smuzhiyun
83*4882a593Smuzhiyun if (test_bit(QEDF_GRCDUMP_CAPTURE, &base_qedf->flags)) {
84*4882a593Smuzhiyun QEDF_INFO(&(base_qedf->dbg_ctx), QEDF_LOG_INFO,
85*4882a593Smuzhiyun "GRC Dump already captured.\n");
86*4882a593Smuzhiyun return;
87*4882a593Smuzhiyun }
88*4882a593Smuzhiyun
89*4882a593Smuzhiyun
90*4882a593Smuzhiyun qedf_get_grc_dump(base_qedf->cdev, qed_ops->common,
91*4882a593Smuzhiyun &base_qedf->grcdump, &base_qedf->grcdump_size);
92*4882a593Smuzhiyun QEDF_ERR(&(base_qedf->dbg_ctx), "GRC Dump captured.\n");
93*4882a593Smuzhiyun set_bit(QEDF_GRCDUMP_CAPTURE, &base_qedf->flags);
94*4882a593Smuzhiyun qedf_uevent_emit(base_qedf->lport->host, QEDF_UEVENT_CODE_GRCDUMP,
95*4882a593Smuzhiyun NULL);
96*4882a593Smuzhiyun }
97*4882a593Smuzhiyun
98*4882a593Smuzhiyun static ssize_t
qedf_sysfs_read_grcdump(struct file * filep,struct kobject * kobj,struct bin_attribute * ba,char * buf,loff_t off,size_t count)99*4882a593Smuzhiyun qedf_sysfs_read_grcdump(struct file *filep, struct kobject *kobj,
100*4882a593Smuzhiyun struct bin_attribute *ba, char *buf, loff_t off,
101*4882a593Smuzhiyun size_t count)
102*4882a593Smuzhiyun {
103*4882a593Smuzhiyun ssize_t ret = 0;
104*4882a593Smuzhiyun struct fc_lport *lport = shost_priv(dev_to_shost(container_of(kobj,
105*4882a593Smuzhiyun struct device, kobj)));
106*4882a593Smuzhiyun struct qedf_ctx *qedf = lport_priv(lport);
107*4882a593Smuzhiyun
108*4882a593Smuzhiyun if (test_bit(QEDF_GRCDUMP_CAPTURE, &qedf->flags)) {
109*4882a593Smuzhiyun ret = memory_read_from_buffer(buf, count, &off,
110*4882a593Smuzhiyun qedf->grcdump, qedf->grcdump_size);
111*4882a593Smuzhiyun } else {
112*4882a593Smuzhiyun QEDF_ERR(&(qedf->dbg_ctx), "GRC Dump not captured!\n");
113*4882a593Smuzhiyun }
114*4882a593Smuzhiyun
115*4882a593Smuzhiyun return ret;
116*4882a593Smuzhiyun }
117*4882a593Smuzhiyun
118*4882a593Smuzhiyun static ssize_t
qedf_sysfs_write_grcdump(struct file * filep,struct kobject * kobj,struct bin_attribute * ba,char * buf,loff_t off,size_t count)119*4882a593Smuzhiyun qedf_sysfs_write_grcdump(struct file *filep, struct kobject *kobj,
120*4882a593Smuzhiyun struct bin_attribute *ba, char *buf, loff_t off,
121*4882a593Smuzhiyun size_t count)
122*4882a593Smuzhiyun {
123*4882a593Smuzhiyun struct fc_lport *lport = NULL;
124*4882a593Smuzhiyun struct qedf_ctx *qedf = NULL;
125*4882a593Smuzhiyun long reading;
126*4882a593Smuzhiyun int ret = 0;
127*4882a593Smuzhiyun char msg[40];
128*4882a593Smuzhiyun
129*4882a593Smuzhiyun if (off != 0)
130*4882a593Smuzhiyun return ret;
131*4882a593Smuzhiyun
132*4882a593Smuzhiyun
133*4882a593Smuzhiyun lport = shost_priv(dev_to_shost(container_of(kobj,
134*4882a593Smuzhiyun struct device, kobj)));
135*4882a593Smuzhiyun qedf = lport_priv(lport);
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun buf[1] = 0;
138*4882a593Smuzhiyun ret = kstrtol(buf, 10, &reading);
139*4882a593Smuzhiyun if (ret) {
140*4882a593Smuzhiyun QEDF_ERR(&(qedf->dbg_ctx), "Invalid input, err(%d)\n", ret);
141*4882a593Smuzhiyun return ret;
142*4882a593Smuzhiyun }
143*4882a593Smuzhiyun
144*4882a593Smuzhiyun memset(msg, 0, sizeof(msg));
145*4882a593Smuzhiyun switch (reading) {
146*4882a593Smuzhiyun case 0:
147*4882a593Smuzhiyun memset(qedf->grcdump, 0, qedf->grcdump_size);
148*4882a593Smuzhiyun clear_bit(QEDF_GRCDUMP_CAPTURE, &qedf->flags);
149*4882a593Smuzhiyun break;
150*4882a593Smuzhiyun case 1:
151*4882a593Smuzhiyun qedf_capture_grc_dump(qedf);
152*4882a593Smuzhiyun break;
153*4882a593Smuzhiyun }
154*4882a593Smuzhiyun
155*4882a593Smuzhiyun return count;
156*4882a593Smuzhiyun }
157*4882a593Smuzhiyun
158*4882a593Smuzhiyun static struct bin_attribute sysfs_grcdump_attr = {
159*4882a593Smuzhiyun .attr = {
160*4882a593Smuzhiyun .name = "grcdump",
161*4882a593Smuzhiyun .mode = S_IRUSR | S_IWUSR,
162*4882a593Smuzhiyun },
163*4882a593Smuzhiyun .size = 0,
164*4882a593Smuzhiyun .read = qedf_sysfs_read_grcdump,
165*4882a593Smuzhiyun .write = qedf_sysfs_write_grcdump,
166*4882a593Smuzhiyun };
167*4882a593Smuzhiyun
168*4882a593Smuzhiyun static struct sysfs_bin_attrs bin_file_entries[] = {
169*4882a593Smuzhiyun {"grcdump", &sysfs_grcdump_attr},
170*4882a593Smuzhiyun {NULL},
171*4882a593Smuzhiyun };
172*4882a593Smuzhiyun
qedf_create_sysfs_ctx_attr(struct qedf_ctx * qedf)173*4882a593Smuzhiyun void qedf_create_sysfs_ctx_attr(struct qedf_ctx *qedf)
174*4882a593Smuzhiyun {
175*4882a593Smuzhiyun qedf_create_sysfs_attr(qedf->lport->host, bin_file_entries);
176*4882a593Smuzhiyun }
177*4882a593Smuzhiyun
qedf_remove_sysfs_ctx_attr(struct qedf_ctx * qedf)178*4882a593Smuzhiyun void qedf_remove_sysfs_ctx_attr(struct qedf_ctx *qedf)
179*4882a593Smuzhiyun {
180*4882a593Smuzhiyun qedf_remove_sysfs_attr(qedf->lport->host, bin_file_entries);
181*4882a593Smuzhiyun }
182