blob: ab4a91393005a920b876c610ed5d3d8605af5037 [file] [log] [blame]
Jan Glauber1e8da952012-11-29 14:36:55 +01001/*
2 * Copyright IBM Corp. 2012
3 *
4 * Author(s):
5 * Jan Glauber <jang@linux.vnet.ibm.com>
6 */
7
8#define COMPONENT "zPCI"
9#define pr_fmt(fmt) COMPONENT ": " fmt
10
11#include <linux/kernel.h>
12#include <linux/stat.h>
13#include <linux/pci.h>
14
15static ssize_t show_fid(struct device *dev, struct device_attribute *attr,
16 char *buf)
17{
Sebastian Ott92948962013-05-17 16:33:40 +020018 struct zpci_dev *zdev = get_zdev(to_pci_dev(dev));
Jan Glauber1e8da952012-11-29 14:36:55 +010019
Sebastian Ott80b054b2013-06-05 16:08:07 +020020 return sprintf(buf, "0x%08x\n", zdev->fid);
Jan Glauber1e8da952012-11-29 14:36:55 +010021}
22static DEVICE_ATTR(function_id, S_IRUGO, show_fid, NULL);
23
24static ssize_t show_fh(struct device *dev, struct device_attribute *attr,
25 char *buf)
26{
Sebastian Ott92948962013-05-17 16:33:40 +020027 struct zpci_dev *zdev = get_zdev(to_pci_dev(dev));
Jan Glauber1e8da952012-11-29 14:36:55 +010028
Sebastian Ott80b054b2013-06-05 16:08:07 +020029 return sprintf(buf, "0x%08x\n", zdev->fh);
Jan Glauber1e8da952012-11-29 14:36:55 +010030}
31static DEVICE_ATTR(function_handle, S_IRUGO, show_fh, NULL);
32
33static ssize_t show_pchid(struct device *dev, struct device_attribute *attr,
34 char *buf)
35{
Sebastian Ott92948962013-05-17 16:33:40 +020036 struct zpci_dev *zdev = get_zdev(to_pci_dev(dev));
Jan Glauber1e8da952012-11-29 14:36:55 +010037
Sebastian Ott80b054b2013-06-05 16:08:07 +020038 return sprintf(buf, "0x%04x\n", zdev->pchid);
Jan Glauber1e8da952012-11-29 14:36:55 +010039}
40static DEVICE_ATTR(pchid, S_IRUGO, show_pchid, NULL);
41
42static ssize_t show_pfgid(struct device *dev, struct device_attribute *attr,
43 char *buf)
44{
Sebastian Ott92948962013-05-17 16:33:40 +020045 struct zpci_dev *zdev = get_zdev(to_pci_dev(dev));
Jan Glauber1e8da952012-11-29 14:36:55 +010046
Sebastian Ott80b054b2013-06-05 16:08:07 +020047 return sprintf(buf, "0x%02x\n", zdev->pfgid);
Jan Glauber1e8da952012-11-29 14:36:55 +010048}
49static DEVICE_ATTR(pfgid, S_IRUGO, show_pfgid, NULL);
50
Tejun Heo0b60f9e2014-02-03 14:03:04 -050051static ssize_t store_recover(struct device *dev, struct device_attribute *attr,
52 const char *buf, size_t count)
Sebastian Ott0ff70ec2013-08-29 19:35:19 +020053{
54 struct pci_dev *pdev = to_pci_dev(dev);
55 struct zpci_dev *zdev = get_zdev(pdev);
56 int ret;
57
Tejun Heo0b60f9e2014-02-03 14:03:04 -050058 if (!device_remove_file_self(dev, attr))
59 return count;
60
Sebastian Ott0ff70ec2013-08-29 19:35:19 +020061 pci_stop_and_remove_bus_device(pdev);
62 ret = zpci_disable_device(zdev);
63 if (ret)
Tejun Heo0b60f9e2014-02-03 14:03:04 -050064 return ret;
Sebastian Ott0ff70ec2013-08-29 19:35:19 +020065
66 ret = zpci_enable_device(zdev);
67 if (ret)
Tejun Heo0b60f9e2014-02-03 14:03:04 -050068 return ret;
Sebastian Ott0ff70ec2013-08-29 19:35:19 +020069
70 pci_rescan_bus(zdev->bus);
Tejun Heo0b60f9e2014-02-03 14:03:04 -050071 return count;
Sebastian Ott0ff70ec2013-08-29 19:35:19 +020072}
73static DEVICE_ATTR(recover, S_IWUSR, NULL, store_recover);
74
Jan Glauber1e8da952012-11-29 14:36:55 +010075static struct device_attribute *zpci_dev_attrs[] = {
76 &dev_attr_function_id,
77 &dev_attr_function_handle,
78 &dev_attr_pchid,
79 &dev_attr_pfgid,
Sebastian Ott0ff70ec2013-08-29 19:35:19 +020080 &dev_attr_recover,
Jan Glauber1e8da952012-11-29 14:36:55 +010081 NULL,
82};
83
84int zpci_sysfs_add_device(struct device *dev)
85{
86 int i, rc = 0;
87
88 for (i = 0; zpci_dev_attrs[i]; i++) {
89 rc = device_create_file(dev, zpci_dev_attrs[i]);
90 if (rc)
91 goto error;
92 }
93 return 0;
94
95error:
96 while (--i >= 0)
97 device_remove_file(dev, zpci_dev_attrs[i]);
98 return rc;
99}
100
101void zpci_sysfs_remove_device(struct device *dev)
102{
103 int i;
104
105 for (i = 0; zpci_dev_attrs[i]; i++)
106 device_remove_file(dev, zpci_dev_attrs[i]);
107}