diff options
author | Upinder Malhi <umalhi@cisco.com> | 2013-09-09 23:38:16 -0400 |
---|---|---|
committer | Roland Dreier <roland@purestorage.com> | 2014-01-14 03:44:28 -0500 |
commit | e3cf00d0a87f025db5855a43a67c67a41fa79fef (patch) | |
tree | 34ab93de5918af08434dcf063e4ff5ae78425989 /drivers/infiniband | |
parent | 7e22e91102c6b9df7c4ae2168910e19d2bb14cd6 (diff) |
IB/usnic: Add Cisco VIC low-level hardware driver
This adds a driver that allows userspace to use UD-like QPs over a
proprietary Cisco transport with Cisco's Virtual Interface Cards (VICs),
including VIC 1240 and 1280 cards.
Signed-off-by: Upinder Malhi <umalhi@cisco.com>
Signed-off-by: Christian Benvenuti <benve@cisco.com>
Signed-off-by: Roland Dreier <roland@purestorage.com>
Diffstat (limited to 'drivers/infiniband')
29 files changed, 4899 insertions, 0 deletions
diff --git a/drivers/infiniband/Kconfig b/drivers/infiniband/Kconfig index 5ceda710f516..10219ee92191 100644 --- a/drivers/infiniband/Kconfig +++ b/drivers/infiniband/Kconfig | |||
@@ -53,6 +53,7 @@ source "drivers/infiniband/hw/mlx4/Kconfig" | |||
53 | source "drivers/infiniband/hw/mlx5/Kconfig" | 53 | source "drivers/infiniband/hw/mlx5/Kconfig" |
54 | source "drivers/infiniband/hw/nes/Kconfig" | 54 | source "drivers/infiniband/hw/nes/Kconfig" |
55 | source "drivers/infiniband/hw/ocrdma/Kconfig" | 55 | source "drivers/infiniband/hw/ocrdma/Kconfig" |
56 | source "drivers/infiniband/hw/usnic/Kconfig" | ||
56 | 57 | ||
57 | source "drivers/infiniband/ulp/ipoib/Kconfig" | 58 | source "drivers/infiniband/ulp/ipoib/Kconfig" |
58 | 59 | ||
diff --git a/drivers/infiniband/Makefile b/drivers/infiniband/Makefile index 1fe69888515f..bf508b5550c4 100644 --- a/drivers/infiniband/Makefile +++ b/drivers/infiniband/Makefile | |||
@@ -10,6 +10,7 @@ obj-$(CONFIG_MLX4_INFINIBAND) += hw/mlx4/ | |||
10 | obj-$(CONFIG_MLX5_INFINIBAND) += hw/mlx5/ | 10 | obj-$(CONFIG_MLX5_INFINIBAND) += hw/mlx5/ |
11 | obj-$(CONFIG_INFINIBAND_NES) += hw/nes/ | 11 | obj-$(CONFIG_INFINIBAND_NES) += hw/nes/ |
12 | obj-$(CONFIG_INFINIBAND_OCRDMA) += hw/ocrdma/ | 12 | obj-$(CONFIG_INFINIBAND_OCRDMA) += hw/ocrdma/ |
13 | obj-$(CONFIG_INFINIBAND_USNIC) += hw/usnic/ | ||
13 | obj-$(CONFIG_INFINIBAND_IPOIB) += ulp/ipoib/ | 14 | obj-$(CONFIG_INFINIBAND_IPOIB) += ulp/ipoib/ |
14 | obj-$(CONFIG_INFINIBAND_SRP) += ulp/srp/ | 15 | obj-$(CONFIG_INFINIBAND_SRP) += ulp/srp/ |
15 | obj-$(CONFIG_INFINIBAND_SRPT) += ulp/srpt/ | 16 | obj-$(CONFIG_INFINIBAND_SRPT) += ulp/srpt/ |
diff --git a/drivers/infiniband/hw/usnic/Kconfig b/drivers/infiniband/hw/usnic/Kconfig new file mode 100644 index 000000000000..2cc8ba00b34b --- /dev/null +++ b/drivers/infiniband/hw/usnic/Kconfig | |||
@@ -0,0 +1,10 @@ | |||
1 | config INFINIBAND_USNIC | ||
2 | tristate "Verbs support for Cisco VIC" | ||
3 | depends on NETDEVICES && ETHERNET && PCI && INTEL_IOMMU | ||
4 | select ENIC | ||
5 | select NET_VENDOR_CISCO | ||
6 | select PCI_IOV | ||
7 | select INFINIBAND_USER_ACCESS | ||
8 | ---help--- | ||
9 | This is a low-level driver for Cisco's Virtual Interface | ||
10 | Cards (VICs), including the VIC 1240 and 1280 cards. | ||
diff --git a/drivers/infiniband/hw/usnic/Makefile b/drivers/infiniband/hw/usnic/Makefile new file mode 100644 index 000000000000..99fb2db47cd5 --- /dev/null +++ b/drivers/infiniband/hw/usnic/Makefile | |||
@@ -0,0 +1,15 @@ | |||
1 | ccflags-y := -Idrivers/net/ethernet/cisco/enic | ||
2 | |||
3 | obj-$(CONFIG_INFINIBAND_USNIC)+= usnic_verbs.o | ||
4 | |||
5 | usnic_verbs-y=\ | ||
6 | usnic_fwd.o \ | ||
7 | usnic_transport.o \ | ||
8 | usnic_uiom.o \ | ||
9 | usnic_uiom_interval_tree.o \ | ||
10 | usnic_vnic.o \ | ||
11 | usnic_ib_main.o \ | ||
12 | usnic_ib_qp_grp.o \ | ||
13 | usnic_ib_sysfs.o \ | ||
14 | usnic_ib_verbs.o \ | ||
15 | usnic_debugfs.o \ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic.h b/drivers/infiniband/hw/usnic/usnic.h new file mode 100644 index 000000000000..d741c76bc4be --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic.h | |||
@@ -0,0 +1,29 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_H_ | ||
20 | #define USNIC_H_ | ||
21 | |||
22 | #define DRV_NAME "usnic_verbs" | ||
23 | |||
24 | #define PCI_DEVICE_ID_CISCO_VIC_USPACE_NIC 0x00cf /* User space NIC */ | ||
25 | |||
26 | #define DRV_VERSION "1.0.2" | ||
27 | #define DRV_RELDATE "September 09, 2013" | ||
28 | |||
29 | #endif /* USNIC_H_ */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_abi.h b/drivers/infiniband/hw/usnic/usnic_abi.h new file mode 100644 index 000000000000..510b7d702465 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_abi.h | |||
@@ -0,0 +1,56 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | |||
20 | #ifndef USNIC_ABI_H | ||
21 | #define USNIC_ABI_H | ||
22 | |||
23 | /* ABI between userspace and kernel */ | ||
24 | #define USNIC_UVERBS_ABI_VERSION 2 | ||
25 | |||
26 | #define USNIC_QP_GRP_MAX_WQS 8 | ||
27 | #define USNIC_QP_GRP_MAX_RQS 8 | ||
28 | #define USNIC_QP_GRP_MAX_CQS 16 | ||
29 | |||
30 | enum usnic_transport_type { | ||
31 | USNIC_TRANSPORT_UNKNOWN = 0, | ||
32 | USNIC_TRANSPORT_ROCE_CUSTOM = 1, | ||
33 | USNIC_TRANSPORT_MAX = 2, | ||
34 | }; | ||
35 | |||
36 | /*TODO: Future - usnic_modify_qp needs to pass in generic filters */ | ||
37 | struct usnic_ib_create_qp_resp { | ||
38 | u32 vfid; | ||
39 | u32 qp_grp_id; | ||
40 | u64 bar_bus_addr; | ||
41 | u32 bar_len; | ||
42 | /* | ||
43 | * WQ, RQ, CQ are explicity specified bc exposing a generic resources inteface | ||
44 | * expands the scope of ABI to many files. | ||
45 | */ | ||
46 | u32 wq_cnt; | ||
47 | u32 rq_cnt; | ||
48 | u32 cq_cnt; | ||
49 | u32 wq_idx[USNIC_QP_GRP_MAX_WQS]; | ||
50 | u32 rq_idx[USNIC_QP_GRP_MAX_RQS]; | ||
51 | u32 cq_idx[USNIC_QP_GRP_MAX_CQS]; | ||
52 | u32 transport; | ||
53 | u32 reserved[9]; | ||
54 | }; | ||
55 | |||
56 | #endif /* USNIC_ABI_H */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_common_pkt_hdr.h b/drivers/infiniband/hw/usnic/usnic_common_pkt_hdr.h new file mode 100644 index 000000000000..393567266142 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_common_pkt_hdr.h | |||
@@ -0,0 +1,27 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_CMN_PKT_HDR_H | ||
20 | #define USNIC_CMN_PKT_HDR_H | ||
21 | |||
22 | #define USNIC_ROCE_ETHERTYPE (0x8915) | ||
23 | #define USNIC_ROCE_GRH_VER (8) | ||
24 | #define USNIC_PROTO_VER (1) | ||
25 | #define USNIC_ROCE_GRH_VER_SHIFT (4) | ||
26 | |||
27 | #endif /* USNIC_COMMON_PKT_HDR_H */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_common_util.h b/drivers/infiniband/hw/usnic/usnic_common_util.h new file mode 100644 index 000000000000..128550a4f9e2 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_common_util.h | |||
@@ -0,0 +1,51 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_CMN_UTIL_H | ||
20 | #define USNIC_CMN_UTIL_H | ||
21 | |||
22 | static inline void | ||
23 | usnic_mac_to_gid(const char *const mac, char *raw_gid) | ||
24 | { | ||
25 | raw_gid[0] = 0xfe; | ||
26 | raw_gid[1] = 0x80; | ||
27 | memset(&raw_gid[2], 0, 6); | ||
28 | raw_gid[8] = mac[0]^2; | ||
29 | raw_gid[9] = mac[1]; | ||
30 | raw_gid[10] = mac[2]; | ||
31 | raw_gid[11] = 0xff; | ||
32 | raw_gid[12] = 0xfe; | ||
33 | raw_gid[13] = mac[3]; | ||
34 | raw_gid[14] = mac[4]; | ||
35 | raw_gid[15] = mac[5]; | ||
36 | } | ||
37 | |||
38 | static inline void | ||
39 | usnic_write_gid_if_id_from_mac(char *mac, char *raw_gid) | ||
40 | { | ||
41 | raw_gid[8] = mac[0]^2; | ||
42 | raw_gid[9] = mac[1]; | ||
43 | raw_gid[10] = mac[2]; | ||
44 | raw_gid[11] = 0xff; | ||
45 | raw_gid[12] = 0xfe; | ||
46 | raw_gid[13] = mac[3]; | ||
47 | raw_gid[14] = mac[4]; | ||
48 | raw_gid[15] = mac[5]; | ||
49 | } | ||
50 | |||
51 | #endif /* USNIC_COMMON_UTIL_H */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_debugfs.c b/drivers/infiniband/hw/usnic/usnic_debugfs.c new file mode 100644 index 000000000000..91386df025ae --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_debugfs.c | |||
@@ -0,0 +1,71 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #include <linux/debugfs.h> | ||
20 | #include <linux/module.h> | ||
21 | |||
22 | #include "usnic.h" | ||
23 | #include "usnic_log.h" | ||
24 | #include "usnic_debugfs.h" | ||
25 | |||
26 | static struct dentry *debugfs_root; | ||
27 | |||
28 | static ssize_t usnic_debugfs_buildinfo_read(struct file *f, char __user *data, | ||
29 | size_t count, loff_t *ppos) | ||
30 | { | ||
31 | char buf[500]; | ||
32 | int res; | ||
33 | |||
34 | if (*ppos > 0) | ||
35 | return 0; | ||
36 | |||
37 | res = scnprintf(buf, sizeof(buf), | ||
38 | "version: %s\n" | ||
39 | "build date: %s\n", | ||
40 | DRV_VERSION, DRV_RELDATE); | ||
41 | |||
42 | return simple_read_from_buffer(data, count, ppos, buf, res); | ||
43 | } | ||
44 | |||
45 | static const struct file_operations usnic_debugfs_buildinfo_ops = { | ||
46 | .owner = THIS_MODULE, | ||
47 | .open = simple_open, | ||
48 | .read = usnic_debugfs_buildinfo_read | ||
49 | }; | ||
50 | |||
51 | void usnic_debugfs_init(void) | ||
52 | { | ||
53 | debugfs_root = debugfs_create_dir(DRV_NAME, NULL); | ||
54 | if (IS_ERR(debugfs_root)) { | ||
55 | usnic_err("Failed to create debugfs root dir, check if debugfs is enabled in kernel configuration\n"); | ||
56 | debugfs_root = NULL; | ||
57 | return; | ||
58 | } | ||
59 | |||
60 | debugfs_create_file("build-info", S_IRUGO, debugfs_root, | ||
61 | NULL, &usnic_debugfs_buildinfo_ops); | ||
62 | } | ||
63 | |||
64 | void usnic_debugfs_exit(void) | ||
65 | { | ||
66 | if (!debugfs_root) | ||
67 | return; | ||
68 | |||
69 | debugfs_remove_recursive(debugfs_root); | ||
70 | debugfs_root = NULL; | ||
71 | } | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_debugfs.h b/drivers/infiniband/hw/usnic/usnic_debugfs.h new file mode 100644 index 000000000000..914a330dd5e7 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_debugfs.h | |||
@@ -0,0 +1,25 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | #ifndef USNIC_DEBUGFS_H_ | ||
19 | #define USNIC_DEBUGFS_H_ | ||
20 | |||
21 | void usnic_debugfs_init(void); | ||
22 | |||
23 | void usnic_debugfs_exit(void); | ||
24 | |||
25 | #endif /*!USNIC_DEBUGFS_H_ */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_fwd.c b/drivers/infiniband/hw/usnic/usnic_fwd.c new file mode 100644 index 000000000000..8e42216362e7 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_fwd.c | |||
@@ -0,0 +1,243 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | #include <linux/netdevice.h> | ||
19 | #include <linux/pci.h> | ||
20 | |||
21 | #include "enic_api.h" | ||
22 | #include "usnic_common_pkt_hdr.h" | ||
23 | #include "usnic_fwd.h" | ||
24 | #include "usnic_log.h" | ||
25 | |||
26 | struct usnic_fwd_dev *usnic_fwd_dev_alloc(struct pci_dev *pdev) | ||
27 | { | ||
28 | struct usnic_fwd_dev *ufdev; | ||
29 | |||
30 | ufdev = kzalloc(sizeof(*ufdev), GFP_KERNEL); | ||
31 | if (!ufdev) | ||
32 | return NULL; | ||
33 | |||
34 | ufdev->pdev = pdev; | ||
35 | ufdev->netdev = pci_get_drvdata(pdev); | ||
36 | spin_lock_init(&ufdev->lock); | ||
37 | |||
38 | return ufdev; | ||
39 | } | ||
40 | |||
41 | void usnic_fwd_dev_free(struct usnic_fwd_dev *ufdev) | ||
42 | { | ||
43 | kfree(ufdev); | ||
44 | } | ||
45 | |||
46 | static int usnic_fwd_devcmd(struct usnic_fwd_dev *ufdev, int vnic_idx, | ||
47 | enum vnic_devcmd_cmd cmd, u64 *a0, u64 *a1) | ||
48 | { | ||
49 | int status; | ||
50 | struct net_device *netdev = ufdev->netdev; | ||
51 | |||
52 | spin_lock(&ufdev->lock); | ||
53 | status = enic_api_devcmd_proxy_by_index(netdev, | ||
54 | vnic_idx, | ||
55 | cmd, | ||
56 | a0, a1, | ||
57 | 1000); | ||
58 | spin_unlock(&ufdev->lock); | ||
59 | if (status) { | ||
60 | if (status == ERR_EINVAL && cmd == CMD_DEL_FILTER) { | ||
61 | usnic_dbg("Dev %s vnic idx %u cmd %u already deleted", | ||
62 | netdev_name(netdev), vnic_idx, cmd); | ||
63 | } else { | ||
64 | usnic_err("Dev %s vnic idx %u cmd %u failed with status %d\n", | ||
65 | netdev_name(netdev), vnic_idx, cmd, | ||
66 | status); | ||
67 | } | ||
68 | } else { | ||
69 | usnic_dbg("Dev %s vnic idx %u cmd %u success", | ||
70 | netdev_name(netdev), vnic_idx, | ||
71 | cmd); | ||
72 | } | ||
73 | |||
74 | return status; | ||
75 | } | ||
76 | |||
77 | int usnic_fwd_add_usnic_filter(struct usnic_fwd_dev *ufdev, int vnic_idx, | ||
78 | int rq_idx, struct usnic_fwd_filter *fwd_filter, | ||
79 | struct usnic_fwd_filter_hndl **filter_hndl) | ||
80 | { | ||
81 | struct filter_tlv *tlv, *tlv_va; | ||
82 | struct filter *filter; | ||
83 | struct filter_action *action; | ||
84 | struct pci_dev *pdev; | ||
85 | struct usnic_fwd_filter_hndl *usnic_filter_hndl; | ||
86 | int status; | ||
87 | u64 a0, a1; | ||
88 | u64 tlv_size; | ||
89 | dma_addr_t tlv_pa; | ||
90 | |||
91 | pdev = ufdev->pdev; | ||
92 | tlv_size = (2*sizeof(struct filter_tlv) + | ||
93 | sizeof(struct filter) + | ||
94 | sizeof(struct filter_action)); | ||
95 | tlv = pci_alloc_consistent(pdev, tlv_size, &tlv_pa); | ||
96 | if (!tlv) { | ||
97 | usnic_err("Failed to allocate memory\n"); | ||
98 | return -ENOMEM; | ||
99 | } | ||
100 | |||
101 | usnic_filter_hndl = kzalloc(sizeof(*usnic_filter_hndl), GFP_ATOMIC); | ||
102 | if (!usnic_filter_hndl) { | ||
103 | usnic_err("Failed to allocate memory for hndl\n"); | ||
104 | pci_free_consistent(pdev, tlv_size, tlv, tlv_pa); | ||
105 | return -ENOMEM; | ||
106 | } | ||
107 | |||
108 | tlv_va = tlv; | ||
109 | a0 = tlv_pa; | ||
110 | a1 = tlv_size; | ||
111 | memset(tlv, 0, tlv_size); | ||
112 | tlv->type = CLSF_TLV_FILTER; | ||
113 | tlv->length = sizeof(struct filter); | ||
114 | filter = (struct filter *)&tlv->val; | ||
115 | filter->type = FILTER_USNIC_ID; | ||
116 | filter->u.usnic.ethtype = USNIC_ROCE_ETHERTYPE; | ||
117 | filter->u.usnic.flags = FILTER_FIELD_USNIC_ETHTYPE | | ||
118 | FILTER_FIELD_USNIC_ID | | ||
119 | FILTER_FIELD_USNIC_PROTO; | ||
120 | filter->u.usnic.proto_version = (USNIC_ROCE_GRH_VER << | ||
121 | USNIC_ROCE_GRH_VER_SHIFT) | ||
122 | | USNIC_PROTO_VER; | ||
123 | filter->u.usnic.usnic_id = fwd_filter->port_num; | ||
124 | tlv = (struct filter_tlv *)((char *)tlv + sizeof(struct filter_tlv) + | ||
125 | sizeof(struct filter)); | ||
126 | tlv->type = CLSF_TLV_ACTION; | ||
127 | tlv->length = sizeof(struct filter_action); | ||
128 | action = (struct filter_action *)&tlv->val; | ||
129 | action->type = FILTER_ACTION_RQ_STEERING; | ||
130 | action->u.rq_idx = rq_idx; | ||
131 | |||
132 | status = usnic_fwd_devcmd(ufdev, vnic_idx, CMD_ADD_FILTER, &a0, &a1); | ||
133 | pci_free_consistent(pdev, tlv_size, tlv_va, tlv_pa); | ||
134 | if (status) { | ||
135 | usnic_err("VF %s Filter add failed with status:%d", | ||
136 | pci_name(pdev), | ||
137 | status); | ||
138 | kfree(usnic_filter_hndl); | ||
139 | return status; | ||
140 | } else { | ||
141 | usnic_dbg("VF %s FILTER ID:%u", | ||
142 | pci_name(pdev), | ||
143 | (u32)a0); | ||
144 | } | ||
145 | |||
146 | usnic_filter_hndl->type = FILTER_USNIC_ID; | ||
147 | usnic_filter_hndl->id = (u32)a0; | ||
148 | usnic_filter_hndl->vnic_idx = vnic_idx; | ||
149 | usnic_filter_hndl->ufdev = ufdev; | ||
150 | usnic_filter_hndl->filter = fwd_filter; | ||
151 | *filter_hndl = usnic_filter_hndl; | ||
152 | |||
153 | return status; | ||
154 | } | ||
155 | |||
156 | int usnic_fwd_del_filter(struct usnic_fwd_filter_hndl *filter_hndl) | ||
157 | { | ||
158 | int status; | ||
159 | u64 a0, a1; | ||
160 | struct net_device *netdev; | ||
161 | |||
162 | netdev = filter_hndl->ufdev->netdev; | ||
163 | a0 = filter_hndl->id; | ||
164 | |||
165 | status = usnic_fwd_devcmd(filter_hndl->ufdev, filter_hndl->vnic_idx, | ||
166 | CMD_DEL_FILTER, &a0, &a1); | ||
167 | if (status) { | ||
168 | if (status == ERR_EINVAL) { | ||
169 | usnic_dbg("Filter %u already deleted for VF Idx %u pf: %s status: %d", | ||
170 | filter_hndl->id, filter_hndl->vnic_idx, | ||
171 | netdev_name(netdev), status); | ||
172 | status = 0; | ||
173 | kfree(filter_hndl); | ||
174 | } else { | ||
175 | usnic_err("PF %s VF Idx %u Filter: %u FILTER DELETE failed with status %d", | ||
176 | netdev_name(netdev), | ||
177 | filter_hndl->vnic_idx, filter_hndl->id, | ||
178 | status); | ||
179 | } | ||
180 | } else { | ||
181 | usnic_dbg("PF %s VF Idx %u Filter: %u FILTER DELETED", | ||
182 | netdev_name(netdev), filter_hndl->vnic_idx, | ||
183 | filter_hndl->id); | ||
184 | kfree(filter_hndl); | ||
185 | } | ||
186 | |||
187 | return status; | ||
188 | } | ||
189 | |||
190 | int usnic_fwd_enable_rq(struct usnic_fwd_dev *ufdev, int vnic_idx, int rq_idx) | ||
191 | { | ||
192 | int status; | ||
193 | struct net_device *pf_netdev; | ||
194 | u64 a0, a1; | ||
195 | |||
196 | pf_netdev = ufdev->netdev; | ||
197 | a0 = rq_idx; | ||
198 | a1 = CMD_QP_RQWQ; | ||
199 | |||
200 | status = usnic_fwd_devcmd(ufdev, vnic_idx, CMD_QP_ENABLE, &a0, &a1); | ||
201 | |||
202 | if (status) { | ||
203 | usnic_err("PF %s VNIC Index %u RQ Index: %u ENABLE Failed with status %d", | ||
204 | netdev_name(pf_netdev), | ||
205 | vnic_idx, | ||
206 | rq_idx, | ||
207 | status); | ||
208 | } else { | ||
209 | usnic_dbg("PF %s VNIC Index %u RQ Index: %u ENABLED", | ||
210 | netdev_name(pf_netdev), | ||
211 | vnic_idx, rq_idx); | ||
212 | } | ||
213 | |||
214 | return status; | ||
215 | } | ||
216 | |||
217 | int usnic_fwd_disable_rq(struct usnic_fwd_dev *ufdev, int vnic_idx, int rq_idx) | ||
218 | { | ||
219 | int status; | ||
220 | u64 a0, a1; | ||
221 | struct net_device *pf_netdev; | ||
222 | |||
223 | pf_netdev = ufdev->netdev; | ||
224 | a0 = rq_idx; | ||
225 | a1 = CMD_QP_RQWQ; | ||
226 | |||
227 | status = usnic_fwd_devcmd(ufdev, vnic_idx, CMD_QP_DISABLE, &a0, &a1); | ||
228 | |||
229 | if (status) { | ||
230 | usnic_err("PF %s VNIC Index %u RQ Index: %u DISABLE Failed with status %d", | ||
231 | netdev_name(pf_netdev), | ||
232 | vnic_idx, | ||
233 | rq_idx, | ||
234 | status); | ||
235 | } else { | ||
236 | usnic_dbg("PF %s VNIC Index %u RQ Index: %u DISABLED", | ||
237 | netdev_name(pf_netdev), | ||
238 | vnic_idx, | ||
239 | rq_idx); | ||
240 | } | ||
241 | |||
242 | return status; | ||
243 | } | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_fwd.h b/drivers/infiniband/hw/usnic/usnic_fwd.h new file mode 100644 index 000000000000..6973901da8af --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_fwd.h | |||
@@ -0,0 +1,58 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_FWD_H_ | ||
20 | #define USNIC_FWD_H_ | ||
21 | |||
22 | #include <linux/if.h> | ||
23 | #include <linux/pci.h> | ||
24 | #include <linux/spinlock.h> | ||
25 | |||
26 | #include "usnic_abi.h" | ||
27 | #include "vnic_devcmd.h" | ||
28 | |||
29 | struct usnic_fwd_dev { | ||
30 | struct pci_dev *pdev; | ||
31 | struct net_device *netdev; | ||
32 | spinlock_t lock; | ||
33 | }; | ||
34 | |||
35 | struct usnic_fwd_filter { | ||
36 | enum usnic_transport_type transport; | ||
37 | u16 port_num; | ||
38 | }; | ||
39 | |||
40 | struct usnic_fwd_filter_hndl { | ||
41 | enum filter_type type; | ||
42 | u32 id; | ||
43 | u32 vnic_idx; | ||
44 | struct usnic_fwd_dev *ufdev; | ||
45 | struct list_head link; | ||
46 | struct usnic_fwd_filter *filter; | ||
47 | }; | ||
48 | |||
49 | struct usnic_fwd_dev *usnic_fwd_dev_alloc(struct pci_dev *pdev); | ||
50 | void usnic_fwd_dev_free(struct usnic_fwd_dev *ufdev); | ||
51 | int usnic_fwd_add_usnic_filter(struct usnic_fwd_dev *ufdev, int vnic_idx, | ||
52 | int rq_idx, struct usnic_fwd_filter *filter, | ||
53 | struct usnic_fwd_filter_hndl **filter_hndl); | ||
54 | int usnic_fwd_del_filter(struct usnic_fwd_filter_hndl *filter_hndl); | ||
55 | int usnic_fwd_enable_rq(struct usnic_fwd_dev *ufdev, int vnic_idx, int rq_idx); | ||
56 | int usnic_fwd_disable_rq(struct usnic_fwd_dev *ufdev, int vnic_idx, int rq_idx); | ||
57 | |||
58 | #endif /* !USNIC_FWD_H_ */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_ib.h b/drivers/infiniband/hw/usnic/usnic_ib.h new file mode 100644 index 000000000000..3511c8521f30 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_ib.h | |||
@@ -0,0 +1,115 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_IB_H_ | ||
20 | #define USNIC_IB_H_ | ||
21 | |||
22 | #include <linux/iommu.h> | ||
23 | #include <linux/netdevice.h> | ||
24 | #include <linux/version.h> | ||
25 | |||
26 | #include <rdma/ib_verbs.h> | ||
27 | |||
28 | |||
29 | #include "usnic.h" | ||
30 | #include "usnic_abi.h" | ||
31 | #include "usnic_vnic.h" | ||
32 | |||
33 | #define USNIC_IB_PORT_CNT 1 | ||
34 | #define USNIC_IB_NUM_COMP_VECTORS 1 | ||
35 | |||
36 | extern unsigned int usnic_ib_share_vf; | ||
37 | |||
38 | struct usnic_ib_ucontext { | ||
39 | struct ib_ucontext ibucontext; | ||
40 | /* Protected by usnic_ib_dev->usdev_lock */ | ||
41 | struct list_head qp_grp_list; | ||
42 | struct list_head link; | ||
43 | }; | ||
44 | |||
45 | struct usnic_ib_pd { | ||
46 | struct ib_pd ibpd; | ||
47 | struct usnic_uiom_pd *umem_pd; | ||
48 | }; | ||
49 | |||
50 | struct usnic_ib_mr { | ||
51 | struct ib_mr ibmr; | ||
52 | struct usnic_uiom_reg *umem; | ||
53 | }; | ||
54 | |||
55 | struct usnic_ib_dev { | ||
56 | struct ib_device ib_dev; | ||
57 | struct pci_dev *pdev; | ||
58 | struct net_device *netdev; | ||
59 | struct usnic_fwd_dev *ufdev; | ||
60 | bool link_up; | ||
61 | struct list_head ib_dev_link; | ||
62 | struct list_head vf_dev_list; | ||
63 | struct list_head ctx_list; | ||
64 | struct mutex usdev_lock; | ||
65 | char mac[ETH_ALEN]; | ||
66 | unsigned int mtu; | ||
67 | |||
68 | /* provisioning information */ | ||
69 | struct kref vf_cnt; | ||
70 | unsigned int vf_res_cnt[USNIC_VNIC_RES_TYPE_MAX]; | ||
71 | |||
72 | /* sysfs vars for QPN reporting */ | ||
73 | struct kobject *qpn_kobj; | ||
74 | }; | ||
75 | |||
76 | struct usnic_ib_vf { | ||
77 | struct usnic_ib_dev *pf; | ||
78 | spinlock_t lock; | ||
79 | struct usnic_vnic *vnic; | ||
80 | unsigned int qp_grp_ref_cnt; | ||
81 | struct usnic_ib_pd *pd; | ||
82 | struct list_head link; | ||
83 | }; | ||
84 | |||
85 | static inline | ||
86 | struct usnic_ib_dev *to_usdev(struct ib_device *ibdev) | ||
87 | { | ||
88 | return container_of(ibdev, struct usnic_ib_dev, ib_dev); | ||
89 | } | ||
90 | |||
91 | static inline | ||
92 | struct usnic_ib_ucontext *to_ucontext(struct ib_ucontext *ibucontext) | ||
93 | { | ||
94 | return container_of(ibucontext, struct usnic_ib_ucontext, ibucontext); | ||
95 | } | ||
96 | |||
97 | static inline | ||
98 | struct usnic_ib_pd *to_upd(struct ib_pd *ibpd) | ||
99 | { | ||
100 | return container_of(ibpd, struct usnic_ib_pd, ibpd); | ||
101 | } | ||
102 | |||
103 | static inline | ||
104 | struct usnic_ib_ucontext *to_uucontext(struct ib_ucontext *ibucontext) | ||
105 | { | ||
106 | return container_of(ibucontext, struct usnic_ib_ucontext, ibucontext); | ||
107 | } | ||
108 | |||
109 | static inline | ||
110 | struct usnic_ib_mr *to_umr(struct ib_mr *ibmr) | ||
111 | { | ||
112 | return container_of(ibmr, struct usnic_ib_mr, ibmr); | ||
113 | } | ||
114 | void usnic_ib_log_vf(struct usnic_ib_vf *vf); | ||
115 | #endif /* USNIC_IB_H_ */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_ib_main.c b/drivers/infiniband/hw/usnic/usnic_ib_main.c new file mode 100644 index 000000000000..dc09c12435b9 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_ib_main.c | |||
@@ -0,0 +1,598 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | * Author: Upinder Malhi <umalhi@cisco.com> | ||
18 | * Author: Anant Deepak <anadeepa@cisco.com> | ||
19 | * Author: Cesare Cantu' <cantuc@cisco.com> | ||
20 | * Author: Jeff Squyres <jsquyres@cisco.com> | ||
21 | * Author: Kiran Thirumalai <kithirum@cisco.com> | ||
22 | * Author: Xuyang Wang <xuywang@cisco.com> | ||
23 | * Author: Reese Faucette <rfaucett@cisco.com> | ||
24 | * | ||
25 | */ | ||
26 | |||
27 | #include <linux/module.h> | ||
28 | #include <linux/init.h> | ||
29 | #include <linux/slab.h> | ||
30 | #include <linux/errno.h> | ||
31 | #include <linux/pci.h> | ||
32 | #include <linux/netdevice.h> | ||
33 | |||
34 | #include <rdma/ib_user_verbs.h> | ||
35 | #include <rdma/ib_addr.h> | ||
36 | |||
37 | #include "usnic_abi.h" | ||
38 | #include "usnic_common_util.h" | ||
39 | #include "usnic_ib.h" | ||
40 | #include "usnic_ib_qp_grp.h" | ||
41 | #include "usnic_log.h" | ||
42 | #include "usnic_fwd.h" | ||
43 | #include "usnic_debugfs.h" | ||
44 | #include "usnic_ib_verbs.h" | ||
45 | #include "usnic_transport.h" | ||
46 | #include "usnic_uiom.h" | ||
47 | #include "usnic_ib_sysfs.h" | ||
48 | |||
49 | unsigned int usnic_log_lvl = USNIC_LOG_LVL_ERR; | ||
50 | unsigned int usnic_ib_share_vf = 1; | ||
51 | |||
52 | static const char usnic_version[] = | ||
53 | DRV_NAME ": Cisco VIC (USNIC) Verbs Driver v" | ||
54 | DRV_VERSION " (" DRV_RELDATE ")\n"; | ||
55 | |||
56 | static DEFINE_MUTEX(usnic_ib_ibdev_list_lock); | ||
57 | static LIST_HEAD(usnic_ib_ibdev_list); | ||
58 | |||
59 | /* Callback dump funcs */ | ||
60 | static int usnic_ib_dump_vf_hdr(void *obj, char *buf, int buf_sz) | ||
61 | { | ||
62 | struct usnic_ib_vf *vf = obj; | ||
63 | return scnprintf(buf, buf_sz, "PF: %s ", vf->pf->ib_dev.name); | ||
64 | } | ||
65 | /* End callback dump funcs */ | ||
66 | |||
67 | static void usnic_ib_dump_vf(struct usnic_ib_vf *vf, char *buf, int buf_sz) | ||
68 | { | ||
69 | usnic_vnic_dump(vf->vnic, buf, buf_sz, vf, | ||
70 | usnic_ib_dump_vf_hdr, | ||
71 | usnic_ib_qp_grp_dump_hdr, usnic_ib_qp_grp_dump_rows); | ||
72 | } | ||
73 | |||
74 | void usnic_ib_log_vf(struct usnic_ib_vf *vf) | ||
75 | { | ||
76 | char buf[1000]; | ||
77 | usnic_ib_dump_vf(vf, buf, sizeof(buf)); | ||
78 | usnic_dbg("%s\n", buf); | ||
79 | } | ||
80 | |||
81 | /* Start of netdev section */ | ||
82 | static inline const char *usnic_ib_netdev_event_to_string(unsigned long event) | ||
83 | { | ||
84 | const char *event2str[] = {"NETDEV_NONE", "NETDEV_UP", "NETDEV_DOWN", | ||
85 | "NETDEV_REBOOT", "NETDEV_CHANGE", | ||
86 | "NETDEV_REGISTER", "NETDEV_UNREGISTER", "NETDEV_CHANGEMTU", | ||
87 | "NETDEV_CHANGEADDR", "NETDEV_GOING_DOWN", "NETDEV_FEAT_CHANGE", | ||
88 | "NETDEV_BONDING_FAILOVER", "NETDEV_PRE_UP", | ||
89 | "NETDEV_PRE_TYPE_CHANGE", "NETDEV_POST_TYPE_CHANGE", | ||
90 | "NETDEV_POST_INT", "NETDEV_UNREGISTER_FINAL", "NETDEV_RELEASE", | ||
91 | "NETDEV_NOTIFY_PEERS", "NETDEV_JOIN" | ||
92 | }; | ||
93 | |||
94 | if (event >= ARRAY_SIZE(event2str)) | ||
95 | return "UNKNOWN_NETDEV_EVENT"; | ||
96 | else | ||
97 | return event2str[event]; | ||
98 | } | ||
99 | |||
100 | static void usnic_ib_qp_grp_modify_active_to_err(struct usnic_ib_dev *us_ibdev) | ||
101 | { | ||
102 | struct usnic_ib_ucontext *ctx; | ||
103 | struct usnic_ib_qp_grp *qp_grp; | ||
104 | enum ib_qp_state cur_state; | ||
105 | int status; | ||
106 | |||
107 | BUG_ON(!mutex_is_locked(&us_ibdev->usdev_lock)); | ||
108 | |||
109 | list_for_each_entry(ctx, &us_ibdev->ctx_list, link) { | ||
110 | list_for_each_entry(qp_grp, &ctx->qp_grp_list, link) { | ||
111 | cur_state = qp_grp->state; | ||
112 | if (cur_state == IB_QPS_INIT || | ||
113 | cur_state == IB_QPS_RTR || | ||
114 | cur_state == IB_QPS_RTS) { | ||
115 | status = usnic_ib_qp_grp_modify(qp_grp, | ||
116 | IB_QPS_ERR, | ||
117 | NULL); | ||
118 | if (status) { | ||
119 | usnic_err("Failed to transistion qp grp %u from %s to %s\n", | ||
120 | qp_grp->grp_id, | ||
121 | usnic_ib_qp_grp_state_to_string | ||
122 | (cur_state), | ||
123 | usnic_ib_qp_grp_state_to_string | ||
124 | (IB_QPS_ERR)); | ||
125 | } | ||
126 | } | ||
127 | } | ||
128 | } | ||
129 | } | ||
130 | |||
131 | static void usnic_ib_handle_usdev_event(struct usnic_ib_dev *us_ibdev, | ||
132 | unsigned long event) | ||
133 | { | ||
134 | struct net_device *netdev; | ||
135 | struct ib_event ib_event; | ||
136 | |||
137 | memset(&ib_event, 0, sizeof(ib_event)); | ||
138 | |||
139 | mutex_lock(&us_ibdev->usdev_lock); | ||
140 | netdev = us_ibdev->netdev; | ||
141 | switch (event) { | ||
142 | case NETDEV_REBOOT: | ||
143 | usnic_info("PF Reset on %s\n", us_ibdev->ib_dev.name); | ||
144 | usnic_ib_qp_grp_modify_active_to_err(us_ibdev); | ||
145 | ib_event.event = IB_EVENT_PORT_ERR; | ||
146 | ib_event.device = &us_ibdev->ib_dev; | ||
147 | ib_event.element.port_num = 1; | ||
148 | ib_dispatch_event(&ib_event); | ||
149 | break; | ||
150 | case NETDEV_UP: | ||
151 | case NETDEV_DOWN: | ||
152 | case NETDEV_CHANGE: | ||
153 | if (!us_ibdev->link_up && netif_carrier_ok(netdev)) { | ||
154 | us_ibdev->link_up = true; | ||
155 | usnic_info("Link UP on %s\n", us_ibdev->ib_dev.name); | ||
156 | ib_event.event = IB_EVENT_PORT_ACTIVE; | ||
157 | ib_event.device = &us_ibdev->ib_dev; | ||
158 | ib_event.element.port_num = 1; | ||
159 | ib_dispatch_event(&ib_event); | ||
160 | } else if (us_ibdev->link_up && !netif_carrier_ok(netdev)) { | ||
161 | us_ibdev->link_up = false; | ||
162 | usnic_info("Link DOWN on %s\n", us_ibdev->ib_dev.name); | ||
163 | usnic_ib_qp_grp_modify_active_to_err(us_ibdev); | ||
164 | ib_event.event = IB_EVENT_PORT_ERR; | ||
165 | ib_event.device = &us_ibdev->ib_dev; | ||
166 | ib_event.element.port_num = 1; | ||
167 | ib_dispatch_event(&ib_event); | ||
168 | } else { | ||
169 | usnic_dbg("Ignorning %s on %s\n", | ||
170 | usnic_ib_netdev_event_to_string(event), | ||
171 | us_ibdev->ib_dev.name); | ||
172 | } | ||
173 | break; | ||
174 | case NETDEV_CHANGEADDR: | ||
175 | if (!memcmp(us_ibdev->mac, netdev->dev_addr, | ||
176 | sizeof(us_ibdev->mac))) { | ||
177 | usnic_dbg("Ignorning addr change on %s\n", | ||
178 | us_ibdev->ib_dev.name); | ||
179 | } else { | ||
180 | usnic_info(" %s old mac: %pM new mac: %pM\n", | ||
181 | us_ibdev->ib_dev.name, | ||
182 | us_ibdev->mac, | ||
183 | netdev->dev_addr); | ||
184 | memcpy(us_ibdev->mac, netdev->dev_addr, | ||
185 | sizeof(us_ibdev->mac)); | ||
186 | usnic_ib_qp_grp_modify_active_to_err(us_ibdev); | ||
187 | ib_event.event = IB_EVENT_GID_CHANGE; | ||
188 | ib_event.device = &us_ibdev->ib_dev; | ||
189 | ib_event.element.port_num = 1; | ||
190 | ib_dispatch_event(&ib_event); | ||
191 | } | ||
192 | |||
193 | break; | ||
194 | case NETDEV_CHANGEMTU: | ||
195 | if (us_ibdev->mtu != netdev->mtu) { | ||
196 | usnic_info("MTU Change on %s old: %u new: %u\n", | ||
197 | us_ibdev->ib_dev.name, | ||
198 | us_ibdev->mtu, netdev->mtu); | ||
199 | us_ibdev->mtu = netdev->mtu; | ||
200 | usnic_ib_qp_grp_modify_active_to_err(us_ibdev); | ||
201 | } else { | ||
202 | usnic_dbg("Ignoring MTU change on %s\n", | ||
203 | us_ibdev->ib_dev.name); | ||
204 | } | ||
205 | break; | ||
206 | default: | ||
207 | usnic_dbg("Ignorning event %s on %s", | ||
208 | usnic_ib_netdev_event_to_string(event), | ||
209 | us_ibdev->ib_dev.name); | ||
210 | } | ||
211 | mutex_unlock(&us_ibdev->usdev_lock); | ||
212 | } | ||
213 | |||
214 | static int usnic_ib_netdevice_event(struct notifier_block *notifier, | ||
215 | unsigned long event, void *ptr) | ||
216 | { | ||
217 | struct usnic_ib_dev *us_ibdev; | ||
218 | |||
219 | struct net_device *netdev = netdev_notifier_info_to_dev(ptr); | ||
220 | |||
221 | mutex_lock(&usnic_ib_ibdev_list_lock); | ||
222 | list_for_each_entry(us_ibdev, &usnic_ib_ibdev_list, ib_dev_link) { | ||
223 | if (us_ibdev->netdev == netdev) { | ||
224 | usnic_ib_handle_usdev_event(us_ibdev, event); | ||
225 | break; | ||
226 | } | ||
227 | } | ||
228 | mutex_unlock(&usnic_ib_ibdev_list_lock); | ||
229 | |||
230 | return NOTIFY_DONE; | ||
231 | } | ||
232 | |||
233 | static struct notifier_block usnic_ib_netdevice_notifier = { | ||
234 | .notifier_call = usnic_ib_netdevice_event | ||
235 | }; | ||
236 | /* End of netdev section */ | ||
237 | |||
238 | /* Start of PF discovery section */ | ||
239 | static void *usnic_ib_device_add(struct pci_dev *dev) | ||
240 | { | ||
241 | struct usnic_ib_dev *us_ibdev; | ||
242 | union ib_gid gid; | ||
243 | |||
244 | usnic_dbg("\n"); | ||
245 | |||
246 | us_ibdev = (struct usnic_ib_dev *)ib_alloc_device(sizeof(*us_ibdev)); | ||
247 | if (IS_ERR_OR_NULL(us_ibdev)) { | ||
248 | usnic_err("Device %s context alloc failed\n", | ||
249 | netdev_name(pci_get_drvdata(dev))); | ||
250 | return ERR_PTR(us_ibdev ? PTR_ERR(us_ibdev) : -EFAULT); | ||
251 | } | ||
252 | |||
253 | us_ibdev->ufdev = usnic_fwd_dev_alloc(dev); | ||
254 | if (IS_ERR_OR_NULL(us_ibdev->ufdev)) { | ||
255 | usnic_err("Failed to alloc ufdev for %s with err %ld\n", | ||
256 | pci_name(dev), PTR_ERR(us_ibdev->ufdev)); | ||
257 | goto err_dealloc; | ||
258 | } | ||
259 | |||
260 | mutex_init(&us_ibdev->usdev_lock); | ||
261 | INIT_LIST_HEAD(&us_ibdev->vf_dev_list); | ||
262 | INIT_LIST_HEAD(&us_ibdev->ctx_list); | ||
263 | |||
264 | us_ibdev->pdev = dev; | ||
265 | us_ibdev->netdev = pci_get_drvdata(dev); | ||
266 | us_ibdev->ib_dev.owner = THIS_MODULE; | ||
267 | us_ibdev->ib_dev.node_type = RDMA_NODE_USNIC; | ||
268 | us_ibdev->ib_dev.phys_port_cnt = USNIC_IB_PORT_CNT; | ||
269 | us_ibdev->ib_dev.num_comp_vectors = USNIC_IB_NUM_COMP_VECTORS; | ||
270 | us_ibdev->ib_dev.dma_device = &dev->dev; | ||
271 | us_ibdev->ib_dev.uverbs_abi_ver = USNIC_UVERBS_ABI_VERSION; | ||
272 | strlcpy(us_ibdev->ib_dev.name, "usnic_%d", IB_DEVICE_NAME_MAX); | ||
273 | |||
274 | us_ibdev->ib_dev.uverbs_cmd_mask = | ||
275 | (1ull << IB_USER_VERBS_CMD_GET_CONTEXT) | | ||
276 | (1ull << IB_USER_VERBS_CMD_QUERY_DEVICE) | | ||
277 | (1ull << IB_USER_VERBS_CMD_QUERY_PORT) | | ||
278 | (1ull << IB_USER_VERBS_CMD_ALLOC_PD) | | ||
279 | (1ull << IB_USER_VERBS_CMD_DEALLOC_PD) | | ||
280 | (1ull << IB_USER_VERBS_CMD_REG_MR) | | ||
281 | (1ull << IB_USER_VERBS_CMD_DEREG_MR) | | ||
282 | (1ull << IB_USER_VERBS_CMD_CREATE_COMP_CHANNEL) | | ||
283 | (1ull << IB_USER_VERBS_CMD_CREATE_CQ) | | ||
284 | (1ull << IB_USER_VERBS_CMD_DESTROY_CQ) | | ||
285 | (1ull << IB_USER_VERBS_CMD_CREATE_QP) | | ||
286 | (1ull << IB_USER_VERBS_CMD_MODIFY_QP) | | ||
287 | (1ull << IB_USER_VERBS_CMD_QUERY_QP) | | ||
288 | (1ull << IB_USER_VERBS_CMD_DESTROY_QP) | | ||
289 | (1ull << IB_USER_VERBS_CMD_ATTACH_MCAST) | | ||
290 | (1ull << IB_USER_VERBS_CMD_DETACH_MCAST) | | ||
291 | (1ull << IB_USER_VERBS_CMD_OPEN_QP); | ||
292 | |||
293 | us_ibdev->ib_dev.query_device = usnic_ib_query_device; | ||
294 | us_ibdev->ib_dev.query_port = usnic_ib_query_port; | ||
295 | us_ibdev->ib_dev.query_pkey = usnic_ib_query_pkey; | ||
296 | us_ibdev->ib_dev.query_gid = usnic_ib_query_gid; | ||
297 | us_ibdev->ib_dev.get_link_layer = usnic_ib_port_link_layer; | ||
298 | us_ibdev->ib_dev.alloc_pd = usnic_ib_alloc_pd; | ||
299 | us_ibdev->ib_dev.dealloc_pd = usnic_ib_dealloc_pd; | ||
300 | us_ibdev->ib_dev.create_qp = usnic_ib_create_qp; | ||
301 | us_ibdev->ib_dev.modify_qp = usnic_ib_modify_qp; | ||
302 | us_ibdev->ib_dev.query_qp = usnic_ib_query_qp; | ||
303 | us_ibdev->ib_dev.destroy_qp = usnic_ib_destroy_qp; | ||
304 | us_ibdev->ib_dev.create_cq = usnic_ib_create_cq; | ||
305 | us_ibdev->ib_dev.destroy_cq = usnic_ib_destroy_cq; | ||
306 | us_ibdev->ib_dev.reg_user_mr = usnic_ib_reg_mr; | ||
307 | us_ibdev->ib_dev.dereg_mr = usnic_ib_dereg_mr; | ||
308 | us_ibdev->ib_dev.alloc_ucontext = usnic_ib_alloc_ucontext; | ||
309 | us_ibdev->ib_dev.dealloc_ucontext = usnic_ib_dealloc_ucontext; | ||
310 | us_ibdev->ib_dev.mmap = usnic_ib_mmap; | ||
311 | us_ibdev->ib_dev.create_ah = usnic_ib_create_ah; | ||
312 | us_ibdev->ib_dev.destroy_ah = usnic_ib_destroy_ah; | ||
313 | us_ibdev->ib_dev.post_send = usnic_ib_post_send; | ||
314 | us_ibdev->ib_dev.post_recv = usnic_ib_post_recv; | ||
315 | us_ibdev->ib_dev.poll_cq = usnic_ib_poll_cq; | ||
316 | us_ibdev->ib_dev.req_notify_cq = usnic_ib_req_notify_cq; | ||
317 | us_ibdev->ib_dev.get_dma_mr = usnic_ib_get_dma_mr; | ||
318 | |||
319 | |||
320 | if (ib_register_device(&us_ibdev->ib_dev, NULL)) | ||
321 | goto err_fwd_dealloc; | ||
322 | |||
323 | us_ibdev->link_up = netif_carrier_ok(us_ibdev->netdev); | ||
324 | us_ibdev->mtu = us_ibdev->netdev->mtu; | ||
325 | memcpy(&us_ibdev->mac, us_ibdev->netdev->dev_addr, | ||
326 | sizeof(us_ibdev->mac)); | ||
327 | usnic_mac_to_gid(us_ibdev->netdev->perm_addr, &gid.raw[0]); | ||
328 | memcpy(&us_ibdev->ib_dev.node_guid, &gid.global.interface_id, | ||
329 | sizeof(gid.global.interface_id)); | ||
330 | kref_init(&us_ibdev->vf_cnt); | ||
331 | |||
332 | usnic_info("Added ibdev: %s netdev: %s with mac %pM Link: %u MTU: %u\n", | ||
333 | us_ibdev->ib_dev.name, netdev_name(us_ibdev->netdev), | ||
334 | us_ibdev->mac, us_ibdev->link_up, us_ibdev->mtu); | ||
335 | return us_ibdev; | ||
336 | |||
337 | err_fwd_dealloc: | ||
338 | usnic_fwd_dev_free(us_ibdev->ufdev); | ||
339 | err_dealloc: | ||
340 | usnic_err("failed -- deallocing device\n"); | ||
341 | ib_dealloc_device(&us_ibdev->ib_dev); | ||
342 | return NULL; | ||
343 | } | ||
344 | |||
345 | static void usnic_ib_device_remove(struct usnic_ib_dev *us_ibdev) | ||
346 | { | ||
347 | usnic_info("Unregistering %s\n", us_ibdev->ib_dev.name); | ||
348 | usnic_ib_sysfs_unregister_usdev(us_ibdev); | ||
349 | usnic_fwd_dev_free(us_ibdev->ufdev); | ||
350 | ib_unregister_device(&us_ibdev->ib_dev); | ||
351 | ib_dealloc_device(&us_ibdev->ib_dev); | ||
352 | } | ||
353 | |||
354 | static void usnic_ib_undiscover_pf(struct kref *kref) | ||
355 | { | ||
356 | struct usnic_ib_dev *us_ibdev, *tmp; | ||
357 | struct pci_dev *dev; | ||
358 | bool found = false; | ||
359 | |||
360 | dev = container_of(kref, struct usnic_ib_dev, vf_cnt)->pdev; | ||
361 | mutex_lock(&usnic_ib_ibdev_list_lock); | ||
362 | list_for_each_entry_safe(us_ibdev, tmp, | ||
363 | &usnic_ib_ibdev_list, ib_dev_link) { | ||
364 | if (us_ibdev->pdev == dev) { | ||
365 | list_del(&us_ibdev->ib_dev_link); | ||
366 | usnic_ib_device_remove(us_ibdev); | ||
367 | found = true; | ||
368 | break; | ||
369 | } | ||
370 | } | ||
371 | |||
372 | WARN(!found, "Failed to remove PF %s\n", pci_name(dev)); | ||
373 | |||
374 | mutex_unlock(&usnic_ib_ibdev_list_lock); | ||
375 | } | ||
376 | |||
377 | static struct usnic_ib_dev *usnic_ib_discover_pf(struct usnic_vnic *vnic) | ||
378 | { | ||
379 | struct usnic_ib_dev *us_ibdev; | ||
380 | struct pci_dev *parent_pci, *vf_pci; | ||
381 | int err; | ||
382 | |||
383 | vf_pci = usnic_vnic_get_pdev(vnic); | ||
384 | parent_pci = pci_physfn(vf_pci); | ||
385 | |||
386 | BUG_ON(!parent_pci); | ||
387 | |||
388 | mutex_lock(&usnic_ib_ibdev_list_lock); | ||
389 | list_for_each_entry(us_ibdev, &usnic_ib_ibdev_list, ib_dev_link) { | ||
390 | if (us_ibdev->pdev == parent_pci) { | ||
391 | kref_get(&us_ibdev->vf_cnt); | ||
392 | goto out; | ||
393 | } | ||
394 | } | ||
395 | |||
396 | us_ibdev = usnic_ib_device_add(parent_pci); | ||
397 | if (IS_ERR_OR_NULL(us_ibdev)) { | ||
398 | us_ibdev = (us_ibdev) ? us_ibdev : ERR_PTR(-EFAULT); | ||
399 | goto out; | ||
400 | } | ||
401 | |||
402 | err = usnic_ib_sysfs_register_usdev(us_ibdev); | ||
403 | if (err) { | ||
404 | usnic_ib_device_remove(us_ibdev); | ||
405 | us_ibdev = ERR_PTR(err); | ||
406 | goto out; | ||
407 | } | ||
408 | |||
409 | list_add(&us_ibdev->ib_dev_link, &usnic_ib_ibdev_list); | ||
410 | out: | ||
411 | mutex_unlock(&usnic_ib_ibdev_list_lock); | ||
412 | return us_ibdev; | ||
413 | } | ||
414 | /* End of PF discovery section */ | ||
415 | |||
416 | /* Start of PCI section */ | ||
417 | |||
418 | static DEFINE_PCI_DEVICE_TABLE(usnic_ib_pci_ids) = { | ||
419 | {PCI_DEVICE(PCI_VENDOR_ID_CISCO, PCI_DEVICE_ID_CISCO_VIC_USPACE_NIC)}, | ||
420 | {0,} | ||
421 | }; | ||
422 | |||
423 | static int usnic_ib_pci_probe(struct pci_dev *pdev, | ||
424 | const struct pci_device_id *id) | ||
425 | { | ||
426 | int err; | ||
427 | struct usnic_ib_dev *pf; | ||
428 | struct usnic_ib_vf *vf; | ||
429 | enum usnic_vnic_res_type res_type; | ||
430 | |||
431 | vf = kzalloc(sizeof(*vf), GFP_KERNEL); | ||
432 | if (!vf) | ||
433 | return -ENOMEM; | ||
434 | |||
435 | err = pci_enable_device(pdev); | ||
436 | if (err) { | ||
437 | usnic_err("Failed to enable %s with err %d\n", | ||
438 | pci_name(pdev), err); | ||
439 | goto out_clean_vf; | ||
440 | } | ||
441 | |||
442 | err = pci_request_regions(pdev, DRV_NAME); | ||
443 | if (err) { | ||
444 | usnic_err("Failed to request region for %s with err %d\n", | ||
445 | pci_name(pdev), err); | ||
446 | goto out_disable_device; | ||
447 | } | ||
448 | |||
449 | pci_set_master(pdev); | ||
450 | pci_set_drvdata(pdev, vf); | ||
451 | |||
452 | vf->vnic = usnic_vnic_alloc(pdev); | ||
453 | if (IS_ERR_OR_NULL(vf->vnic)) { | ||
454 | err = (vf->vnic ? PTR_ERR(vf->vnic) : -ENOMEM); | ||
455 | usnic_err("Failed to alloc vnic for %s with err %d\n", | ||
456 | pci_name(pdev), err); | ||
457 | goto out_release_regions; | ||
458 | } | ||
459 | |||
460 | pf = usnic_ib_discover_pf(vf->vnic); | ||
461 | if (IS_ERR_OR_NULL(pf)) { | ||
462 | usnic_err("Failed to discover pf of vnic %s with err%ld\n", | ||
463 | pci_name(pdev), PTR_ERR(pf)); | ||
464 | err = (pf ? PTR_ERR(pf) : -EFAULT); | ||
465 | goto out_clean_vnic; | ||
466 | } | ||
467 | |||
468 | vf->pf = pf; | ||
469 | spin_lock_init(&vf->lock); | ||
470 | mutex_lock(&pf->usdev_lock); | ||
471 | list_add_tail(&vf->link, &pf->vf_dev_list); | ||
472 | /* | ||
473 | * Save max settings (will be same for each VF, easier to re-write than | ||
474 | * to say "if (!set) { set_values(); set=1; } | ||
475 | */ | ||
476 | for (res_type = USNIC_VNIC_RES_TYPE_EOL+1; | ||
477 | res_type < USNIC_VNIC_RES_TYPE_MAX; | ||
478 | res_type++) { | ||
479 | pf->vf_res_cnt[res_type] = usnic_vnic_res_cnt(vf->vnic, | ||
480 | res_type); | ||
481 | } | ||
482 | |||
483 | mutex_unlock(&pf->usdev_lock); | ||
484 | |||
485 | usnic_info("Registering usnic VF %s into PF %s\n", pci_name(pdev), | ||
486 | pf->ib_dev.name); | ||
487 | usnic_ib_log_vf(vf); | ||
488 | return 0; | ||
489 | |||
490 | out_clean_vnic: | ||
491 | usnic_vnic_free(vf->vnic); | ||
492 | out_release_regions: | ||
493 | pci_set_drvdata(pdev, NULL); | ||
494 | pci_clear_master(pdev); | ||
495 | pci_release_regions(pdev); | ||
496 | out_disable_device: | ||
497 | pci_disable_device(pdev); | ||
498 | out_clean_vf: | ||
499 | kfree(vf); | ||
500 | return err; | ||
501 | } | ||
502 | |||
503 | static void usnic_ib_pci_remove(struct pci_dev *pdev) | ||
504 | { | ||
505 | struct usnic_ib_vf *vf = pci_get_drvdata(pdev); | ||
506 | struct usnic_ib_dev *pf = vf->pf; | ||
507 | |||
508 | mutex_lock(&pf->usdev_lock); | ||
509 | list_del(&vf->link); | ||
510 | mutex_unlock(&pf->usdev_lock); | ||
511 | |||
512 | kref_put(&pf->vf_cnt, usnic_ib_undiscover_pf); | ||
513 | usnic_vnic_free(vf->vnic); | ||
514 | pci_set_drvdata(pdev, NULL); | ||
515 | pci_clear_master(pdev); | ||
516 | pci_release_regions(pdev); | ||
517 | pci_disable_device(pdev); | ||
518 | kfree(vf); | ||
519 | |||
520 | usnic_info("Removed VF %s\n", pci_name(pdev)); | ||
521 | } | ||
522 | |||
523 | /* PCI driver entry points */ | ||
524 | static struct pci_driver usnic_ib_pci_driver = { | ||
525 | .name = DRV_NAME, | ||
526 | .id_table = usnic_ib_pci_ids, | ||
527 | .probe = usnic_ib_pci_probe, | ||
528 | .remove = usnic_ib_pci_remove, | ||
529 | }; | ||
530 | /* End of PCI section */ | ||
531 | |||
532 | /* Start of module section */ | ||
533 | static int __init usnic_ib_init(void) | ||
534 | { | ||
535 | int err; | ||
536 | |||
537 | printk_once(KERN_INFO "%s", usnic_version); | ||
538 | |||
539 | err = usnic_uiom_init(DRV_NAME); | ||
540 | if (err) { | ||
541 | usnic_err("Unable to initalize umem with err %d\n", err); | ||
542 | return err; | ||
543 | } | ||
544 | |||
545 | if (pci_register_driver(&usnic_ib_pci_driver)) { | ||
546 | usnic_err("Unable to register with PCI\n"); | ||
547 | goto out_umem_fini; | ||
548 | } | ||
549 | |||
550 | err = register_netdevice_notifier(&usnic_ib_netdevice_notifier); | ||
551 | if (err) { | ||
552 | usnic_err("Failed to register netdev notifier\n"); | ||
553 | goto out_pci_unreg; | ||
554 | } | ||
555 | |||
556 | err = usnic_transport_init(); | ||
557 | if (err) { | ||
558 | usnic_err("Failed to initialize transport\n"); | ||
559 | goto out_unreg_netdev_notifier; | ||
560 | } | ||
561 | |||
562 | usnic_debugfs_init(); | ||
563 | |||
564 | return 0; | ||
565 | |||
566 | out_unreg_netdev_notifier: | ||
567 | unregister_netdevice_notifier(&usnic_ib_netdevice_notifier); | ||
568 | out_pci_unreg: | ||
569 | pci_unregister_driver(&usnic_ib_pci_driver); | ||
570 | out_umem_fini: | ||
571 | usnic_uiom_fini(); | ||
572 | |||
573 | return err; | ||
574 | } | ||
575 | |||
576 | static void __exit usnic_ib_destroy(void) | ||
577 | { | ||
578 | usnic_dbg("\n"); | ||
579 | usnic_debugfs_exit(); | ||
580 | usnic_transport_fini(); | ||
581 | unregister_netdevice_notifier(&usnic_ib_netdevice_notifier); | ||
582 | pci_unregister_driver(&usnic_ib_pci_driver); | ||
583 | usnic_uiom_fini(); | ||
584 | } | ||
585 | |||
586 | MODULE_DESCRIPTION("Cisco VIC (usNIC) Verbs Driver"); | ||
587 | MODULE_AUTHOR("Upinder Malhi <umalhi@cisco.com>"); | ||
588 | MODULE_LICENSE("Dual BSD/GPL"); | ||
589 | MODULE_VERSION(DRV_VERSION); | ||
590 | module_param(usnic_log_lvl, uint, S_IRUGO | S_IWUSR); | ||
591 | module_param(usnic_ib_share_vf, uint, S_IRUGO | S_IWUSR); | ||
592 | MODULE_PARM_DESC(usnic_log_lvl, " Off=0, Err=1, Info=2, Debug=3"); | ||
593 | MODULE_PARM_DESC(usnic_ib_share_vf, "Off=0, On=1 VF sharing amongst QPs"); | ||
594 | MODULE_DEVICE_TABLE(pci, usnic_ib_pci_ids); | ||
595 | |||
596 | module_init(usnic_ib_init); | ||
597 | module_exit(usnic_ib_destroy); | ||
598 | /* End of module section */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_ib_qp_grp.c b/drivers/infiniband/hw/usnic/usnic_ib_qp_grp.c new file mode 100644 index 000000000000..ca5fa6ad59ac --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_ib_qp_grp.c | |||
@@ -0,0 +1,541 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | #include <linux/errno.h> | ||
19 | #include <linux/module.h> | ||
20 | #include <linux/spinlock.h> | ||
21 | |||
22 | #include "usnic_log.h" | ||
23 | #include "usnic_vnic.h" | ||
24 | #include "usnic_fwd.h" | ||
25 | #include "usnic_uiom.h" | ||
26 | #include "usnic_ib_qp_grp.h" | ||
27 | #include "usnic_ib_sysfs.h" | ||
28 | #include "usnic_transport.h" | ||
29 | |||
30 | const char *usnic_ib_qp_grp_state_to_string(enum ib_qp_state state) | ||
31 | { | ||
32 | switch (state) { | ||
33 | case IB_QPS_RESET: | ||
34 | return "Rst"; | ||
35 | case IB_QPS_INIT: | ||
36 | return "Init"; | ||
37 | case IB_QPS_RTR: | ||
38 | return "RTR"; | ||
39 | case IB_QPS_RTS: | ||
40 | return "RTS"; | ||
41 | case IB_QPS_SQD: | ||
42 | return "SQD"; | ||
43 | case IB_QPS_SQE: | ||
44 | return "SQE"; | ||
45 | case IB_QPS_ERR: | ||
46 | return "ERR"; | ||
47 | default: | ||
48 | return "UNKOWN STATE"; | ||
49 | |||
50 | } | ||
51 | } | ||
52 | |||
53 | int usnic_ib_qp_grp_dump_hdr(char *buf, int buf_sz) | ||
54 | { | ||
55 | return scnprintf(buf, buf_sz, "|QPN\t|State\t|PID\t|VF Idx\t|Fil ID"); | ||
56 | } | ||
57 | |||
58 | int usnic_ib_qp_grp_dump_rows(void *obj, char *buf, int buf_sz) | ||
59 | { | ||
60 | struct usnic_ib_qp_grp *qp_grp = obj; | ||
61 | struct usnic_fwd_filter_hndl *default_filter_hndl; | ||
62 | if (obj) { | ||
63 | default_filter_hndl = list_first_entry(&qp_grp->filter_hndls, | ||
64 | struct usnic_fwd_filter_hndl, link); | ||
65 | return scnprintf(buf, buf_sz, "|%d\t|%s\t|%d\t|%hu\t|%d", | ||
66 | qp_grp->ibqp.qp_num, | ||
67 | usnic_ib_qp_grp_state_to_string( | ||
68 | qp_grp->state), | ||
69 | qp_grp->owner_pid, | ||
70 | usnic_vnic_get_index(qp_grp->vf->vnic), | ||
71 | default_filter_hndl->id); | ||
72 | } else { | ||
73 | return scnprintf(buf, buf_sz, "|N/A\t|N/A\t|N/A\t|N/A\t|N/A"); | ||
74 | } | ||
75 | } | ||
76 | |||
77 | static int add_fwd_filter(struct usnic_ib_qp_grp *qp_grp, | ||
78 | struct usnic_fwd_filter *fwd_filter) | ||
79 | { | ||
80 | struct usnic_fwd_filter_hndl *filter_hndl; | ||
81 | int status; | ||
82 | struct usnic_vnic_res_chunk *chunk; | ||
83 | int rq_idx; | ||
84 | |||
85 | WARN_ON(!spin_is_locked(&qp_grp->lock)); | ||
86 | |||
87 | chunk = usnic_ib_qp_grp_get_chunk(qp_grp, USNIC_VNIC_RES_TYPE_RQ); | ||
88 | if (IS_ERR_OR_NULL(chunk) || chunk->cnt < 1) { | ||
89 | usnic_err("Failed to get RQ info for qp_grp %u\n", | ||
90 | qp_grp->grp_id); | ||
91 | return -EFAULT; | ||
92 | } | ||
93 | |||
94 | rq_idx = chunk->res[0]->vnic_idx; | ||
95 | |||
96 | switch (qp_grp->transport) { | ||
97 | case USNIC_TRANSPORT_ROCE_CUSTOM: | ||
98 | status = usnic_fwd_add_usnic_filter(qp_grp->ufdev, | ||
99 | usnic_vnic_get_index(qp_grp->vf->vnic), | ||
100 | rq_idx, | ||
101 | fwd_filter, | ||
102 | &filter_hndl); | ||
103 | break; | ||
104 | default: | ||
105 | usnic_err("Unable to install filter for qp_grp %u for transport %d", | ||
106 | qp_grp->grp_id, qp_grp->transport); | ||
107 | status = -EINVAL; | ||
108 | } | ||
109 | |||
110 | if (status) | ||
111 | return status; | ||
112 | |||
113 | list_add_tail(&filter_hndl->link, &qp_grp->filter_hndls); | ||
114 | return 0; | ||
115 | } | ||
116 | |||
117 | static int del_all_filters(struct usnic_ib_qp_grp *qp_grp) | ||
118 | { | ||
119 | int err, status; | ||
120 | struct usnic_fwd_filter_hndl *filter_hndl, *tmp; | ||
121 | |||
122 | WARN_ON(!spin_is_locked(&qp_grp->lock)); | ||
123 | |||
124 | status = 0; | ||
125 | |||
126 | list_for_each_entry_safe(filter_hndl, tmp, | ||
127 | &qp_grp->filter_hndls, link) { | ||
128 | list_del(&filter_hndl->link); | ||
129 | err = usnic_fwd_del_filter(filter_hndl); | ||
130 | if (err) { | ||
131 | usnic_err("Failed to delete filter %u of qp_grp %d\n", | ||
132 | filter_hndl->id, qp_grp->grp_id); | ||
133 | } | ||
134 | status |= err; | ||
135 | } | ||
136 | |||
137 | return status; | ||
138 | } | ||
139 | |||
140 | static int enable_qp_grp(struct usnic_ib_qp_grp *qp_grp) | ||
141 | { | ||
142 | |||
143 | int status; | ||
144 | int i, vnic_idx; | ||
145 | struct usnic_vnic_res_chunk *res_chunk; | ||
146 | struct usnic_vnic_res *res; | ||
147 | |||
148 | WARN_ON(!spin_is_locked(&qp_grp->lock)); | ||
149 | |||
150 | vnic_idx = usnic_vnic_get_index(qp_grp->vf->vnic); | ||
151 | |||
152 | res_chunk = usnic_ib_qp_grp_get_chunk(qp_grp, USNIC_VNIC_RES_TYPE_RQ); | ||
153 | if (IS_ERR_OR_NULL(res_chunk)) { | ||
154 | usnic_err("Unable to get %s with err %ld\n", | ||
155 | usnic_vnic_res_type_to_str(USNIC_VNIC_RES_TYPE_RQ), | ||
156 | PTR_ERR(res_chunk)); | ||
157 | return res_chunk ? PTR_ERR(res_chunk) : -ENOMEM; | ||
158 | } | ||
159 | |||
160 | for (i = 0; i < res_chunk->cnt; i++) { | ||
161 | res = res_chunk->res[i]; | ||
162 | status = usnic_fwd_enable_rq(qp_grp->ufdev, vnic_idx, | ||
163 | res->vnic_idx); | ||
164 | if (status) { | ||
165 | usnic_err("Failed to enable rq %d of %s:%d\n with err %d\n", | ||
166 | res->vnic_idx, | ||
167 | netdev_name(qp_grp->ufdev->netdev), | ||
168 | vnic_idx, status); | ||
169 | goto out_err; | ||
170 | } | ||
171 | } | ||
172 | |||
173 | return 0; | ||
174 | |||
175 | out_err: | ||
176 | for (i--; i >= 0; i--) { | ||
177 | res = res_chunk->res[i]; | ||
178 | usnic_fwd_disable_rq(qp_grp->ufdev, vnic_idx, | ||
179 | res->vnic_idx); | ||
180 | } | ||
181 | |||
182 | return status; | ||
183 | } | ||
184 | |||
185 | static int disable_qp_grp(struct usnic_ib_qp_grp *qp_grp) | ||
186 | { | ||
187 | int i, vnic_idx; | ||
188 | struct usnic_vnic_res_chunk *res_chunk; | ||
189 | struct usnic_vnic_res *res; | ||
190 | int status = 0; | ||
191 | |||
192 | WARN_ON(!spin_is_locked(&qp_grp->lock)); | ||
193 | vnic_idx = usnic_vnic_get_index(qp_grp->vf->vnic); | ||
194 | |||
195 | res_chunk = usnic_ib_qp_grp_get_chunk(qp_grp, USNIC_VNIC_RES_TYPE_RQ); | ||
196 | if (IS_ERR_OR_NULL(res_chunk)) { | ||
197 | usnic_err("Unable to get %s with err %ld\n", | ||
198 | usnic_vnic_res_type_to_str(USNIC_VNIC_RES_TYPE_RQ), | ||
199 | PTR_ERR(res_chunk)); | ||
200 | return res_chunk ? PTR_ERR(res_chunk) : -ENOMEM; | ||
201 | } | ||
202 | |||
203 | for (i = 0; i < res_chunk->cnt; i++) { | ||
204 | res = res_chunk->res[i]; | ||
205 | status = usnic_fwd_disable_rq(qp_grp->ufdev, vnic_idx, | ||
206 | res->vnic_idx); | ||
207 | if (status) { | ||
208 | usnic_err("Failed to disable rq %d of %s:%d\n with err %d\n", | ||
209 | res->vnic_idx, | ||
210 | netdev_name(qp_grp->ufdev->netdev), | ||
211 | vnic_idx, status); | ||
212 | } | ||
213 | } | ||
214 | |||
215 | return status; | ||
216 | |||
217 | } | ||
218 | |||
219 | int usnic_ib_qp_grp_modify(struct usnic_ib_qp_grp *qp_grp, | ||
220 | enum ib_qp_state new_state, | ||
221 | struct usnic_fwd_filter *fwd_filter) | ||
222 | { | ||
223 | int status = 0; | ||
224 | int vnic_idx; | ||
225 | struct ib_event ib_event; | ||
226 | enum ib_qp_state old_state; | ||
227 | |||
228 | old_state = qp_grp->state; | ||
229 | vnic_idx = usnic_vnic_get_index(qp_grp->vf->vnic); | ||
230 | |||
231 | spin_lock(&qp_grp->lock); | ||
232 | switch (new_state) { | ||
233 | case IB_QPS_RESET: | ||
234 | switch (old_state) { | ||
235 | case IB_QPS_RESET: | ||
236 | /* NO-OP */ | ||
237 | break; | ||
238 | case IB_QPS_INIT: | ||
239 | status = del_all_filters(qp_grp); | ||
240 | break; | ||
241 | case IB_QPS_RTR: | ||
242 | case IB_QPS_RTS: | ||
243 | case IB_QPS_ERR: | ||
244 | status = disable_qp_grp(qp_grp); | ||
245 | status &= del_all_filters(qp_grp); | ||
246 | break; | ||
247 | default: | ||
248 | status = -EINVAL; | ||
249 | } | ||
250 | break; | ||
251 | case IB_QPS_INIT: | ||
252 | switch (old_state) { | ||
253 | case IB_QPS_RESET: | ||
254 | status = add_fwd_filter(qp_grp, fwd_filter); | ||
255 | break; | ||
256 | case IB_QPS_INIT: | ||
257 | status = add_fwd_filter(qp_grp, fwd_filter); | ||
258 | break; | ||
259 | case IB_QPS_RTR: | ||
260 | status = disable_qp_grp(qp_grp); | ||
261 | break; | ||
262 | case IB_QPS_RTS: | ||
263 | status = disable_qp_grp(qp_grp); | ||
264 | break; | ||
265 | default: | ||
266 | status = -EINVAL; | ||
267 | } | ||
268 | break; | ||
269 | case IB_QPS_RTR: | ||
270 | switch (old_state) { | ||
271 | case IB_QPS_INIT: | ||
272 | status = enable_qp_grp(qp_grp); | ||
273 | break; | ||
274 | default: | ||
275 | status = -EINVAL; | ||
276 | } | ||
277 | break; | ||
278 | case IB_QPS_RTS: | ||
279 | switch (old_state) { | ||
280 | case IB_QPS_RTR: | ||
281 | /* NO-OP FOR NOW */ | ||
282 | break; | ||
283 | default: | ||
284 | status = -EINVAL; | ||
285 | } | ||
286 | break; | ||
287 | case IB_QPS_ERR: | ||
288 | ib_event.device = &qp_grp->vf->pf->ib_dev; | ||
289 | ib_event.element.qp = &qp_grp->ibqp; | ||
290 | ib_event.event = IB_EVENT_QP_FATAL; | ||
291 | |||
292 | switch (old_state) { | ||
293 | case IB_QPS_RESET: | ||
294 | qp_grp->ibqp.event_handler(&ib_event, | ||
295 | qp_grp->ibqp.qp_context); | ||
296 | break; | ||
297 | case IB_QPS_INIT: | ||
298 | status = del_all_filters(qp_grp); | ||
299 | qp_grp->ibqp.event_handler(&ib_event, | ||
300 | qp_grp->ibqp.qp_context); | ||
301 | break; | ||
302 | case IB_QPS_RTR: | ||
303 | case IB_QPS_RTS: | ||
304 | status = disable_qp_grp(qp_grp); | ||
305 | status &= del_all_filters(qp_grp); | ||
306 | qp_grp->ibqp.event_handler(&ib_event, | ||
307 | qp_grp->ibqp.qp_context); | ||
308 | break; | ||
309 | default: | ||
310 | status = -EINVAL; | ||
311 | } | ||
312 | break; | ||
313 | default: | ||
314 | status = -EINVAL; | ||
315 | } | ||
316 | spin_unlock(&qp_grp->lock); | ||
317 | |||
318 | if (!status) { | ||
319 | qp_grp->state = new_state; | ||
320 | usnic_info("Transistioned %u from %s to %s", | ||
321 | qp_grp->grp_id, | ||
322 | usnic_ib_qp_grp_state_to_string(old_state), | ||
323 | usnic_ib_qp_grp_state_to_string(new_state)); | ||
324 | } else { | ||
325 | usnic_err("Failed to transistion %u from %s to %s", | ||
326 | qp_grp->grp_id, | ||
327 | usnic_ib_qp_grp_state_to_string(old_state), | ||
328 | usnic_ib_qp_grp_state_to_string(new_state)); | ||
329 | } | ||
330 | |||
331 | return status; | ||
332 | } | ||
333 | |||
334 | static struct usnic_vnic_res_chunk** | ||
335 | alloc_res_chunk_list(struct usnic_vnic *vnic, | ||
336 | struct usnic_vnic_res_spec *res_spec, void *owner_obj) | ||
337 | { | ||
338 | enum usnic_vnic_res_type res_type; | ||
339 | struct usnic_vnic_res_chunk **res_chunk_list; | ||
340 | int err, i, res_cnt, res_lst_sz; | ||
341 | |||
342 | for (res_lst_sz = 0; | ||
343 | res_spec->resources[res_lst_sz].type != USNIC_VNIC_RES_TYPE_EOL; | ||
344 | res_lst_sz++) { | ||
345 | /* Do Nothing */ | ||
346 | } | ||
347 | |||
348 | res_chunk_list = kzalloc(sizeof(*res_chunk_list)*(res_lst_sz+1), | ||
349 | GFP_ATOMIC); | ||
350 | if (!res_chunk_list) | ||
351 | return ERR_PTR(-ENOMEM); | ||
352 | |||
353 | for (i = 0; res_spec->resources[i].type != USNIC_VNIC_RES_TYPE_EOL; | ||
354 | i++) { | ||
355 | res_type = res_spec->resources[i].type; | ||
356 | res_cnt = res_spec->resources[i].cnt; | ||
357 | |||
358 | res_chunk_list[i] = usnic_vnic_get_resources(vnic, res_type, | ||
359 | res_cnt, owner_obj); | ||
360 | if (IS_ERR_OR_NULL(res_chunk_list[i])) { | ||
361 | err = (res_chunk_list[i] ? | ||
362 | PTR_ERR(res_chunk_list[i]) : -ENOMEM); | ||
363 | usnic_err("Failed to get %s from %s with err %d\n", | ||
364 | usnic_vnic_res_type_to_str(res_type), | ||
365 | usnic_vnic_pci_name(vnic), | ||
366 | err); | ||
367 | goto out_free_res; | ||
368 | } | ||
369 | } | ||
370 | |||
371 | return res_chunk_list; | ||
372 | |||
373 | out_free_res: | ||
374 | for (i--; i > 0; i--) | ||
375 | usnic_vnic_put_resources(res_chunk_list[i]); | ||
376 | kfree(res_chunk_list); | ||
377 | return ERR_PTR(err); | ||
378 | } | ||
379 | |||
380 | static void free_qp_grp_res(struct usnic_vnic_res_chunk **res_chunk_list) | ||
381 | { | ||
382 | int i; | ||
383 | for (i = 0; res_chunk_list[i]; i++) | ||
384 | usnic_vnic_put_resources(res_chunk_list[i]); | ||
385 | kfree(res_chunk_list); | ||
386 | } | ||
387 | |||
388 | static int qp_grp_and_vf_bind(struct usnic_ib_vf *vf, | ||
389 | struct usnic_ib_pd *pd, | ||
390 | struct usnic_ib_qp_grp *qp_grp) | ||
391 | { | ||
392 | int err; | ||
393 | struct pci_dev *pdev; | ||
394 | |||
395 | WARN_ON(!spin_is_locked(&vf->lock)); | ||
396 | |||
397 | pdev = usnic_vnic_get_pdev(vf->vnic); | ||
398 | if (vf->qp_grp_ref_cnt == 0) { | ||
399 | err = usnic_uiom_attach_dev_to_pd(pd->umem_pd, &pdev->dev); | ||
400 | if (err) { | ||
401 | usnic_err("Failed to attach %s to domain\n", | ||
402 | pci_name(pdev)); | ||
403 | return err; | ||
404 | } | ||
405 | vf->pd = pd; | ||
406 | } | ||
407 | vf->qp_grp_ref_cnt++; | ||
408 | |||
409 | WARN_ON(vf->pd != pd); | ||
410 | qp_grp->vf = vf; | ||
411 | |||
412 | return 0; | ||
413 | } | ||
414 | |||
415 | static void qp_grp_and_vf_unbind(struct usnic_ib_qp_grp *qp_grp) | ||
416 | { | ||
417 | struct pci_dev *pdev; | ||
418 | struct usnic_ib_pd *pd; | ||
419 | |||
420 | WARN_ON(!spin_is_locked(&qp_grp->vf->lock)); | ||
421 | |||
422 | pd = qp_grp->vf->pd; | ||
423 | pdev = usnic_vnic_get_pdev(qp_grp->vf->vnic); | ||
424 | if (--qp_grp->vf->qp_grp_ref_cnt == 0) { | ||
425 | qp_grp->vf->pd = NULL; | ||
426 | usnic_uiom_detach_dev_from_pd(pd->umem_pd, &pdev->dev); | ||
427 | } | ||
428 | qp_grp->vf = NULL; | ||
429 | } | ||
430 | |||
431 | static void log_spec(struct usnic_vnic_res_spec *res_spec) | ||
432 | { | ||
433 | char buf[512]; | ||
434 | usnic_vnic_spec_dump(buf, sizeof(buf), res_spec); | ||
435 | usnic_dbg("%s\n", buf); | ||
436 | } | ||
437 | |||
438 | struct usnic_ib_qp_grp * | ||
439 | usnic_ib_qp_grp_create(struct usnic_fwd_dev *ufdev, | ||
440 | struct usnic_ib_vf *vf, | ||
441 | struct usnic_ib_pd *pd, | ||
442 | struct usnic_vnic_res_spec *res_spec, | ||
443 | enum usnic_transport_type transport) | ||
444 | { | ||
445 | struct usnic_ib_qp_grp *qp_grp; | ||
446 | u16 port_num; | ||
447 | int err; | ||
448 | |||
449 | WARN_ON(!spin_is_locked(&vf->lock)); | ||
450 | |||
451 | err = usnic_vnic_res_spec_satisfied(&min_transport_spec[transport], | ||
452 | res_spec); | ||
453 | if (err) { | ||
454 | usnic_err("Spec does not meet miniumum req for transport %d\n", | ||
455 | transport); | ||
456 | log_spec(res_spec); | ||
457 | return ERR_PTR(err); | ||
458 | } | ||
459 | |||
460 | port_num = usnic_transport_rsrv_port(transport, 0); | ||
461 | if (!port_num) { | ||
462 | usnic_err("Unable to allocate port for %s\n", | ||
463 | netdev_name(ufdev->netdev)); | ||
464 | return ERR_PTR(-EINVAL); | ||
465 | } | ||
466 | |||
467 | qp_grp = kzalloc(sizeof(*qp_grp), GFP_ATOMIC); | ||
468 | if (!qp_grp) { | ||
469 | usnic_err("Unable to alloc qp_grp - Out of memory\n"); | ||
470 | return NULL; | ||
471 | } | ||
472 | |||
473 | qp_grp->res_chunk_list = alloc_res_chunk_list(vf->vnic, res_spec, | ||
474 | qp_grp); | ||
475 | if (IS_ERR_OR_NULL(qp_grp->res_chunk_list)) { | ||
476 | err = qp_grp->res_chunk_list ? | ||
477 | PTR_ERR(qp_grp->res_chunk_list) : -ENOMEM; | ||
478 | usnic_err("Unable to alloc res for %d with err %d\n", | ||
479 | qp_grp->grp_id, err); | ||
480 | goto out_free_port; | ||
481 | } | ||
482 | |||
483 | INIT_LIST_HEAD(&qp_grp->filter_hndls); | ||
484 | spin_lock_init(&qp_grp->lock); | ||
485 | qp_grp->ufdev = ufdev; | ||
486 | qp_grp->transport = transport; | ||
487 | qp_grp->filters[DFLT_FILTER_IDX].transport = transport; | ||
488 | qp_grp->filters[DFLT_FILTER_IDX].port_num = port_num; | ||
489 | qp_grp->state = IB_QPS_RESET; | ||
490 | qp_grp->owner_pid = current->pid; | ||
491 | |||
492 | /* qp_num is same as default filter port_num */ | ||
493 | qp_grp->ibqp.qp_num = qp_grp->filters[DFLT_FILTER_IDX].port_num; | ||
494 | qp_grp->grp_id = qp_grp->ibqp.qp_num; | ||
495 | |||
496 | err = qp_grp_and_vf_bind(vf, pd, qp_grp); | ||
497 | if (err) | ||
498 | goto out_free_port; | ||
499 | |||
500 | usnic_ib_sysfs_qpn_add(qp_grp); | ||
501 | |||
502 | return qp_grp; | ||
503 | |||
504 | out_free_port: | ||
505 | kfree(qp_grp); | ||
506 | usnic_transport_unrsrv_port(transport, port_num); | ||
507 | |||
508 | return ERR_PTR(err); | ||
509 | } | ||
510 | |||
511 | void usnic_ib_qp_grp_destroy(struct usnic_ib_qp_grp *qp_grp) | ||
512 | { | ||
513 | u16 default_port_num; | ||
514 | enum usnic_transport_type transport; | ||
515 | |||
516 | WARN_ON(qp_grp->state != IB_QPS_RESET); | ||
517 | WARN_ON(!spin_is_locked(&qp_grp->vf->lock)); | ||
518 | |||
519 | transport = qp_grp->filters[DFLT_FILTER_IDX].transport; | ||
520 | default_port_num = qp_grp->filters[DFLT_FILTER_IDX].port_num; | ||
521 | |||
522 | usnic_ib_sysfs_qpn_remove(qp_grp); | ||
523 | qp_grp_and_vf_unbind(qp_grp); | ||
524 | free_qp_grp_res(qp_grp->res_chunk_list); | ||
525 | kfree(qp_grp); | ||
526 | usnic_transport_unrsrv_port(transport, default_port_num); | ||
527 | } | ||
528 | |||
529 | struct usnic_vnic_res_chunk* | ||
530 | usnic_ib_qp_grp_get_chunk(struct usnic_ib_qp_grp *qp_grp, | ||
531 | enum usnic_vnic_res_type res_type) | ||
532 | { | ||
533 | int i; | ||
534 | |||
535 | for (i = 0; qp_grp->res_chunk_list[i]; i++) { | ||
536 | if (qp_grp->res_chunk_list[i]->type == res_type) | ||
537 | return qp_grp->res_chunk_list[i]; | ||
538 | } | ||
539 | |||
540 | return ERR_PTR(-EINVAL); | ||
541 | } | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_ib_qp_grp.h b/drivers/infiniband/hw/usnic/usnic_ib_qp_grp.h new file mode 100644 index 000000000000..6416a956dc4a --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_ib_qp_grp.h | |||
@@ -0,0 +1,95 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_IB_QP_GRP_H_ | ||
20 | #define USNIC_IB_QP_GRP_H_ | ||
21 | |||
22 | #include <rdma/ib_verbs.h> | ||
23 | |||
24 | #include "usnic_ib.h" | ||
25 | #include "usnic_abi.h" | ||
26 | #include "usnic_fwd.h" | ||
27 | #include "usnic_vnic.h" | ||
28 | |||
29 | #define MAX_QP_GRP_FILTERS 10 | ||
30 | #define DFLT_FILTER_IDX 0 | ||
31 | |||
32 | /* | ||
33 | * The qp group struct represents all the hw resources needed to present a ib_qp | ||
34 | */ | ||
35 | struct usnic_ib_qp_grp { | ||
36 | struct ib_qp ibqp; | ||
37 | enum ib_qp_state state; | ||
38 | int grp_id; | ||
39 | |||
40 | struct usnic_fwd_dev *ufdev; | ||
41 | short unsigned filter_cnt; | ||
42 | struct usnic_fwd_filter filters[MAX_QP_GRP_FILTERS]; | ||
43 | struct list_head filter_hndls; | ||
44 | enum usnic_transport_type transport; | ||
45 | struct usnic_ib_ucontext *ctx; | ||
46 | |||
47 | struct usnic_vnic_res_chunk **res_chunk_list; | ||
48 | |||
49 | pid_t owner_pid; | ||
50 | struct usnic_ib_vf *vf; | ||
51 | struct list_head link; | ||
52 | |||
53 | spinlock_t lock; | ||
54 | |||
55 | struct kobject kobj; | ||
56 | }; | ||
57 | |||
58 | static const struct | ||
59 | usnic_vnic_res_spec min_transport_spec[USNIC_TRANSPORT_MAX] = { | ||
60 | { /*USNIC_TRANSPORT_UNKNOWN*/ | ||
61 | .resources = { | ||
62 | {.type = USNIC_VNIC_RES_TYPE_EOL, .cnt = 0,}, | ||
63 | }, | ||
64 | }, | ||
65 | { /*USNIC_TRANSPORT_ROCE_CUSTOM*/ | ||
66 | .resources = { | ||
67 | {.type = USNIC_VNIC_RES_TYPE_WQ, .cnt = 1,}, | ||
68 | {.type = USNIC_VNIC_RES_TYPE_RQ, .cnt = 1,}, | ||
69 | {.type = USNIC_VNIC_RES_TYPE_CQ, .cnt = 1,}, | ||
70 | {.type = USNIC_VNIC_RES_TYPE_EOL, .cnt = 0,}, | ||
71 | }, | ||
72 | }, | ||
73 | }; | ||
74 | |||
75 | const char *usnic_ib_qp_grp_state_to_string(enum ib_qp_state state); | ||
76 | int usnic_ib_qp_grp_dump_hdr(char *buf, int buf_sz); | ||
77 | int usnic_ib_qp_grp_dump_rows(void *obj, char *buf, int buf_sz); | ||
78 | struct usnic_ib_qp_grp * | ||
79 | usnic_ib_qp_grp_create(struct usnic_fwd_dev *ufdev, struct usnic_ib_vf *vf, | ||
80 | struct usnic_ib_pd *pd, | ||
81 | struct usnic_vnic_res_spec *res_spec, | ||
82 | enum usnic_transport_type transport); | ||
83 | void usnic_ib_qp_grp_destroy(struct usnic_ib_qp_grp *qp_grp); | ||
84 | int usnic_ib_qp_grp_modify(struct usnic_ib_qp_grp *qp_grp, | ||
85 | enum ib_qp_state new_state, | ||
86 | struct usnic_fwd_filter *fwd_filter); | ||
87 | struct usnic_vnic_res_chunk | ||
88 | *usnic_ib_qp_grp_get_chunk(struct usnic_ib_qp_grp *qp_grp, | ||
89 | enum usnic_vnic_res_type type); | ||
90 | static inline | ||
91 | struct usnic_ib_qp_grp *to_uqp_grp(struct ib_qp *ibqp) | ||
92 | { | ||
93 | return container_of(ibqp, struct usnic_ib_qp_grp, ibqp); | ||
94 | } | ||
95 | #endif /* USNIC_IB_QP_GRP_H_ */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_ib_sysfs.c b/drivers/infiniband/hw/usnic/usnic_ib_sysfs.c new file mode 100644 index 000000000000..bad985e9df08 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_ib_sysfs.c | |||
@@ -0,0 +1,351 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #include <linux/module.h> | ||
20 | #include <linux/init.h> | ||
21 | #include <linux/errno.h> | ||
22 | |||
23 | #include <rdma/ib_user_verbs.h> | ||
24 | #include <rdma/ib_addr.h> | ||
25 | |||
26 | #include "usnic_common_util.h" | ||
27 | #include "usnic_ib.h" | ||
28 | #include "usnic_ib_qp_grp.h" | ||
29 | #include "usnic_vnic.h" | ||
30 | #include "usnic_ib_verbs.h" | ||
31 | #include "usnic_log.h" | ||
32 | |||
33 | #define UPDATE_PTR_LEFT(N, P, L) \ | ||
34 | do { \ | ||
35 | L -= (N); \ | ||
36 | P += (N); \ | ||
37 | } while (0) | ||
38 | |||
39 | static ssize_t usnic_ib_show_fw_ver(struct device *device, | ||
40 | struct device_attribute *attr, | ||
41 | char *buf) | ||
42 | { | ||
43 | struct usnic_ib_dev *us_ibdev = | ||
44 | container_of(device, struct usnic_ib_dev, ib_dev.dev); | ||
45 | struct ethtool_drvinfo info; | ||
46 | |||
47 | mutex_lock(&us_ibdev->usdev_lock); | ||
48 | us_ibdev->netdev->ethtool_ops->get_drvinfo(us_ibdev->netdev, &info); | ||
49 | mutex_unlock(&us_ibdev->usdev_lock); | ||
50 | |||
51 | return scnprintf(buf, PAGE_SIZE, "%s\n", info.fw_version); | ||
52 | } | ||
53 | |||
54 | static ssize_t usnic_ib_show_board(struct device *device, | ||
55 | struct device_attribute *attr, | ||
56 | char *buf) | ||
57 | { | ||
58 | struct usnic_ib_dev *us_ibdev = | ||
59 | container_of(device, struct usnic_ib_dev, ib_dev.dev); | ||
60 | unsigned short subsystem_device_id; | ||
61 | |||
62 | mutex_lock(&us_ibdev->usdev_lock); | ||
63 | subsystem_device_id = us_ibdev->pdev->subsystem_device; | ||
64 | mutex_unlock(&us_ibdev->usdev_lock); | ||
65 | |||
66 | return scnprintf(buf, PAGE_SIZE, "%hu\n", subsystem_device_id); | ||
67 | } | ||
68 | |||
69 | /* | ||
70 | * Report the configuration for this PF | ||
71 | */ | ||
72 | static ssize_t | ||
73 | usnic_ib_show_config(struct device *device, struct device_attribute *attr, | ||
74 | char *buf) | ||
75 | { | ||
76 | struct usnic_ib_dev *us_ibdev; | ||
77 | char *ptr; | ||
78 | unsigned left; | ||
79 | unsigned n; | ||
80 | enum usnic_vnic_res_type res_type; | ||
81 | |||
82 | us_ibdev = container_of(device, struct usnic_ib_dev, ib_dev.dev); | ||
83 | |||
84 | /* Buffer space limit is 1 page */ | ||
85 | ptr = buf; | ||
86 | left = PAGE_SIZE; | ||
87 | |||
88 | mutex_lock(&us_ibdev->usdev_lock); | ||
89 | if (atomic_read(&us_ibdev->vf_cnt.refcount) > 0) { | ||
90 | char *busname; | ||
91 | |||
92 | /* | ||
93 | * bus name seems to come with annoying prefix. | ||
94 | * Remove it if it is predictable | ||
95 | */ | ||
96 | busname = us_ibdev->pdev->bus->name; | ||
97 | if (strncmp(busname, "PCI Bus ", 8) == 0) | ||
98 | busname += 8; | ||
99 | |||
100 | n = scnprintf(ptr, left, | ||
101 | "%s: %s:%d.%d, %s, %pM, %u VFs\n Per VF:", | ||
102 | us_ibdev->ib_dev.name, | ||
103 | busname, | ||
104 | PCI_SLOT(us_ibdev->pdev->devfn), | ||
105 | PCI_FUNC(us_ibdev->pdev->devfn), | ||
106 | netdev_name(us_ibdev->netdev), | ||
107 | us_ibdev->mac, | ||
108 | atomic_read(&us_ibdev->vf_cnt.refcount)); | ||
109 | UPDATE_PTR_LEFT(n, ptr, left); | ||
110 | |||
111 | for (res_type = USNIC_VNIC_RES_TYPE_EOL; | ||
112 | res_type < USNIC_VNIC_RES_TYPE_MAX; | ||
113 | res_type++) { | ||
114 | if (us_ibdev->vf_res_cnt[res_type] == 0) | ||
115 | continue; | ||
116 | n = scnprintf(ptr, left, " %d %s%s", | ||
117 | us_ibdev->vf_res_cnt[res_type], | ||
118 | usnic_vnic_res_type_to_str(res_type), | ||
119 | (res_type < (USNIC_VNIC_RES_TYPE_MAX - 1)) ? | ||
120 | "," : ""); | ||
121 | UPDATE_PTR_LEFT(n, ptr, left); | ||
122 | } | ||
123 | n = scnprintf(ptr, left, "\n"); | ||
124 | UPDATE_PTR_LEFT(n, ptr, left); | ||
125 | } else { | ||
126 | n = scnprintf(ptr, left, "%s: no VFs\n", | ||
127 | us_ibdev->ib_dev.name); | ||
128 | UPDATE_PTR_LEFT(n, ptr, left); | ||
129 | } | ||
130 | mutex_unlock(&us_ibdev->usdev_lock); | ||
131 | |||
132 | return ptr - buf; | ||
133 | } | ||
134 | |||
135 | static ssize_t | ||
136 | usnic_ib_show_iface(struct device *device, struct device_attribute *attr, | ||
137 | char *buf) | ||
138 | { | ||
139 | struct usnic_ib_dev *us_ibdev; | ||
140 | |||
141 | us_ibdev = container_of(device, struct usnic_ib_dev, ib_dev.dev); | ||
142 | |||
143 | return scnprintf(buf, PAGE_SIZE, "%s\n", | ||
144 | netdev_name(us_ibdev->netdev)); | ||
145 | } | ||
146 | |||
147 | static ssize_t | ||
148 | usnic_ib_show_max_vf(struct device *device, struct device_attribute *attr, | ||
149 | char *buf) | ||
150 | { | ||
151 | struct usnic_ib_dev *us_ibdev; | ||
152 | |||
153 | us_ibdev = container_of(device, struct usnic_ib_dev, ib_dev.dev); | ||
154 | |||
155 | return scnprintf(buf, PAGE_SIZE, "%u\n", | ||
156 | atomic_read(&us_ibdev->vf_cnt.refcount)); | ||
157 | } | ||
158 | |||
159 | static ssize_t | ||
160 | usnic_ib_show_qp_per_vf(struct device *device, struct device_attribute *attr, | ||
161 | char *buf) | ||
162 | { | ||
163 | struct usnic_ib_dev *us_ibdev; | ||
164 | int qp_per_vf; | ||
165 | |||
166 | us_ibdev = container_of(device, struct usnic_ib_dev, ib_dev.dev); | ||
167 | qp_per_vf = max(us_ibdev->vf_res_cnt[USNIC_VNIC_RES_TYPE_WQ], | ||
168 | us_ibdev->vf_res_cnt[USNIC_VNIC_RES_TYPE_RQ]); | ||
169 | |||
170 | return scnprintf(buf, PAGE_SIZE, | ||
171 | "%d\n", qp_per_vf); | ||
172 | } | ||
173 | |||
174 | static ssize_t | ||
175 | usnic_ib_show_cq_per_vf(struct device *device, struct device_attribute *attr, | ||
176 | char *buf) | ||
177 | { | ||
178 | struct usnic_ib_dev *us_ibdev; | ||
179 | |||
180 | us_ibdev = container_of(device, struct usnic_ib_dev, ib_dev.dev); | ||
181 | |||
182 | return scnprintf(buf, PAGE_SIZE, "%d\n", | ||
183 | us_ibdev->vf_res_cnt[USNIC_VNIC_RES_TYPE_CQ]); | ||
184 | } | ||
185 | |||
186 | static DEVICE_ATTR(fw_ver, S_IRUGO, usnic_ib_show_fw_ver, NULL); | ||
187 | static DEVICE_ATTR(board_id, S_IRUGO, usnic_ib_show_board, NULL); | ||
188 | static DEVICE_ATTR(config, S_IRUGO, usnic_ib_show_config, NULL); | ||
189 | static DEVICE_ATTR(iface, S_IRUGO, usnic_ib_show_iface, NULL); | ||
190 | static DEVICE_ATTR(max_vf, S_IRUGO, usnic_ib_show_max_vf, NULL); | ||
191 | static DEVICE_ATTR(qp_per_vf, S_IRUGO, usnic_ib_show_qp_per_vf, NULL); | ||
192 | static DEVICE_ATTR(cq_per_vf, S_IRUGO, usnic_ib_show_cq_per_vf, NULL); | ||
193 | |||
194 | static struct device_attribute *usnic_class_attributes[] = { | ||
195 | &dev_attr_fw_ver, | ||
196 | &dev_attr_board_id, | ||
197 | &dev_attr_config, | ||
198 | &dev_attr_iface, | ||
199 | &dev_attr_max_vf, | ||
200 | &dev_attr_qp_per_vf, | ||
201 | &dev_attr_cq_per_vf, | ||
202 | }; | ||
203 | |||
204 | struct qpn_attribute { | ||
205 | struct attribute attr; | ||
206 | ssize_t (*show)(struct usnic_ib_qp_grp *, char *buf); | ||
207 | }; | ||
208 | |||
209 | /* | ||
210 | * Definitions for supporting QPN entries in sysfs | ||
211 | */ | ||
212 | static ssize_t | ||
213 | usnic_ib_qpn_attr_show(struct kobject *kobj, struct attribute *attr, char *buf) | ||
214 | { | ||
215 | struct usnic_ib_qp_grp *qp_grp; | ||
216 | struct qpn_attribute *qpn_attr; | ||
217 | |||
218 | qp_grp = container_of(kobj, struct usnic_ib_qp_grp, kobj); | ||
219 | qpn_attr = container_of(attr, struct qpn_attribute, attr); | ||
220 | |||
221 | return qpn_attr->show(qp_grp, buf); | ||
222 | } | ||
223 | |||
224 | static const struct sysfs_ops usnic_ib_qpn_sysfs_ops = { | ||
225 | .show = usnic_ib_qpn_attr_show | ||
226 | }; | ||
227 | |||
228 | #define QPN_ATTR_RO(NAME) \ | ||
229 | struct qpn_attribute qpn_attr_##NAME = __ATTR_RO(NAME) | ||
230 | |||
231 | static ssize_t context_show(struct usnic_ib_qp_grp *qp_grp, char *buf) | ||
232 | { | ||
233 | return scnprintf(buf, PAGE_SIZE, "0x%p\n", qp_grp->ctx); | ||
234 | } | ||
235 | |||
236 | static ssize_t summary_show(struct usnic_ib_qp_grp *qp_grp, char *buf) | ||
237 | { | ||
238 | int i, j, n; | ||
239 | int left; | ||
240 | char *ptr; | ||
241 | struct usnic_vnic_res_chunk *res_chunk; | ||
242 | struct usnic_fwd_filter_hndl *default_filter_hndl; | ||
243 | struct usnic_vnic_res *vnic_res; | ||
244 | |||
245 | left = PAGE_SIZE; | ||
246 | ptr = buf; | ||
247 | default_filter_hndl = list_first_entry(&qp_grp->filter_hndls, | ||
248 | struct usnic_fwd_filter_hndl, link); | ||
249 | |||
250 | n = scnprintf(ptr, left, | ||
251 | "QPN: %d State: (%s) PID: %u VF Idx: %hu Filter ID: 0x%x ", | ||
252 | qp_grp->ibqp.qp_num, | ||
253 | usnic_ib_qp_grp_state_to_string(qp_grp->state), | ||
254 | qp_grp->owner_pid, | ||
255 | usnic_vnic_get_index(qp_grp->vf->vnic), | ||
256 | default_filter_hndl->id); | ||
257 | UPDATE_PTR_LEFT(n, ptr, left); | ||
258 | |||
259 | for (i = 0; qp_grp->res_chunk_list[i]; i++) { | ||
260 | res_chunk = qp_grp->res_chunk_list[i]; | ||
261 | for (j = 0; j < res_chunk->cnt; j++) { | ||
262 | vnic_res = res_chunk->res[j]; | ||
263 | n = scnprintf(ptr, left, "%s[%d] ", | ||
264 | usnic_vnic_res_type_to_str(vnic_res->type), | ||
265 | vnic_res->vnic_idx); | ||
266 | UPDATE_PTR_LEFT(n, ptr, left); | ||
267 | } | ||
268 | } | ||
269 | |||
270 | n = scnprintf(ptr, left, "\n"); | ||
271 | UPDATE_PTR_LEFT(n, ptr, left); | ||
272 | |||
273 | return ptr - buf; | ||
274 | } | ||
275 | |||
276 | static QPN_ATTR_RO(context); | ||
277 | static QPN_ATTR_RO(summary); | ||
278 | |||
279 | static struct attribute *usnic_ib_qpn_default_attrs[] = { | ||
280 | &qpn_attr_context.attr, | ||
281 | &qpn_attr_summary.attr, | ||
282 | NULL | ||
283 | }; | ||
284 | |||
285 | static struct kobj_type usnic_ib_qpn_type = { | ||
286 | .sysfs_ops = &usnic_ib_qpn_sysfs_ops, | ||
287 | .default_attrs = usnic_ib_qpn_default_attrs | ||
288 | }; | ||
289 | |||
290 | int usnic_ib_sysfs_register_usdev(struct usnic_ib_dev *us_ibdev) | ||
291 | { | ||
292 | int i; | ||
293 | int err; | ||
294 | for (i = 0; i < ARRAY_SIZE(usnic_class_attributes); ++i) { | ||
295 | err = device_create_file(&us_ibdev->ib_dev.dev, | ||
296 | usnic_class_attributes[i]); | ||
297 | if (err) { | ||
298 | usnic_err("Failed to create device file %d for %s eith err %d", | ||
299 | i, us_ibdev->ib_dev.name, err); | ||
300 | return -EINVAL; | ||
301 | } | ||
302 | } | ||
303 | |||
304 | /* create kernel object for looking at individual QPs */ | ||
305 | kobject_get(&us_ibdev->ib_dev.dev.kobj); | ||
306 | us_ibdev->qpn_kobj = kobject_create_and_add("qpn", | ||
307 | &us_ibdev->ib_dev.dev.kobj); | ||
308 | if (us_ibdev->qpn_kobj == NULL) { | ||
309 | kobject_put(&us_ibdev->ib_dev.dev.kobj); | ||
310 | return -ENOMEM; | ||
311 | } | ||
312 | |||
313 | return 0; | ||
314 | } | ||
315 | |||
316 | void usnic_ib_sysfs_unregister_usdev(struct usnic_ib_dev *us_ibdev) | ||
317 | { | ||
318 | int i; | ||
319 | for (i = 0; i < ARRAY_SIZE(usnic_class_attributes); ++i) { | ||
320 | device_remove_file(&us_ibdev->ib_dev.dev, | ||
321 | usnic_class_attributes[i]); | ||
322 | } | ||
323 | |||
324 | kobject_put(us_ibdev->qpn_kobj); | ||
325 | } | ||
326 | |||
327 | void usnic_ib_sysfs_qpn_add(struct usnic_ib_qp_grp *qp_grp) | ||
328 | { | ||
329 | struct usnic_ib_dev *us_ibdev; | ||
330 | int err; | ||
331 | |||
332 | us_ibdev = qp_grp->vf->pf; | ||
333 | |||
334 | err = kobject_init_and_add(&qp_grp->kobj, &usnic_ib_qpn_type, | ||
335 | kobject_get(us_ibdev->qpn_kobj), | ||
336 | "%d", qp_grp->grp_id); | ||
337 | if (err) { | ||
338 | kobject_put(us_ibdev->qpn_kobj); | ||
339 | return; | ||
340 | } | ||
341 | } | ||
342 | |||
343 | void usnic_ib_sysfs_qpn_remove(struct usnic_ib_qp_grp *qp_grp) | ||
344 | { | ||
345 | struct usnic_ib_dev *us_ibdev; | ||
346 | |||
347 | us_ibdev = qp_grp->vf->pf; | ||
348 | |||
349 | kobject_put(&qp_grp->kobj); | ||
350 | kobject_put(us_ibdev->qpn_kobj); | ||
351 | } | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_ib_sysfs.h b/drivers/infiniband/hw/usnic/usnic_ib_sysfs.h new file mode 100644 index 000000000000..0d09b493cd02 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_ib_sysfs.h | |||
@@ -0,0 +1,29 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_IB_SYSFS_H_ | ||
20 | #define USNIC_IB_SYSFS_H_ | ||
21 | |||
22 | #include "usnic_ib.h" | ||
23 | |||
24 | int usnic_ib_sysfs_register_usdev(struct usnic_ib_dev *us_ibdev); | ||
25 | void usnic_ib_sysfs_unregister_usdev(struct usnic_ib_dev *us_ibdev); | ||
26 | void usnic_ib_sysfs_qpn_add(struct usnic_ib_qp_grp *qp_grp); | ||
27 | void usnic_ib_sysfs_qpn_remove(struct usnic_ib_qp_grp *qp_grp); | ||
28 | |||
29 | #endif /* !USNIC_IB_SYSFS_H_ */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_ib_verbs.c b/drivers/infiniband/hw/usnic/usnic_ib_verbs.c new file mode 100644 index 000000000000..8f8dfa2672b0 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_ib_verbs.c | |||
@@ -0,0 +1,736 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | #include <linux/module.h> | ||
19 | #include <linux/init.h> | ||
20 | #include <linux/slab.h> | ||
21 | #include <linux/errno.h> | ||
22 | |||
23 | #include <rdma/ib_user_verbs.h> | ||
24 | #include <rdma/ib_addr.h> | ||
25 | |||
26 | #include "usnic_abi.h" | ||
27 | #include "usnic_ib.h" | ||
28 | #include "usnic_common_util.h" | ||
29 | #include "usnic_ib_qp_grp.h" | ||
30 | #include "usnic_fwd.h" | ||
31 | #include "usnic_log.h" | ||
32 | #include "usnic_uiom.h" | ||
33 | #include "usnic_transport.h" | ||
34 | |||
35 | #define USNIC_DEFAULT_TRANSPORT USNIC_TRANSPORT_ROCE_CUSTOM | ||
36 | |||
37 | static void usnic_ib_fw_string_to_u64(char *fw_ver_str, u64 *fw_ver) | ||
38 | { | ||
39 | *fw_ver = (u64) *fw_ver_str; | ||
40 | } | ||
41 | |||
42 | static int usnic_ib_fill_create_qp_resp(struct usnic_ib_qp_grp *qp_grp, | ||
43 | struct ib_udata *udata) | ||
44 | { | ||
45 | struct usnic_ib_dev *us_ibdev; | ||
46 | struct usnic_ib_create_qp_resp resp; | ||
47 | struct pci_dev *pdev; | ||
48 | struct vnic_dev_bar *bar; | ||
49 | struct usnic_vnic_res_chunk *chunk; | ||
50 | int i, err; | ||
51 | |||
52 | memset(&resp, 0, sizeof(resp)); | ||
53 | |||
54 | us_ibdev = qp_grp->vf->pf; | ||
55 | pdev = usnic_vnic_get_pdev(qp_grp->vf->vnic); | ||
56 | if (!pdev) { | ||
57 | usnic_err("Failed to get pdev of qp_grp %d\n", | ||
58 | qp_grp->grp_id); | ||
59 | return -EFAULT; | ||
60 | } | ||
61 | |||
62 | bar = usnic_vnic_get_bar(qp_grp->vf->vnic, 0); | ||
63 | if (!bar) { | ||
64 | usnic_err("Failed to get bar0 of qp_grp %d vf %s", | ||
65 | qp_grp->grp_id, pci_name(pdev)); | ||
66 | return -EFAULT; | ||
67 | } | ||
68 | |||
69 | resp.vfid = usnic_vnic_get_index(qp_grp->vf->vnic); | ||
70 | resp.bar_bus_addr = bar->bus_addr; | ||
71 | resp.bar_len = bar->len; | ||
72 | resp.transport = qp_grp->transport; | ||
73 | |||
74 | chunk = usnic_ib_qp_grp_get_chunk(qp_grp, USNIC_VNIC_RES_TYPE_RQ); | ||
75 | if (IS_ERR_OR_NULL(chunk)) { | ||
76 | usnic_err("Failed to get chunk %s for qp_grp %d with err %ld\n", | ||
77 | usnic_vnic_res_type_to_str(USNIC_VNIC_RES_TYPE_RQ), | ||
78 | qp_grp->grp_id, | ||
79 | PTR_ERR(chunk)); | ||
80 | return chunk ? PTR_ERR(chunk) : -ENOMEM; | ||
81 | } | ||
82 | |||
83 | WARN_ON(chunk->type != USNIC_VNIC_RES_TYPE_RQ); | ||
84 | resp.rq_cnt = chunk->cnt; | ||
85 | for (i = 0; i < chunk->cnt; i++) | ||
86 | resp.rq_idx[i] = chunk->res[i]->vnic_idx; | ||
87 | |||
88 | chunk = usnic_ib_qp_grp_get_chunk(qp_grp, USNIC_VNIC_RES_TYPE_WQ); | ||
89 | if (IS_ERR_OR_NULL(chunk)) { | ||
90 | usnic_err("Failed to get chunk %s for qp_grp %d with err %ld\n", | ||
91 | usnic_vnic_res_type_to_str(USNIC_VNIC_RES_TYPE_WQ), | ||
92 | qp_grp->grp_id, | ||
93 | PTR_ERR(chunk)); | ||
94 | return chunk ? PTR_ERR(chunk) : -ENOMEM; | ||
95 | } | ||
96 | |||
97 | WARN_ON(chunk->type != USNIC_VNIC_RES_TYPE_WQ); | ||
98 | resp.wq_cnt = chunk->cnt; | ||
99 | for (i = 0; i < chunk->cnt; i++) | ||
100 | resp.wq_idx[i] = chunk->res[i]->vnic_idx; | ||
101 | |||
102 | chunk = usnic_ib_qp_grp_get_chunk(qp_grp, USNIC_VNIC_RES_TYPE_CQ); | ||
103 | if (IS_ERR_OR_NULL(chunk)) { | ||
104 | usnic_err("Failed to get chunk %s for qp_grp %d with err %ld\n", | ||
105 | usnic_vnic_res_type_to_str(USNIC_VNIC_RES_TYPE_CQ), | ||
106 | qp_grp->grp_id, | ||
107 | PTR_ERR(chunk)); | ||
108 | return chunk ? PTR_ERR(chunk) : -ENOMEM; | ||
109 | } | ||
110 | |||
111 | WARN_ON(chunk->type != USNIC_VNIC_RES_TYPE_CQ); | ||
112 | resp.cq_cnt = chunk->cnt; | ||
113 | for (i = 0; i < chunk->cnt; i++) | ||
114 | resp.cq_idx[i] = chunk->res[i]->vnic_idx; | ||
115 | |||
116 | err = ib_copy_to_udata(udata, &resp, sizeof(resp)); | ||
117 | if (err) { | ||
118 | usnic_err("Failed to copy udata for %s", us_ibdev->ib_dev.name); | ||
119 | return err; | ||
120 | } | ||
121 | |||
122 | return 0; | ||
123 | } | ||
124 | |||
125 | static struct usnic_ib_qp_grp* | ||
126 | find_free_vf_and_create_qp_grp(struct usnic_ib_dev *us_ibdev, | ||
127 | struct usnic_ib_pd *pd, | ||
128 | enum usnic_transport_type transport, | ||
129 | struct usnic_vnic_res_spec *res_spec) | ||
130 | { | ||
131 | struct usnic_ib_vf *vf; | ||
132 | struct usnic_vnic *vnic; | ||
133 | struct usnic_ib_qp_grp *qp_grp; | ||
134 | struct device *dev, **dev_list; | ||
135 | int i, found = 0; | ||
136 | |||
137 | BUG_ON(!mutex_is_locked(&us_ibdev->usdev_lock)); | ||
138 | |||
139 | if (list_empty(&us_ibdev->vf_dev_list)) { | ||
140 | usnic_info("No vfs to allocate\n"); | ||
141 | return NULL; | ||
142 | } | ||
143 | |||
144 | if (!us_ibdev->link_up) { | ||
145 | usnic_info("Cannot allocate qp b/c PF link is down\n"); | ||
146 | return NULL; | ||
147 | } | ||
148 | |||
149 | if (usnic_ib_share_vf) { | ||
150 | /* Try to find resouces on a used vf which is in pd */ | ||
151 | dev_list = usnic_uiom_get_dev_list(pd->umem_pd); | ||
152 | for (i = 0; dev_list[i]; i++) { | ||
153 | dev = dev_list[i]; | ||
154 | vf = pci_get_drvdata(to_pci_dev(dev)); | ||
155 | spin_lock(&vf->lock); | ||
156 | vnic = vf->vnic; | ||
157 | if (!usnic_vnic_check_room(vnic, res_spec)) { | ||
158 | usnic_dbg("Found used vnic %s from %s\n", | ||
159 | us_ibdev->ib_dev.name, | ||
160 | pci_name(usnic_vnic_get_pdev( | ||
161 | vnic))); | ||
162 | found = 1; | ||
163 | break; | ||
164 | } | ||
165 | spin_unlock(&vf->lock); | ||
166 | |||
167 | } | ||
168 | usnic_uiom_free_dev_list(dev_list); | ||
169 | } | ||
170 | |||
171 | if (!found) { | ||
172 | /* Try to find resources on an unused vf */ | ||
173 | list_for_each_entry(vf, &us_ibdev->vf_dev_list, link) { | ||
174 | spin_lock(&vf->lock); | ||
175 | vnic = vf->vnic; | ||
176 | if (vf->qp_grp_ref_cnt == 0 && | ||
177 | usnic_vnic_check_room(vnic, res_spec) == 0) { | ||
178 | found = 1; | ||
179 | break; | ||
180 | } | ||
181 | spin_unlock(&vf->lock); | ||
182 | } | ||
183 | } | ||
184 | |||
185 | if (!found) { | ||
186 | usnic_info("No free qp grp found on %s\n", | ||
187 | us_ibdev->ib_dev.name); | ||
188 | return ERR_PTR(-ENOMEM); | ||
189 | } | ||
190 | |||
191 | qp_grp = usnic_ib_qp_grp_create(us_ibdev->ufdev, vf, pd, res_spec, | ||
192 | transport); | ||
193 | spin_unlock(&vf->lock); | ||
194 | if (IS_ERR_OR_NULL(qp_grp)) { | ||
195 | usnic_err("Failed to allocate qp_grp\n"); | ||
196 | return ERR_PTR(qp_grp ? PTR_ERR(qp_grp) : -ENOMEM); | ||
197 | } | ||
198 | |||
199 | return qp_grp; | ||
200 | } | ||
201 | |||
202 | static void qp_grp_destroy(struct usnic_ib_qp_grp *qp_grp) | ||
203 | { | ||
204 | struct usnic_ib_vf *vf = qp_grp->vf; | ||
205 | |||
206 | WARN_ON(qp_grp->state != IB_QPS_RESET); | ||
207 | |||
208 | spin_lock(&vf->lock); | ||
209 | usnic_ib_qp_grp_destroy(qp_grp); | ||
210 | spin_unlock(&vf->lock); | ||
211 | } | ||
212 | |||
213 | static void eth_speed_to_ib_speed(int speed, u8 *active_speed, | ||
214 | u8 *active_width) | ||
215 | { | ||
216 | if (speed <= 10000) { | ||
217 | *active_width = IB_WIDTH_1X; | ||
218 | *active_speed = IB_SPEED_FDR10; | ||
219 | } else if (speed <= 20000) { | ||
220 | *active_width = IB_WIDTH_4X; | ||
221 | *active_speed = IB_SPEED_DDR; | ||
222 | } else if (speed <= 30000) { | ||
223 | *active_width = IB_WIDTH_4X; | ||
224 | *active_speed = IB_SPEED_QDR; | ||
225 | } else if (speed <= 40000) { | ||
226 | *active_width = IB_WIDTH_4X; | ||
227 | *active_speed = IB_SPEED_FDR10; | ||
228 | } else { | ||
229 | *active_width = IB_WIDTH_4X; | ||
230 | *active_speed = IB_SPEED_EDR; | ||
231 | } | ||
232 | } | ||
233 | |||
234 | /* Start of ib callback functions */ | ||
235 | |||
236 | enum rdma_link_layer usnic_ib_port_link_layer(struct ib_device *device, | ||
237 | u8 port_num) | ||
238 | { | ||
239 | return IB_LINK_LAYER_ETHERNET; | ||
240 | } | ||
241 | |||
242 | int usnic_ib_query_device(struct ib_device *ibdev, | ||
243 | struct ib_device_attr *props) | ||
244 | { | ||
245 | struct usnic_ib_dev *us_ibdev = to_usdev(ibdev); | ||
246 | union ib_gid gid; | ||
247 | struct ethtool_drvinfo info; | ||
248 | struct ethtool_cmd cmd; | ||
249 | int qp_per_vf; | ||
250 | |||
251 | usnic_dbg("\n"); | ||
252 | mutex_lock(&us_ibdev->usdev_lock); | ||
253 | us_ibdev->netdev->ethtool_ops->get_drvinfo(us_ibdev->netdev, &info); | ||
254 | us_ibdev->netdev->ethtool_ops->get_settings(us_ibdev->netdev, &cmd); | ||
255 | memset(props, 0, sizeof(*props)); | ||
256 | usnic_mac_to_gid(us_ibdev->mac, &gid.raw[0]); | ||
257 | memcpy(&props->sys_image_guid, &gid.global.interface_id, | ||
258 | sizeof(gid.global.interface_id)); | ||
259 | usnic_ib_fw_string_to_u64(&info.fw_version[0], &props->fw_ver); | ||
260 | props->max_mr_size = USNIC_UIOM_MAX_MR_SIZE; | ||
261 | props->page_size_cap = USNIC_UIOM_PAGE_SIZE; | ||
262 | props->vendor_id = PCI_VENDOR_ID_CISCO; | ||
263 | props->vendor_part_id = PCI_DEVICE_ID_CISCO_VIC_USPACE_NIC; | ||
264 | props->hw_ver = us_ibdev->pdev->subsystem_device; | ||
265 | qp_per_vf = max(us_ibdev->vf_res_cnt[USNIC_VNIC_RES_TYPE_WQ], | ||
266 | us_ibdev->vf_res_cnt[USNIC_VNIC_RES_TYPE_RQ]); | ||
267 | props->max_qp = qp_per_vf * | ||
268 | atomic_read(&us_ibdev->vf_cnt.refcount); | ||
269 | props->device_cap_flags = IB_DEVICE_PORT_ACTIVE_EVENT | | ||
270 | IB_DEVICE_SYS_IMAGE_GUID | IB_DEVICE_BLOCK_MULTICAST_LOOPBACK; | ||
271 | props->max_cq = us_ibdev->vf_res_cnt[USNIC_VNIC_RES_TYPE_CQ] * | ||
272 | atomic_read(&us_ibdev->vf_cnt.refcount); | ||
273 | props->max_pd = USNIC_UIOM_MAX_PD_CNT; | ||
274 | props->max_mr = USNIC_UIOM_MAX_MR_CNT; | ||
275 | props->local_ca_ack_delay = 0; | ||
276 | props->max_pkeys = 0; | ||
277 | props->atomic_cap = IB_ATOMIC_NONE; | ||
278 | props->masked_atomic_cap = props->atomic_cap; | ||
279 | props->max_qp_rd_atom = 0; | ||
280 | props->max_qp_init_rd_atom = 0; | ||
281 | props->max_res_rd_atom = 0; | ||
282 | props->max_srq = 0; | ||
283 | props->max_srq_wr = 0; | ||
284 | props->max_srq_sge = 0; | ||
285 | props->max_fast_reg_page_list_len = 0; | ||
286 | props->max_mcast_grp = 0; | ||
287 | props->max_mcast_qp_attach = 0; | ||
288 | props->max_total_mcast_qp_attach = 0; | ||
289 | props->max_map_per_fmr = 0; | ||
290 | /* Owned by Userspace | ||
291 | * max_qp_wr, max_sge, max_sge_rd, max_cqe */ | ||
292 | mutex_unlock(&us_ibdev->usdev_lock); | ||
293 | |||
294 | return 0; | ||
295 | } | ||
296 | |||
297 | int usnic_ib_query_port(struct ib_device *ibdev, u8 port, | ||
298 | struct ib_port_attr *props) | ||
299 | { | ||
300 | struct usnic_ib_dev *us_ibdev = to_usdev(ibdev); | ||
301 | struct ethtool_cmd cmd; | ||
302 | |||
303 | usnic_dbg("\n"); | ||
304 | |||
305 | mutex_lock(&us_ibdev->usdev_lock); | ||
306 | us_ibdev->netdev->ethtool_ops->get_settings(us_ibdev->netdev, &cmd); | ||
307 | memset(props, 0, sizeof(*props)); | ||
308 | |||
309 | props->lid = 0; | ||
310 | props->lmc = 1; | ||
311 | props->sm_lid = 0; | ||
312 | props->sm_sl = 0; | ||
313 | |||
314 | if (us_ibdev->link_up) { | ||
315 | props->state = IB_PORT_ACTIVE; | ||
316 | props->phys_state = 5; | ||
317 | } else { | ||
318 | props->state = IB_PORT_DOWN; | ||
319 | props->phys_state = 3; | ||
320 | } | ||
321 | |||
322 | props->port_cap_flags = 0; | ||
323 | props->gid_tbl_len = 1; | ||
324 | props->pkey_tbl_len = 1; | ||
325 | props->bad_pkey_cntr = 0; | ||
326 | props->qkey_viol_cntr = 0; | ||
327 | eth_speed_to_ib_speed(cmd.speed, &props->active_speed, | ||
328 | &props->active_width); | ||
329 | props->max_mtu = IB_MTU_4096; | ||
330 | props->active_mtu = iboe_get_mtu(us_ibdev->mtu); | ||
331 | /* Userspace will adjust for hdrs */ | ||
332 | props->max_msg_sz = us_ibdev->mtu; | ||
333 | props->max_vl_num = 1; | ||
334 | mutex_unlock(&us_ibdev->usdev_lock); | ||
335 | |||
336 | return 0; | ||
337 | } | ||
338 | |||
339 | int usnic_ib_query_qp(struct ib_qp *qp, struct ib_qp_attr *qp_attr, | ||
340 | int qp_attr_mask, | ||
341 | struct ib_qp_init_attr *qp_init_attr) | ||
342 | { | ||
343 | struct usnic_ib_qp_grp *qp_grp; | ||
344 | struct usnic_ib_vf *vf; | ||
345 | int err; | ||
346 | |||
347 | usnic_dbg("\n"); | ||
348 | |||
349 | memset(qp_attr, 0, sizeof(*qp_attr)); | ||
350 | memset(qp_init_attr, 0, sizeof(*qp_init_attr)); | ||
351 | |||
352 | qp_grp = to_uqp_grp(qp); | ||
353 | vf = qp_grp->vf; | ||
354 | mutex_lock(&vf->pf->usdev_lock); | ||
355 | usnic_dbg("\n"); | ||
356 | qp_attr->qp_state = qp_grp->state; | ||
357 | qp_attr->cur_qp_state = qp_grp->state; | ||
358 | |||
359 | switch (qp_grp->ibqp.qp_type) { | ||
360 | case IB_QPT_UD: | ||
361 | qp_attr->qkey = 0; | ||
362 | break; | ||
363 | default: | ||
364 | usnic_err("Unexpected qp_type %d\n", qp_grp->ibqp.qp_type); | ||
365 | err = -EINVAL; | ||
366 | goto err_out; | ||
367 | } | ||
368 | |||
369 | mutex_unlock(&vf->pf->usdev_lock); | ||
370 | return 0; | ||
371 | |||
372 | err_out: | ||
373 | mutex_unlock(&vf->pf->usdev_lock); | ||
374 | return err; | ||
375 | } | ||
376 | |||
377 | int usnic_ib_query_gid(struct ib_device *ibdev, u8 port, int index, | ||
378 | union ib_gid *gid) | ||
379 | { | ||
380 | |||
381 | struct usnic_ib_dev *us_ibdev = to_usdev(ibdev); | ||
382 | usnic_dbg("\n"); | ||
383 | |||
384 | if (index > 1) | ||
385 | return -EINVAL; | ||
386 | |||
387 | mutex_lock(&us_ibdev->usdev_lock); | ||
388 | memset(&(gid->raw[0]), 0, sizeof(gid->raw)); | ||
389 | usnic_mac_to_gid(us_ibdev->mac, &gid->raw[0]); | ||
390 | mutex_unlock(&us_ibdev->usdev_lock); | ||
391 | |||
392 | return 0; | ||
393 | } | ||
394 | |||
395 | int usnic_ib_query_pkey(struct ib_device *ibdev, u8 port, u16 index, | ||
396 | u16 *pkey) | ||
397 | { | ||
398 | if (index > 1) | ||
399 | return -EINVAL; | ||
400 | |||
401 | *pkey = 0xffff; | ||
402 | return 0; | ||
403 | } | ||
404 | |||
405 | struct ib_pd *usnic_ib_alloc_pd(struct ib_device *ibdev, | ||
406 | struct ib_ucontext *context, | ||
407 | struct ib_udata *udata) | ||
408 | { | ||
409 | struct usnic_ib_pd *pd; | ||
410 | void *umem_pd; | ||
411 | |||
412 | usnic_dbg("\n"); | ||
413 | |||
414 | pd = kzalloc(sizeof(*pd), GFP_KERNEL); | ||
415 | if (!pd) | ||
416 | return ERR_PTR(-ENOMEM); | ||
417 | |||
418 | umem_pd = pd->umem_pd = usnic_uiom_alloc_pd(); | ||
419 | if (IS_ERR_OR_NULL(umem_pd)) { | ||
420 | kfree(pd); | ||
421 | return ERR_PTR(umem_pd ? PTR_ERR(umem_pd) : -ENOMEM); | ||
422 | } | ||
423 | |||
424 | usnic_info("domain 0x%p allocated for context 0x%p and device %s\n", | ||
425 | pd, context, ibdev->name); | ||
426 | return &pd->ibpd; | ||
427 | } | ||
428 | |||
429 | int usnic_ib_dealloc_pd(struct ib_pd *pd) | ||
430 | { | ||
431 | usnic_info("freeing domain 0x%p\n", pd); | ||
432 | |||
433 | usnic_uiom_dealloc_pd((to_upd(pd))->umem_pd); | ||
434 | kfree(pd); | ||
435 | return 0; | ||
436 | } | ||
437 | |||
438 | struct ib_qp *usnic_ib_create_qp(struct ib_pd *pd, | ||
439 | struct ib_qp_init_attr *init_attr, | ||
440 | struct ib_udata *udata) | ||
441 | { | ||
442 | int err; | ||
443 | struct usnic_ib_dev *us_ibdev; | ||
444 | struct usnic_ib_qp_grp *qp_grp; | ||
445 | struct usnic_ib_ucontext *ucontext; | ||
446 | int cq_cnt; | ||
447 | struct usnic_vnic_res_spec res_spec; | ||
448 | |||
449 | usnic_dbg("\n"); | ||
450 | |||
451 | ucontext = to_uucontext(pd->uobject->context); | ||
452 | us_ibdev = to_usdev(pd->device); | ||
453 | |||
454 | if (init_attr->qp_type != IB_QPT_UD) { | ||
455 | usnic_err("%s asked to make a non-UD QP: %d\n", | ||
456 | us_ibdev->ib_dev.name, init_attr->qp_type); | ||
457 | return ERR_PTR(-EINVAL); | ||
458 | } | ||
459 | |||
460 | mutex_lock(&us_ibdev->usdev_lock); | ||
461 | cq_cnt = (init_attr->send_cq == init_attr->recv_cq) ? 1 : 2, | ||
462 | res_spec = min_transport_spec[USNIC_DEFAULT_TRANSPORT]; | ||
463 | usnic_vnic_res_spec_update(&res_spec, USNIC_VNIC_RES_TYPE_CQ, cq_cnt); | ||
464 | qp_grp = find_free_vf_and_create_qp_grp(us_ibdev, to_upd(pd), | ||
465 | USNIC_DEFAULT_TRANSPORT, | ||
466 | &res_spec); | ||
467 | if (IS_ERR_OR_NULL(qp_grp)) { | ||
468 | err = (qp_grp ? PTR_ERR(qp_grp) : -ENOMEM); | ||
469 | goto out_release_mutex; | ||
470 | } | ||
471 | |||
472 | err = usnic_ib_fill_create_qp_resp(qp_grp, udata); | ||
473 | if (err) { | ||
474 | err = -EBUSY; | ||
475 | goto out_release_qp_grp; | ||
476 | } | ||
477 | |||
478 | qp_grp->ctx = ucontext; | ||
479 | list_add_tail(&qp_grp->link, &ucontext->qp_grp_list); | ||
480 | usnic_ib_log_vf(qp_grp->vf); | ||
481 | mutex_unlock(&us_ibdev->usdev_lock); | ||
482 | return &qp_grp->ibqp; | ||
483 | |||
484 | out_release_qp_grp: | ||
485 | qp_grp_destroy(qp_grp); | ||
486 | out_release_mutex: | ||
487 | mutex_unlock(&us_ibdev->usdev_lock); | ||
488 | return ERR_PTR(err); | ||
489 | } | ||
490 | |||
491 | int usnic_ib_destroy_qp(struct ib_qp *qp) | ||
492 | { | ||
493 | struct usnic_ib_qp_grp *qp_grp; | ||
494 | struct usnic_ib_vf *vf; | ||
495 | |||
496 | usnic_dbg("\n"); | ||
497 | |||
498 | qp_grp = to_uqp_grp(qp); | ||
499 | vf = qp_grp->vf; | ||
500 | mutex_lock(&vf->pf->usdev_lock); | ||
501 | if (usnic_ib_qp_grp_modify(qp_grp, IB_QPS_RESET, NULL)) { | ||
502 | usnic_err("Failed to move qp grp %u to reset\n", | ||
503 | qp_grp->grp_id); | ||
504 | } | ||
505 | |||
506 | list_del(&qp_grp->link); | ||
507 | qp_grp_destroy(qp_grp); | ||
508 | mutex_unlock(&vf->pf->usdev_lock); | ||
509 | |||
510 | return 0; | ||
511 | } | ||
512 | |||
513 | int usnic_ib_modify_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr, | ||
514 | int attr_mask, struct ib_udata *udata) | ||
515 | { | ||
516 | struct usnic_ib_qp_grp *qp_grp; | ||
517 | int status; | ||
518 | usnic_dbg("\n"); | ||
519 | |||
520 | qp_grp = to_uqp_grp(ibqp); | ||
521 | |||
522 | /* TODO: Future Support All States */ | ||
523 | mutex_lock(&qp_grp->vf->pf->usdev_lock); | ||
524 | if ((attr_mask & IB_QP_STATE) && attr->qp_state == IB_QPS_INIT) { | ||
525 | status = usnic_ib_qp_grp_modify(qp_grp, IB_QPS_INIT, | ||
526 | &qp_grp->filters[DFLT_FILTER_IDX]); | ||
527 | } else if ((attr_mask & IB_QP_STATE) && attr->qp_state == IB_QPS_RTR) { | ||
528 | status = usnic_ib_qp_grp_modify(qp_grp, IB_QPS_RTR, NULL); | ||
529 | } else if ((attr_mask & IB_QP_STATE) && attr->qp_state == IB_QPS_RTS) { | ||
530 | status = usnic_ib_qp_grp_modify(qp_grp, IB_QPS_RTS, NULL); | ||
531 | } else { | ||
532 | usnic_err("Unexpected combination mask: %u state: %u\n", | ||
533 | attr_mask & IB_QP_STATE, attr->qp_state); | ||
534 | status = -EINVAL; | ||
535 | } | ||
536 | |||
537 | mutex_unlock(&qp_grp->vf->pf->usdev_lock); | ||
538 | return status; | ||
539 | } | ||
540 | |||
541 | struct ib_cq *usnic_ib_create_cq(struct ib_device *ibdev, int entries, | ||
542 | int vector, struct ib_ucontext *context, | ||
543 | struct ib_udata *udata) | ||
544 | { | ||
545 | struct ib_cq *cq; | ||
546 | |||
547 | usnic_dbg("\n"); | ||
548 | cq = kzalloc(sizeof(*cq), GFP_KERNEL); | ||
549 | if (!cq) | ||
550 | return ERR_PTR(-EBUSY); | ||
551 | |||
552 | return cq; | ||
553 | } | ||
554 | |||
555 | int usnic_ib_destroy_cq(struct ib_cq *cq) | ||
556 | { | ||
557 | usnic_dbg("\n"); | ||
558 | kfree(cq); | ||
559 | return 0; | ||
560 | } | ||
561 | |||
562 | struct ib_mr *usnic_ib_reg_mr(struct ib_pd *pd, u64 start, u64 length, | ||
563 | u64 virt_addr, int access_flags, | ||
564 | struct ib_udata *udata) | ||
565 | { | ||
566 | struct usnic_ib_mr *mr; | ||
567 | int err; | ||
568 | |||
569 | usnic_dbg("start 0x%llx va 0x%llx length 0x%llx\n", start, | ||
570 | virt_addr, length); | ||
571 | |||
572 | mr = kzalloc(sizeof(*mr), GFP_KERNEL); | ||
573 | if (IS_ERR_OR_NULL(mr)) | ||
574 | return ERR_PTR(mr ? PTR_ERR(mr) : -ENOMEM); | ||
575 | |||
576 | mr->umem = usnic_uiom_reg_get(to_upd(pd)->umem_pd, start, length, | ||
577 | access_flags, 0); | ||
578 | if (IS_ERR_OR_NULL(mr->umem)) { | ||
579 | err = (mr->umem) ? PTR_ERR(mr->umem) : -EFAULT; | ||
580 | goto err_free; | ||
581 | } | ||
582 | |||
583 | mr->ibmr.lkey = mr->ibmr.rkey = 0; | ||
584 | return &mr->ibmr; | ||
585 | |||
586 | err_free: | ||
587 | kfree(mr); | ||
588 | return ERR_PTR(err); | ||
589 | } | ||
590 | |||
591 | int usnic_ib_dereg_mr(struct ib_mr *ibmr) | ||
592 | { | ||
593 | struct usnic_ib_mr *mr = to_umr(ibmr); | ||
594 | |||
595 | usnic_dbg("va 0x%lx length 0x%zx\n", mr->umem->va, mr->umem->length); | ||
596 | |||
597 | usnic_uiom_reg_release(mr->umem, ibmr->pd->uobject->context->closing); | ||
598 | kfree(mr); | ||
599 | return 0; | ||
600 | } | ||
601 | |||
602 | struct ib_ucontext *usnic_ib_alloc_ucontext(struct ib_device *ibdev, | ||
603 | struct ib_udata *udata) | ||
604 | { | ||
605 | struct usnic_ib_ucontext *context; | ||
606 | struct usnic_ib_dev *us_ibdev = to_usdev(ibdev); | ||
607 | usnic_dbg("\n"); | ||
608 | |||
609 | context = kmalloc(sizeof(*context), GFP_KERNEL); | ||
610 | if (!context) | ||
611 | return ERR_PTR(-ENOMEM); | ||
612 | |||
613 | INIT_LIST_HEAD(&context->qp_grp_list); | ||
614 | mutex_lock(&us_ibdev->usdev_lock); | ||
615 | list_add_tail(&context->link, &us_ibdev->ctx_list); | ||
616 | mutex_unlock(&us_ibdev->usdev_lock); | ||
617 | |||
618 | return &context->ibucontext; | ||
619 | } | ||
620 | |||
621 | int usnic_ib_dealloc_ucontext(struct ib_ucontext *ibcontext) | ||
622 | { | ||
623 | struct usnic_ib_ucontext *context = to_uucontext(ibcontext); | ||
624 | struct usnic_ib_dev *us_ibdev = to_usdev(ibcontext->device); | ||
625 | usnic_dbg("\n"); | ||
626 | |||
627 | mutex_lock(&us_ibdev->usdev_lock); | ||
628 | BUG_ON(!list_empty(&context->qp_grp_list)); | ||
629 | list_del(&context->link); | ||
630 | mutex_unlock(&us_ibdev->usdev_lock); | ||
631 | kfree(context); | ||
632 | return 0; | ||
633 | } | ||
634 | |||
635 | int usnic_ib_mmap(struct ib_ucontext *context, | ||
636 | struct vm_area_struct *vma) | ||
637 | { | ||
638 | struct usnic_ib_ucontext *uctx = to_ucontext(context); | ||
639 | struct usnic_ib_dev *us_ibdev; | ||
640 | struct usnic_ib_qp_grp *qp_grp; | ||
641 | struct usnic_ib_vf *vf; | ||
642 | struct vnic_dev_bar *bar; | ||
643 | dma_addr_t bus_addr; | ||
644 | unsigned int len; | ||
645 | unsigned int vfid; | ||
646 | |||
647 | usnic_dbg("\n"); | ||
648 | |||
649 | us_ibdev = to_usdev(context->device); | ||
650 | vma->vm_flags |= VM_IO; | ||
651 | vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot); | ||
652 | vfid = vma->vm_pgoff; | ||
653 | usnic_dbg("Page Offset %lu PAGE_SHIFT %u VFID %u\n", | ||
654 | vma->vm_pgoff, PAGE_SHIFT, vfid); | ||
655 | |||
656 | mutex_lock(&us_ibdev->usdev_lock); | ||
657 | list_for_each_entry(qp_grp, &uctx->qp_grp_list, link) { | ||
658 | vf = qp_grp->vf; | ||
659 | if (usnic_vnic_get_index(vf->vnic) == vfid) { | ||
660 | bar = usnic_vnic_get_bar(vf->vnic, 0); | ||
661 | if ((vma->vm_end - vma->vm_start) != bar->len) { | ||
662 | usnic_err("Bar0 Len %lu - Request map %lu\n", | ||
663 | bar->len, | ||
664 | vma->vm_end - vma->vm_start); | ||
665 | mutex_unlock(&us_ibdev->usdev_lock); | ||
666 | return -EINVAL; | ||
667 | } | ||
668 | bus_addr = bar->bus_addr; | ||
669 | len = bar->len; | ||
670 | usnic_dbg("bus: %pa vaddr: %p size: %ld\n", | ||
671 | &bus_addr, bar->vaddr, bar->len); | ||
672 | mutex_unlock(&us_ibdev->usdev_lock); | ||
673 | |||
674 | return remap_pfn_range(vma, | ||
675 | vma->vm_start, | ||
676 | bus_addr >> PAGE_SHIFT, | ||
677 | len, vma->vm_page_prot); | ||
678 | } | ||
679 | } | ||
680 | |||
681 | mutex_unlock(&us_ibdev->usdev_lock); | ||
682 | usnic_err("No VF %u found\n", vfid); | ||
683 | return -EINVAL; | ||
684 | } | ||
685 | |||
686 | /* In ib callbacks section - Start of stub funcs */ | ||
687 | struct ib_ah *usnic_ib_create_ah(struct ib_pd *pd, | ||
688 | struct ib_ah_attr *ah_attr) | ||
689 | { | ||
690 | usnic_dbg("\n"); | ||
691 | return ERR_PTR(-EPERM); | ||
692 | } | ||
693 | |||
694 | int usnic_ib_destroy_ah(struct ib_ah *ah) | ||
695 | { | ||
696 | usnic_dbg("\n"); | ||
697 | return -EINVAL; | ||
698 | } | ||
699 | |||
700 | int usnic_ib_post_send(struct ib_qp *ibqp, struct ib_send_wr *wr, | ||
701 | struct ib_send_wr **bad_wr) | ||
702 | { | ||
703 | usnic_dbg("\n"); | ||
704 | return -EINVAL; | ||
705 | } | ||
706 | |||
707 | int usnic_ib_post_recv(struct ib_qp *ibqp, struct ib_recv_wr *wr, | ||
708 | struct ib_recv_wr **bad_wr) | ||
709 | { | ||
710 | usnic_dbg("\n"); | ||
711 | return -EINVAL; | ||
712 | } | ||
713 | |||
714 | int usnic_ib_poll_cq(struct ib_cq *ibcq, int num_entries, | ||
715 | struct ib_wc *wc) | ||
716 | { | ||
717 | usnic_dbg("\n"); | ||
718 | return -EINVAL; | ||
719 | } | ||
720 | |||
721 | int usnic_ib_req_notify_cq(struct ib_cq *cq, | ||
722 | enum ib_cq_notify_flags flags) | ||
723 | { | ||
724 | usnic_dbg("\n"); | ||
725 | return -EINVAL; | ||
726 | } | ||
727 | |||
728 | struct ib_mr *usnic_ib_get_dma_mr(struct ib_pd *pd, int acc) | ||
729 | { | ||
730 | usnic_dbg("\n"); | ||
731 | return ERR_PTR(-ENOMEM); | ||
732 | } | ||
733 | |||
734 | |||
735 | /* In ib callbacks section - End of stub funcs */ | ||
736 | /* End of ib callbacks section */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_ib_verbs.h b/drivers/infiniband/hw/usnic/usnic_ib_verbs.h new file mode 100644 index 000000000000..bb864f5aed70 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_ib_verbs.h | |||
@@ -0,0 +1,72 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_IB_VERBS_H_ | ||
20 | #define USNIC_IB_VERBS_H_ | ||
21 | |||
22 | #include "usnic_ib.h" | ||
23 | |||
24 | enum rdma_link_layer usnic_ib_port_link_layer(struct ib_device *device, | ||
25 | u8 port_num); | ||
26 | int usnic_ib_query_device(struct ib_device *ibdev, | ||
27 | struct ib_device_attr *props); | ||
28 | int usnic_ib_query_port(struct ib_device *ibdev, u8 port, | ||
29 | struct ib_port_attr *props); | ||
30 | int usnic_ib_query_qp(struct ib_qp *qp, struct ib_qp_attr *qp_attr, | ||
31 | int qp_attr_mask, | ||
32 | struct ib_qp_init_attr *qp_init_attr); | ||
33 | int usnic_ib_query_gid(struct ib_device *ibdev, u8 port, int index, | ||
34 | union ib_gid *gid); | ||
35 | int usnic_ib_query_pkey(struct ib_device *ibdev, u8 port, u16 index, | ||
36 | u16 *pkey); | ||
37 | struct ib_pd *usnic_ib_alloc_pd(struct ib_device *ibdev, | ||
38 | struct ib_ucontext *context, | ||
39 | struct ib_udata *udata); | ||
40 | int usnic_ib_dealloc_pd(struct ib_pd *pd); | ||
41 | struct ib_qp *usnic_ib_create_qp(struct ib_pd *pd, | ||
42 | struct ib_qp_init_attr *init_attr, | ||
43 | struct ib_udata *udata); | ||
44 | int usnic_ib_destroy_qp(struct ib_qp *qp); | ||
45 | int usnic_ib_modify_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr, | ||
46 | int attr_mask, struct ib_udata *udata); | ||
47 | struct ib_cq *usnic_ib_create_cq(struct ib_device *ibdev, int entries, | ||
48 | int vector, struct ib_ucontext *context, | ||
49 | struct ib_udata *udata); | ||
50 | int usnic_ib_destroy_cq(struct ib_cq *cq); | ||
51 | struct ib_mr *usnic_ib_reg_mr(struct ib_pd *pd, u64 start, u64 length, | ||
52 | u64 virt_addr, int access_flags, | ||
53 | struct ib_udata *udata); | ||
54 | int usnic_ib_dereg_mr(struct ib_mr *ibmr); | ||
55 | struct ib_ucontext *usnic_ib_alloc_ucontext(struct ib_device *ibdev, | ||
56 | struct ib_udata *udata); | ||
57 | int usnic_ib_dealloc_ucontext(struct ib_ucontext *ibcontext); | ||
58 | int usnic_ib_mmap(struct ib_ucontext *context, | ||
59 | struct vm_area_struct *vma); | ||
60 | struct ib_ah *usnic_ib_create_ah(struct ib_pd *pd, | ||
61 | struct ib_ah_attr *ah_attr); | ||
62 | int usnic_ib_destroy_ah(struct ib_ah *ah); | ||
63 | int usnic_ib_post_send(struct ib_qp *ibqp, struct ib_send_wr *wr, | ||
64 | struct ib_send_wr **bad_wr); | ||
65 | int usnic_ib_post_recv(struct ib_qp *ibqp, struct ib_recv_wr *wr, | ||
66 | struct ib_recv_wr **bad_wr); | ||
67 | int usnic_ib_poll_cq(struct ib_cq *ibcq, int num_entries, | ||
68 | struct ib_wc *wc); | ||
69 | int usnic_ib_req_notify_cq(struct ib_cq *cq, | ||
70 | enum ib_cq_notify_flags flags); | ||
71 | struct ib_mr *usnic_ib_get_dma_mr(struct ib_pd *pd, int acc); | ||
72 | #endif /* !USNIC_IB_VERBS_H */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_log.h b/drivers/infiniband/hw/usnic/usnic_log.h new file mode 100644 index 000000000000..75777a66c684 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_log.h | |||
@@ -0,0 +1,58 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_LOG_H_ | ||
20 | #define USNIC_LOG_H_ | ||
21 | |||
22 | #include "usnic.h" | ||
23 | |||
24 | extern unsigned int usnic_log_lvl; | ||
25 | |||
26 | #define USNIC_LOG_LVL_NONE (0) | ||
27 | #define USNIC_LOG_LVL_ERR (1) | ||
28 | #define USNIC_LOG_LVL_INFO (2) | ||
29 | #define USNIC_LOG_LVL_DBG (3) | ||
30 | |||
31 | #define usnic_printk(lvl, args...) \ | ||
32 | do { \ | ||
33 | printk(lvl "%s:%s:%d: ", DRV_NAME, __func__, \ | ||
34 | __LINE__); \ | ||
35 | printk(args); \ | ||
36 | } while (0) | ||
37 | |||
38 | #define usnic_dbg(args...) \ | ||
39 | do { \ | ||
40 | if (unlikely(usnic_log_lvl >= USNIC_LOG_LVL_DBG)) { \ | ||
41 | usnic_printk(KERN_INFO, args); \ | ||
42 | } \ | ||
43 | } while (0) | ||
44 | |||
45 | #define usnic_info(args...) \ | ||
46 | do { \ | ||
47 | if (usnic_log_lvl >= USNIC_LOG_LVL_INFO) { \ | ||
48 | usnic_printk(KERN_INFO, args); \ | ||
49 | } \ | ||
50 | } while (0) | ||
51 | |||
52 | #define usnic_err(args...) \ | ||
53 | do { \ | ||
54 | if (usnic_log_lvl >= USNIC_LOG_LVL_ERR) { \ | ||
55 | usnic_printk(KERN_ERR, args); \ | ||
56 | } \ | ||
57 | } while (0) | ||
58 | #endif /* !USNIC_LOG_H_ */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_transport.c b/drivers/infiniband/hw/usnic/usnic_transport.c new file mode 100644 index 000000000000..723bd6c3a8f8 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_transport.c | |||
@@ -0,0 +1,125 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | #include <linux/bitmap.h> | ||
19 | #include <linux/module.h> | ||
20 | #include <linux/slab.h> | ||
21 | |||
22 | #include "usnic_transport.h" | ||
23 | #include "usnic_log.h" | ||
24 | |||
25 | /* ROCE */ | ||
26 | static unsigned long *roce_bitmap; | ||
27 | static u16 roce_next_port = 1; | ||
28 | #define ROCE_BITMAP_SZ ((1 << (8 /*CHAR_BIT*/ * sizeof(u16)))/8 /*CHAR BIT*/) | ||
29 | static DEFINE_SPINLOCK(roce_bitmap_lock); | ||
30 | |||
31 | static const char *transport_to_str(enum usnic_transport_type type) | ||
32 | { | ||
33 | switch (type) { | ||
34 | case USNIC_TRANSPORT_UNKNOWN: | ||
35 | return "Unknown"; | ||
36 | case USNIC_TRANSPORT_ROCE_CUSTOM: | ||
37 | return "roce custom"; | ||
38 | case USNIC_TRANSPORT_MAX: | ||
39 | return "Max?"; | ||
40 | default: | ||
41 | return "Not known"; | ||
42 | } | ||
43 | } | ||
44 | |||
45 | /* | ||
46 | * reserve a port number. if "0" specified, we will try to pick one | ||
47 | * starting at roce_next_port. roce_next_port will take on the values | ||
48 | * 1..4096 | ||
49 | */ | ||
50 | u16 usnic_transport_rsrv_port(enum usnic_transport_type type, u16 port_num) | ||
51 | { | ||
52 | if (type == USNIC_TRANSPORT_ROCE_CUSTOM) { | ||
53 | spin_lock(&roce_bitmap_lock); | ||
54 | if (!port_num) { | ||
55 | port_num = bitmap_find_next_zero_area(roce_bitmap, | ||
56 | ROCE_BITMAP_SZ, | ||
57 | roce_next_port /* start */, | ||
58 | 1 /* nr */, | ||
59 | 0 /* align */); | ||
60 | roce_next_port = (port_num & 4095) + 1; | ||
61 | } else if (test_bit(port_num, roce_bitmap)) { | ||
62 | usnic_err("Failed to allocate port for %s\n", | ||
63 | transport_to_str(type)); | ||
64 | spin_unlock(&roce_bitmap_lock); | ||
65 | goto out_fail; | ||
66 | } | ||
67 | bitmap_set(roce_bitmap, port_num, 1); | ||
68 | spin_unlock(&roce_bitmap_lock); | ||
69 | } else { | ||
70 | usnic_err("Failed to allocate port - transport %s unsupported\n", | ||
71 | transport_to_str(type)); | ||
72 | goto out_fail; | ||
73 | } | ||
74 | |||
75 | usnic_dbg("Allocating port %hu for %s\n", port_num, | ||
76 | transport_to_str(type)); | ||
77 | return port_num; | ||
78 | |||
79 | out_fail: | ||
80 | return 0; | ||
81 | } | ||
82 | |||
83 | void usnic_transport_unrsrv_port(enum usnic_transport_type type, u16 port_num) | ||
84 | { | ||
85 | if (type == USNIC_TRANSPORT_ROCE_CUSTOM) { | ||
86 | spin_lock(&roce_bitmap_lock); | ||
87 | if (!port_num) { | ||
88 | usnic_err("Unreserved unvalid port num 0 for %s\n", | ||
89 | transport_to_str(type)); | ||
90 | goto out_roce_custom; | ||
91 | } | ||
92 | |||
93 | if (!test_bit(port_num, roce_bitmap)) { | ||
94 | usnic_err("Unreserving invalid %hu for %s\n", | ||
95 | port_num, | ||
96 | transport_to_str(type)); | ||
97 | goto out_roce_custom; | ||
98 | } | ||
99 | bitmap_clear(roce_bitmap, port_num, 1); | ||
100 | usnic_dbg("Freeing port %hu for %s\n", port_num, | ||
101 | transport_to_str(type)); | ||
102 | out_roce_custom: | ||
103 | spin_unlock(&roce_bitmap_lock); | ||
104 | } else { | ||
105 | usnic_err("Freeing invalid port %hu for %d\n", port_num, type); | ||
106 | } | ||
107 | } | ||
108 | |||
109 | int usnic_transport_init(void) | ||
110 | { | ||
111 | roce_bitmap = kzalloc(ROCE_BITMAP_SZ, GFP_KERNEL); | ||
112 | if (!roce_bitmap) { | ||
113 | usnic_err("Failed to allocate bit map"); | ||
114 | return -ENOMEM; | ||
115 | } | ||
116 | |||
117 | /* Do not ever allocate bit 0, hence set it here */ | ||
118 | bitmap_set(roce_bitmap, 0, 1); | ||
119 | return 0; | ||
120 | } | ||
121 | |||
122 | void usnic_transport_fini(void) | ||
123 | { | ||
124 | kfree(roce_bitmap); | ||
125 | } | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_transport.h b/drivers/infiniband/hw/usnic/usnic_transport.h new file mode 100644 index 000000000000..091fdaf4d25a --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_transport.h | |||
@@ -0,0 +1,28 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_TRANSPORT_H_ | ||
20 | #define USNIC_TRANSPORT_H_ | ||
21 | |||
22 | #include "usnic_abi.h" | ||
23 | |||
24 | u16 usnic_transport_rsrv_port(enum usnic_transport_type type, u16 port_num); | ||
25 | void usnic_transport_unrsrv_port(enum usnic_transport_type type, u16 port_num); | ||
26 | int usnic_transport_init(void); | ||
27 | void usnic_transport_fini(void); | ||
28 | #endif /* !USNIC_TRANSPORT_H */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_uiom.c b/drivers/infiniband/hw/usnic/usnic_uiom.c new file mode 100644 index 000000000000..c841a752dbd0 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_uiom.c | |||
@@ -0,0 +1,603 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2005 Topspin Communications. All rights reserved. | ||
3 | * Copyright (c) 2005 Mellanox Technologies. All rights reserved. | ||
4 | * Copyright (c) 2013 Cisco Systems. All rights reserved. | ||
5 | * | ||
6 | * This software is available to you under a choice of one of two | ||
7 | * licenses. You may choose to be licensed under the terms of the GNU | ||
8 | * General Public License (GPL) Version 2, available from the file | ||
9 | * COPYING in the main directory of this source tree, or the | ||
10 | * OpenIB.org BSD license below: | ||
11 | * | ||
12 | * Redistribution and use in source and binary forms, with or | ||
13 | * without modification, are permitted provided that the following | ||
14 | * conditions are met: | ||
15 | * | ||
16 | * - Redistributions of source code must retain the above | ||
17 | * copyright notice, this list of conditions and the following | ||
18 | * disclaimer. | ||
19 | * | ||
20 | * - Redistributions in binary form must reproduce the above | ||
21 | * copyright notice, this list of conditions and the following | ||
22 | * disclaimer in the documentation and/or other materials | ||
23 | * provided with the distribution. | ||
24 | * | ||
25 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
26 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
27 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
28 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
29 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
30 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
31 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
32 | * SOFTWARE. | ||
33 | */ | ||
34 | |||
35 | #include <linux/mm.h> | ||
36 | #include <linux/dma-mapping.h> | ||
37 | #include <linux/sched.h> | ||
38 | #include <linux/hugetlb.h> | ||
39 | #include <linux/dma-attrs.h> | ||
40 | #include <linux/iommu.h> | ||
41 | #include <linux/workqueue.h> | ||
42 | #include <linux/list.h> | ||
43 | #include <linux/pci.h> | ||
44 | |||
45 | #include "usnic_log.h" | ||
46 | #include "usnic_uiom.h" | ||
47 | #include "usnic_uiom_interval_tree.h" | ||
48 | |||
49 | static struct workqueue_struct *usnic_uiom_wq; | ||
50 | |||
51 | #define USNIC_UIOM_PAGE_CHUNK \ | ||
52 | ((PAGE_SIZE - offsetof(struct usnic_uiom_chunk, page_list)) /\ | ||
53 | ((void *) &((struct usnic_uiom_chunk *) 0)->page_list[1] - \ | ||
54 | (void *) &((struct usnic_uiom_chunk *) 0)->page_list[0])) | ||
55 | |||
56 | static void usnic_uiom_reg_account(struct work_struct *work) | ||
57 | { | ||
58 | struct usnic_uiom_reg *umem = container_of(work, | ||
59 | struct usnic_uiom_reg, work); | ||
60 | |||
61 | down_write(&umem->mm->mmap_sem); | ||
62 | umem->mm->locked_vm -= umem->diff; | ||
63 | up_write(&umem->mm->mmap_sem); | ||
64 | mmput(umem->mm); | ||
65 | kfree(umem); | ||
66 | } | ||
67 | |||
68 | static int usnic_uiom_dma_fault(struct iommu_domain *domain, | ||
69 | struct device *dev, | ||
70 | unsigned long iova, int flags, | ||
71 | void *token) | ||
72 | { | ||
73 | usnic_err("Device %s iommu fault domain 0x%pK va 0x%lx flags 0x%x\n", | ||
74 | dev_name(dev), | ||
75 | domain, iova, flags); | ||
76 | return -ENOSYS; | ||
77 | } | ||
78 | |||
79 | static void usnic_uiom_put_pages(struct list_head *chunk_list, int dirty) | ||
80 | { | ||
81 | struct usnic_uiom_chunk *chunk, *tmp; | ||
82 | struct page *page; | ||
83 | int i; | ||
84 | dma_addr_t pa; | ||
85 | |||
86 | list_for_each_entry_safe(chunk, tmp, chunk_list, list) { | ||
87 | for (i = 0; i < chunk->nents; i++) { | ||
88 | page = sg_page(&chunk->page_list[i]); | ||
89 | pa = sg_phys(&chunk->page_list[i]); | ||
90 | if (dirty) | ||
91 | set_page_dirty_lock(page); | ||
92 | put_page(page); | ||
93 | usnic_dbg("pa: %pa\n", &pa); | ||
94 | } | ||
95 | kfree(chunk); | ||
96 | } | ||
97 | } | ||
98 | |||
99 | static int usnic_uiom_get_pages(unsigned long addr, size_t size, int writable, | ||
100 | int dmasync, struct list_head *chunk_list) | ||
101 | { | ||
102 | struct page **page_list; | ||
103 | struct usnic_uiom_chunk *chunk; | ||
104 | unsigned long locked; | ||
105 | unsigned long lock_limit; | ||
106 | unsigned long cur_base; | ||
107 | unsigned long npages; | ||
108 | int ret; | ||
109 | int off; | ||
110 | int i; | ||
111 | int flags; | ||
112 | dma_addr_t pa; | ||
113 | DEFINE_DMA_ATTRS(attrs); | ||
114 | |||
115 | if (dmasync) | ||
116 | dma_set_attr(DMA_ATTR_WRITE_BARRIER, &attrs); | ||
117 | |||
118 | if (!can_do_mlock()) | ||
119 | return -EPERM; | ||
120 | |||
121 | INIT_LIST_HEAD(chunk_list); | ||
122 | |||
123 | page_list = (struct page **) __get_free_page(GFP_KERNEL); | ||
124 | if (!page_list) | ||
125 | return -ENOMEM; | ||
126 | |||
127 | npages = PAGE_ALIGN(size + (addr & ~PAGE_MASK)) >> PAGE_SHIFT; | ||
128 | |||
129 | down_write(¤t->mm->mmap_sem); | ||
130 | |||
131 | locked = npages + current->mm->locked_vm; | ||
132 | lock_limit = rlimit(RLIMIT_MEMLOCK) >> PAGE_SHIFT; | ||
133 | |||
134 | if ((locked > lock_limit) && !capable(CAP_IPC_LOCK)) { | ||
135 | ret = -ENOMEM; | ||
136 | goto out; | ||
137 | } | ||
138 | |||
139 | flags = IOMMU_READ | IOMMU_CACHE; | ||
140 | flags |= (writable) ? IOMMU_WRITE : 0; | ||
141 | cur_base = addr & PAGE_MASK; | ||
142 | ret = 0; | ||
143 | |||
144 | while (npages) { | ||
145 | ret = get_user_pages(current, current->mm, cur_base, | ||
146 | min_t(unsigned long, npages, | ||
147 | PAGE_SIZE / sizeof(struct page *)), | ||
148 | 1, !writable, page_list, NULL); | ||
149 | |||
150 | if (ret < 0) | ||
151 | goto out; | ||
152 | |||
153 | npages -= ret; | ||
154 | off = 0; | ||
155 | |||
156 | while (ret) { | ||
157 | chunk = kmalloc(sizeof(*chunk) + | ||
158 | sizeof(struct scatterlist) * | ||
159 | min_t(int, ret, USNIC_UIOM_PAGE_CHUNK), | ||
160 | GFP_KERNEL); | ||
161 | if (!chunk) { | ||
162 | ret = -ENOMEM; | ||
163 | goto out; | ||
164 | } | ||
165 | |||
166 | chunk->nents = min_t(int, ret, USNIC_UIOM_PAGE_CHUNK); | ||
167 | sg_init_table(chunk->page_list, chunk->nents); | ||
168 | for (i = 0; i < chunk->nents; ++i) { | ||
169 | sg_set_page(&chunk->page_list[i], | ||
170 | page_list[i + off], | ||
171 | PAGE_SIZE, 0); | ||
172 | pa = sg_phys(&chunk->page_list[i]); | ||
173 | usnic_dbg("va: 0x%lx pa: %pa\n", | ||
174 | cur_base + i*PAGE_SIZE, &pa); | ||
175 | } | ||
176 | cur_base += chunk->nents * PAGE_SIZE; | ||
177 | ret -= chunk->nents; | ||
178 | off += chunk->nents; | ||
179 | list_add_tail(&chunk->list, chunk_list); | ||
180 | } | ||
181 | |||
182 | ret = 0; | ||
183 | } | ||
184 | |||
185 | out: | ||
186 | if (ret < 0) | ||
187 | usnic_uiom_put_pages(chunk_list, 0); | ||
188 | else | ||
189 | current->mm->locked_vm = locked; | ||
190 | |||
191 | up_write(¤t->mm->mmap_sem); | ||
192 | free_page((unsigned long) page_list); | ||
193 | return ret; | ||
194 | } | ||
195 | |||
196 | static void usnic_uiom_unmap_sorted_intervals(struct list_head *intervals, | ||
197 | struct usnic_uiom_pd *pd) | ||
198 | { | ||
199 | struct usnic_uiom_interval_node *interval, *tmp; | ||
200 | long unsigned va, size; | ||
201 | |||
202 | list_for_each_entry_safe(interval, tmp, intervals, link) { | ||
203 | va = interval->start << PAGE_SHIFT; | ||
204 | size = ((interval->last - interval->start) + 1) << PAGE_SHIFT; | ||
205 | while (size > 0) { | ||
206 | /* Workaround for RH 970401 */ | ||
207 | usnic_dbg("va 0x%lx size 0x%lx", va, PAGE_SIZE); | ||
208 | iommu_unmap(pd->domain, va, PAGE_SIZE); | ||
209 | va += PAGE_SIZE; | ||
210 | size -= PAGE_SIZE; | ||
211 | } | ||
212 | } | ||
213 | } | ||
214 | |||
215 | static void __usnic_uiom_reg_release(struct usnic_uiom_pd *pd, | ||
216 | struct usnic_uiom_reg *uiomr, | ||
217 | int dirty) | ||
218 | { | ||
219 | int npages; | ||
220 | unsigned long vpn_start, vpn_last; | ||
221 | struct usnic_uiom_interval_node *interval, *tmp; | ||
222 | int writable = 0; | ||
223 | LIST_HEAD(rm_intervals); | ||
224 | |||
225 | npages = PAGE_ALIGN(uiomr->length + uiomr->offset) >> PAGE_SHIFT; | ||
226 | vpn_start = (uiomr->va & PAGE_MASK) >> PAGE_SHIFT; | ||
227 | vpn_last = vpn_start + npages - 1; | ||
228 | |||
229 | spin_lock(&pd->lock); | ||
230 | usnic_uiom_remove_interval(&pd->rb_root, vpn_start, | ||
231 | vpn_last, &rm_intervals); | ||
232 | usnic_uiom_unmap_sorted_intervals(&rm_intervals, pd); | ||
233 | |||
234 | list_for_each_entry_safe(interval, tmp, &rm_intervals, link) { | ||
235 | if (interval->flags & IOMMU_WRITE) | ||
236 | writable = 1; | ||
237 | list_del(&interval->link); | ||
238 | kfree(interval); | ||
239 | } | ||
240 | |||
241 | usnic_uiom_put_pages(&uiomr->chunk_list, dirty & writable); | ||
242 | spin_unlock(&pd->lock); | ||
243 | } | ||
244 | |||
245 | static int usnic_uiom_map_sorted_intervals(struct list_head *intervals, | ||
246 | struct usnic_uiom_reg *uiomr) | ||
247 | { | ||
248 | int i, err; | ||
249 | size_t size; | ||
250 | struct usnic_uiom_chunk *chunk; | ||
251 | struct usnic_uiom_interval_node *interval_node; | ||
252 | dma_addr_t pa; | ||
253 | dma_addr_t pa_start = 0; | ||
254 | dma_addr_t pa_end = 0; | ||
255 | long int va_start = -EINVAL; | ||
256 | struct usnic_uiom_pd *pd = uiomr->pd; | ||
257 | long int va = uiomr->va & PAGE_MASK; | ||
258 | int flags = IOMMU_READ | IOMMU_CACHE; | ||
259 | |||
260 | flags |= (uiomr->writable) ? IOMMU_WRITE : 0; | ||
261 | chunk = list_first_entry(&uiomr->chunk_list, struct usnic_uiom_chunk, | ||
262 | list); | ||
263 | list_for_each_entry(interval_node, intervals, link) { | ||
264 | iter_chunk: | ||
265 | for (i = 0; i < chunk->nents; i++, va += PAGE_SIZE) { | ||
266 | pa = sg_phys(&chunk->page_list[i]); | ||
267 | if ((va >> PAGE_SHIFT) < interval_node->start) | ||
268 | continue; | ||
269 | |||
270 | if ((va >> PAGE_SHIFT) == interval_node->start) { | ||
271 | /* First page of the interval */ | ||
272 | va_start = va; | ||
273 | pa_start = pa; | ||
274 | pa_end = pa; | ||
275 | } | ||
276 | |||
277 | WARN_ON(va_start == -EINVAL); | ||
278 | |||
279 | if ((pa_end + PAGE_SIZE != pa) && | ||
280 | (pa != pa_start)) { | ||
281 | /* PAs are not contiguous */ | ||
282 | size = pa_end - pa_start + PAGE_SIZE; | ||
283 | usnic_dbg("va 0x%lx pa %pa size 0x%zx flags 0x%x", | ||
284 | va_start, &pa_start, size, flags); | ||
285 | err = iommu_map(pd->domain, va_start, pa_start, | ||
286 | size, flags); | ||
287 | if (err) { | ||
288 | usnic_err("Failed to map va 0x%lx pa 0x%pa size 0x%zx with err %d\n", | ||
289 | va_start, &pa_start, size, err); | ||
290 | goto err_out; | ||
291 | } | ||
292 | va_start = va; | ||
293 | pa_start = pa; | ||
294 | pa_end = pa; | ||
295 | } | ||
296 | |||
297 | if ((va >> PAGE_SHIFT) == interval_node->last) { | ||
298 | /* Last page of the interval */ | ||
299 | size = pa - pa_start + PAGE_SIZE; | ||
300 | usnic_dbg("va 0x%lx pa %pa size 0x%zx flags 0x%x\n", | ||
301 | va_start, &pa_start, size, flags); | ||
302 | err = iommu_map(pd->domain, va_start, pa_start, | ||
303 | size, flags); | ||
304 | if (err) { | ||
305 | usnic_err("Failed to map va 0x%lx pa %pa size 0x%zx with err %d\n", | ||
306 | va_start, &pa_start, size, err); | ||
307 | goto err_out; | ||
308 | } | ||
309 | break; | ||
310 | } | ||
311 | |||
312 | if (pa != pa_start) | ||
313 | pa_end += PAGE_SIZE; | ||
314 | } | ||
315 | |||
316 | if (i == chunk->nents) { | ||
317 | /* | ||
318 | * Hit last entry of the chunk, | ||
319 | * hence advance to next chunk | ||
320 | */ | ||
321 | chunk = list_first_entry(&chunk->list, | ||
322 | struct usnic_uiom_chunk, | ||
323 | list); | ||
324 | goto iter_chunk; | ||
325 | } | ||
326 | } | ||
327 | |||
328 | return 0; | ||
329 | |||
330 | err_out: | ||
331 | usnic_uiom_unmap_sorted_intervals(intervals, pd); | ||
332 | return err; | ||
333 | } | ||
334 | |||
335 | struct usnic_uiom_reg *usnic_uiom_reg_get(struct usnic_uiom_pd *pd, | ||
336 | unsigned long addr, size_t size, | ||
337 | int writable, int dmasync) | ||
338 | { | ||
339 | struct usnic_uiom_reg *uiomr; | ||
340 | unsigned long va_base, vpn_start, vpn_last; | ||
341 | unsigned long npages; | ||
342 | int offset, err; | ||
343 | LIST_HEAD(sorted_diff_intervals); | ||
344 | |||
345 | /* | ||
346 | * Intel IOMMU map throws an error if a translation entry is | ||
347 | * changed from read to write. This module may not unmap | ||
348 | * and then remap the entry after fixing the permission | ||
349 | * b/c this open up a small windows where hw DMA may page fault | ||
350 | * Hence, make all entries to be writable. | ||
351 | */ | ||
352 | writable = 1; | ||
353 | |||
354 | va_base = addr & PAGE_MASK; | ||
355 | offset = addr & ~PAGE_MASK; | ||
356 | npages = PAGE_ALIGN(size + offset) >> PAGE_SHIFT; | ||
357 | vpn_start = (addr & PAGE_MASK) >> PAGE_SHIFT; | ||
358 | vpn_last = vpn_start + npages - 1; | ||
359 | |||
360 | uiomr = kmalloc(sizeof(*uiomr), GFP_KERNEL); | ||
361 | if (!uiomr) | ||
362 | return ERR_PTR(-ENOMEM); | ||
363 | |||
364 | uiomr->va = va_base; | ||
365 | uiomr->offset = offset; | ||
366 | uiomr->length = size; | ||
367 | uiomr->writable = writable; | ||
368 | uiomr->pd = pd; | ||
369 | |||
370 | err = usnic_uiom_get_pages(addr, size, writable, dmasync, | ||
371 | &uiomr->chunk_list); | ||
372 | if (err) { | ||
373 | usnic_err("Failed get_pages vpn [0x%lx,0x%lx] err %d\n", | ||
374 | vpn_start, vpn_last, err); | ||
375 | goto out_free_uiomr; | ||
376 | } | ||
377 | |||
378 | spin_lock(&pd->lock); | ||
379 | err = usnic_uiom_get_intervals_diff(vpn_start, vpn_last, | ||
380 | (writable) ? IOMMU_WRITE : 0, | ||
381 | IOMMU_WRITE, | ||
382 | &pd->rb_root, | ||
383 | &sorted_diff_intervals); | ||
384 | if (err) { | ||
385 | usnic_err("Failed disjoint interval vpn [0x%lx,0x%lx] err %d\n", | ||
386 | vpn_start, vpn_last, err); | ||
387 | goto out_put_pages; | ||
388 | } | ||
389 | |||
390 | err = usnic_uiom_map_sorted_intervals(&sorted_diff_intervals, uiomr); | ||
391 | if (err) { | ||
392 | usnic_err("Failed map interval vpn [0x%lx,0x%lx] err %d\n", | ||
393 | vpn_start, vpn_last, err); | ||
394 | goto out_put_intervals; | ||
395 | |||
396 | } | ||
397 | |||
398 | err = usnic_uiom_insert_interval(&pd->rb_root, vpn_start, vpn_last, | ||
399 | (writable) ? IOMMU_WRITE : 0); | ||
400 | if (err) { | ||
401 | usnic_err("Failed insert interval vpn [0x%lx,0x%lx] err %d\n", | ||
402 | vpn_start, vpn_last, err); | ||
403 | goto out_unmap_intervals; | ||
404 | } | ||
405 | |||
406 | usnic_uiom_put_interval_set(&sorted_diff_intervals); | ||
407 | spin_unlock(&pd->lock); | ||
408 | |||
409 | return uiomr; | ||
410 | |||
411 | out_unmap_intervals: | ||
412 | usnic_uiom_unmap_sorted_intervals(&sorted_diff_intervals, pd); | ||
413 | out_put_intervals: | ||
414 | usnic_uiom_put_interval_set(&sorted_diff_intervals); | ||
415 | out_put_pages: | ||
416 | usnic_uiom_put_pages(&uiomr->chunk_list, 0); | ||
417 | spin_unlock(&pd->lock); | ||
418 | out_free_uiomr: | ||
419 | kfree(uiomr); | ||
420 | return ERR_PTR(err); | ||
421 | } | ||
422 | |||
423 | void usnic_uiom_reg_release(struct usnic_uiom_reg *uiomr, int closing) | ||
424 | { | ||
425 | struct mm_struct *mm; | ||
426 | unsigned long diff; | ||
427 | |||
428 | __usnic_uiom_reg_release(uiomr->pd, uiomr, 1); | ||
429 | |||
430 | mm = get_task_mm(current); | ||
431 | if (!mm) { | ||
432 | kfree(uiomr); | ||
433 | return; | ||
434 | } | ||
435 | |||
436 | diff = PAGE_ALIGN(uiomr->length + uiomr->offset) >> PAGE_SHIFT; | ||
437 | |||
438 | /* | ||
439 | * We may be called with the mm's mmap_sem already held. This | ||
440 | * can happen when a userspace munmap() is the call that drops | ||
441 | * the last reference to our file and calls our release | ||
442 | * method. If there are memory regions to destroy, we'll end | ||
443 | * up here and not be able to take the mmap_sem. In that case | ||
444 | * we defer the vm_locked accounting to the system workqueue. | ||
445 | */ | ||
446 | if (closing) { | ||
447 | if (!down_write_trylock(&mm->mmap_sem)) { | ||
448 | INIT_WORK(&uiomr->work, usnic_uiom_reg_account); | ||
449 | uiomr->mm = mm; | ||
450 | uiomr->diff = diff; | ||
451 | |||
452 | queue_work(usnic_uiom_wq, &uiomr->work); | ||
453 | return; | ||
454 | } | ||
455 | } else | ||
456 | down_write(&mm->mmap_sem); | ||
457 | |||
458 | current->mm->locked_vm -= diff; | ||
459 | up_write(&mm->mmap_sem); | ||
460 | mmput(mm); | ||
461 | kfree(uiomr); | ||
462 | } | ||
463 | |||
464 | struct usnic_uiom_pd *usnic_uiom_alloc_pd(void) | ||
465 | { | ||
466 | struct usnic_uiom_pd *pd; | ||
467 | void *domain; | ||
468 | |||
469 | pd = kzalloc(sizeof(*pd), GFP_KERNEL); | ||
470 | if (!pd) | ||
471 | return ERR_PTR(-ENOMEM); | ||
472 | |||
473 | pd->domain = domain = iommu_domain_alloc(&pci_bus_type); | ||
474 | if (IS_ERR_OR_NULL(domain)) { | ||
475 | usnic_err("Failed to allocate IOMMU domain with err %ld\n", | ||
476 | PTR_ERR(pd->domain)); | ||
477 | kfree(pd); | ||
478 | return ERR_PTR(domain ? PTR_ERR(domain) : -ENOMEM); | ||
479 | } | ||
480 | |||
481 | iommu_set_fault_handler(pd->domain, usnic_uiom_dma_fault, NULL); | ||
482 | |||
483 | spin_lock_init(&pd->lock); | ||
484 | INIT_LIST_HEAD(&pd->devs); | ||
485 | |||
486 | return pd; | ||
487 | } | ||
488 | |||
489 | void usnic_uiom_dealloc_pd(struct usnic_uiom_pd *pd) | ||
490 | { | ||
491 | iommu_domain_free(pd->domain); | ||
492 | kfree(pd); | ||
493 | } | ||
494 | |||
495 | int usnic_uiom_attach_dev_to_pd(struct usnic_uiom_pd *pd, struct device *dev) | ||
496 | { | ||
497 | struct usnic_uiom_dev *uiom_dev; | ||
498 | int err; | ||
499 | |||
500 | uiom_dev = kzalloc(sizeof(*uiom_dev), GFP_KERNEL); | ||
501 | if (!uiom_dev) | ||
502 | return -ENOMEM; | ||
503 | uiom_dev->dev = dev; | ||
504 | |||
505 | err = iommu_attach_device(pd->domain, dev); | ||
506 | if (err) | ||
507 | goto out_free_dev; | ||
508 | |||
509 | if (!iommu_domain_has_cap(pd->domain, IOMMU_CAP_CACHE_COHERENCY)) { | ||
510 | usnic_err("IOMMU of %s does not support cache coherency\n", | ||
511 | dev_name(dev)); | ||
512 | err = -EINVAL; | ||
513 | goto out_detach_device; | ||
514 | } | ||
515 | |||
516 | spin_lock(&pd->lock); | ||
517 | list_add_tail(&uiom_dev->link, &pd->devs); | ||
518 | pd->dev_cnt++; | ||
519 | spin_unlock(&pd->lock); | ||
520 | |||
521 | return 0; | ||
522 | |||
523 | out_detach_device: | ||
524 | iommu_detach_device(pd->domain, dev); | ||
525 | out_free_dev: | ||
526 | kfree(uiom_dev); | ||
527 | return err; | ||
528 | } | ||
529 | |||
530 | void usnic_uiom_detach_dev_from_pd(struct usnic_uiom_pd *pd, struct device *dev) | ||
531 | { | ||
532 | struct usnic_uiom_dev *uiom_dev; | ||
533 | int found = 0; | ||
534 | |||
535 | spin_lock(&pd->lock); | ||
536 | list_for_each_entry(uiom_dev, &pd->devs, link) { | ||
537 | if (uiom_dev->dev == dev) { | ||
538 | found = 1; | ||
539 | break; | ||
540 | } | ||
541 | } | ||
542 | |||
543 | if (!found) { | ||
544 | usnic_err("Unable to free dev %s - not found\n", | ||
545 | dev_name(dev)); | ||
546 | spin_unlock(&pd->lock); | ||
547 | return; | ||
548 | } | ||
549 | |||
550 | list_del(&uiom_dev->link); | ||
551 | pd->dev_cnt--; | ||
552 | spin_unlock(&pd->lock); | ||
553 | |||
554 | return iommu_detach_device(pd->domain, dev); | ||
555 | } | ||
556 | |||
557 | struct device **usnic_uiom_get_dev_list(struct usnic_uiom_pd *pd) | ||
558 | { | ||
559 | struct usnic_uiom_dev *uiom_dev; | ||
560 | struct device **devs; | ||
561 | int i = 0; | ||
562 | |||
563 | spin_lock(&pd->lock); | ||
564 | devs = kcalloc(pd->dev_cnt + 1, sizeof(*devs), GFP_ATOMIC); | ||
565 | if (!devs) { | ||
566 | devs = ERR_PTR(-ENOMEM); | ||
567 | goto out; | ||
568 | } | ||
569 | |||
570 | list_for_each_entry(uiom_dev, &pd->devs, link) { | ||
571 | devs[i++] = uiom_dev->dev; | ||
572 | } | ||
573 | out: | ||
574 | spin_unlock(&pd->lock); | ||
575 | return devs; | ||
576 | } | ||
577 | |||
578 | void usnic_uiom_free_dev_list(struct device **devs) | ||
579 | { | ||
580 | kfree(devs); | ||
581 | } | ||
582 | |||
583 | int usnic_uiom_init(char *drv_name) | ||
584 | { | ||
585 | if (!iommu_present(&pci_bus_type)) { | ||
586 | usnic_err("IOMMU required but not present or enabled. USNIC QPs will not function w/o enabling IOMMU\n"); | ||
587 | return -EPERM; | ||
588 | } | ||
589 | |||
590 | usnic_uiom_wq = create_workqueue(drv_name); | ||
591 | if (!usnic_uiom_wq) { | ||
592 | usnic_err("Unable to alloc wq for drv %s\n", drv_name); | ||
593 | return -ENOMEM; | ||
594 | } | ||
595 | |||
596 | return 0; | ||
597 | } | ||
598 | |||
599 | void usnic_uiom_fini(void) | ||
600 | { | ||
601 | flush_workqueue(usnic_uiom_wq); | ||
602 | destroy_workqueue(usnic_uiom_wq); | ||
603 | } | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_uiom.h b/drivers/infiniband/hw/usnic/usnic_uiom.h new file mode 100644 index 000000000000..70440996e8f2 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_uiom.h | |||
@@ -0,0 +1,80 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_UIOM_H_ | ||
20 | #define USNIC_UIOM_H_ | ||
21 | |||
22 | #include <linux/list.h> | ||
23 | #include <linux/scatterlist.h> | ||
24 | |||
25 | #include "usnic_uiom_interval_tree.h" | ||
26 | |||
27 | #define USNIC_UIOM_READ (1) | ||
28 | #define USNIC_UIOM_WRITE (2) | ||
29 | |||
30 | #define USNIC_UIOM_MAX_PD_CNT (1000) | ||
31 | #define USNIC_UIOM_MAX_MR_CNT (1000000) | ||
32 | #define USNIC_UIOM_MAX_MR_SIZE (~0UL) | ||
33 | #define USNIC_UIOM_PAGE_SIZE (PAGE_SIZE) | ||
34 | |||
35 | struct usnic_uiom_dev { | ||
36 | struct device *dev; | ||
37 | struct list_head link; | ||
38 | }; | ||
39 | |||
40 | struct usnic_uiom_pd { | ||
41 | struct iommu_domain *domain; | ||
42 | spinlock_t lock; | ||
43 | struct rb_root rb_root; | ||
44 | struct list_head devs; | ||
45 | int dev_cnt; | ||
46 | }; | ||
47 | |||
48 | struct usnic_uiom_reg { | ||
49 | struct usnic_uiom_pd *pd; | ||
50 | unsigned long va; | ||
51 | size_t length; | ||
52 | int offset; | ||
53 | int page_size; | ||
54 | int writable; | ||
55 | struct list_head chunk_list; | ||
56 | struct work_struct work; | ||
57 | struct mm_struct *mm; | ||
58 | unsigned long diff; | ||
59 | }; | ||
60 | |||
61 | struct usnic_uiom_chunk { | ||
62 | struct list_head list; | ||
63 | int nents; | ||
64 | struct scatterlist page_list[0]; | ||
65 | }; | ||
66 | |||
67 | struct usnic_uiom_pd *usnic_uiom_alloc_pd(void); | ||
68 | void usnic_uiom_dealloc_pd(struct usnic_uiom_pd *pd); | ||
69 | int usnic_uiom_attach_dev_to_pd(struct usnic_uiom_pd *pd, struct device *dev); | ||
70 | void usnic_uiom_detach_dev_from_pd(struct usnic_uiom_pd *pd, | ||
71 | struct device *dev); | ||
72 | struct device **usnic_uiom_get_dev_list(struct usnic_uiom_pd *pd); | ||
73 | void usnic_uiom_free_dev_list(struct device **devs); | ||
74 | struct usnic_uiom_reg *usnic_uiom_reg_get(struct usnic_uiom_pd *pd, | ||
75 | unsigned long addr, size_t size, | ||
76 | int access, int dmasync); | ||
77 | void usnic_uiom_reg_release(struct usnic_uiom_reg *uiomr, int closing); | ||
78 | int usnic_uiom_init(char *drv_name); | ||
79 | void usnic_uiom_fini(void); | ||
80 | #endif /* USNIC_UIOM_H_ */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_uiom_interval_tree.c b/drivers/infiniband/hw/usnic/usnic_uiom_interval_tree.c new file mode 100644 index 000000000000..7e1dafccb11e --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_uiom_interval_tree.c | |||
@@ -0,0 +1,237 @@ | |||
1 | #include <linux/init.h> | ||
2 | #include <linux/list.h> | ||
3 | #include <linux/slab.h> | ||
4 | #include <linux/list_sort.h> | ||
5 | #include <linux/version.h> | ||
6 | |||
7 | #include <linux/interval_tree_generic.h> | ||
8 | #include "usnic_uiom_interval_tree.h" | ||
9 | |||
10 | #define START(node) ((node)->start) | ||
11 | #define LAST(node) ((node)->last) | ||
12 | |||
13 | #define MAKE_NODE(node, start, end, ref_cnt, flags, err, err_out) \ | ||
14 | do { \ | ||
15 | node = usnic_uiom_interval_node_alloc(start, \ | ||
16 | end, ref_cnt, flags); \ | ||
17 | if (!node) { \ | ||
18 | err = -ENOMEM; \ | ||
19 | goto err_out; \ | ||
20 | } \ | ||
21 | } while (0) | ||
22 | |||
23 | #define MARK_FOR_ADD(node, list) (list_add_tail(&node->link, list)) | ||
24 | |||
25 | #define MAKE_NODE_AND_APPEND(node, start, end, ref_cnt, flags, err, \ | ||
26 | err_out, list) \ | ||
27 | do { \ | ||
28 | MAKE_NODE(node, start, end, \ | ||
29 | ref_cnt, flags, err, \ | ||
30 | err_out); \ | ||
31 | MARK_FOR_ADD(node, list); \ | ||
32 | } while (0) | ||
33 | |||
34 | #define FLAGS_EQUAL(flags1, flags2, mask) \ | ||
35 | (((flags1) & (mask)) == ((flags2) & (mask))) | ||
36 | |||
37 | static struct usnic_uiom_interval_node* | ||
38 | usnic_uiom_interval_node_alloc(long int start, long int last, int ref_cnt, | ||
39 | int flags) | ||
40 | { | ||
41 | struct usnic_uiom_interval_node *interval = kzalloc(sizeof(*interval), | ||
42 | GFP_ATOMIC); | ||
43 | if (!interval) | ||
44 | return NULL; | ||
45 | |||
46 | interval->start = start; | ||
47 | interval->last = last; | ||
48 | interval->flags = flags; | ||
49 | interval->ref_cnt = ref_cnt; | ||
50 | |||
51 | return interval; | ||
52 | } | ||
53 | |||
54 | static int interval_cmp(void *priv, struct list_head *a, struct list_head *b) | ||
55 | { | ||
56 | struct usnic_uiom_interval_node *node_a, *node_b; | ||
57 | |||
58 | node_a = list_entry(a, struct usnic_uiom_interval_node, link); | ||
59 | node_b = list_entry(b, struct usnic_uiom_interval_node, link); | ||
60 | |||
61 | /* long to int */ | ||
62 | if (node_a->start < node_b->start) | ||
63 | return -1; | ||
64 | else if (node_a->start > node_b->start) | ||
65 | return 1; | ||
66 | |||
67 | return 0; | ||
68 | } | ||
69 | |||
70 | static void | ||
71 | find_intervals_intersection_sorted(struct rb_root *root, unsigned long start, | ||
72 | unsigned long last, | ||
73 | struct list_head *list) | ||
74 | { | ||
75 | struct usnic_uiom_interval_node *node; | ||
76 | |||
77 | INIT_LIST_HEAD(list); | ||
78 | |||
79 | for (node = usnic_uiom_interval_tree_iter_first(root, start, last); | ||
80 | node; | ||
81 | node = usnic_uiom_interval_tree_iter_next(node, start, last)) | ||
82 | list_add_tail(&node->link, list); | ||
83 | |||
84 | list_sort(NULL, list, interval_cmp); | ||
85 | } | ||
86 | |||
87 | int usnic_uiom_get_intervals_diff(unsigned long start, unsigned long last, | ||
88 | int flags, int flag_mask, | ||
89 | struct rb_root *root, | ||
90 | struct list_head *diff_set) | ||
91 | { | ||
92 | struct usnic_uiom_interval_node *interval, *tmp; | ||
93 | int err = 0; | ||
94 | long int pivot = start; | ||
95 | LIST_HEAD(intersection_set); | ||
96 | |||
97 | INIT_LIST_HEAD(diff_set); | ||
98 | |||
99 | find_intervals_intersection_sorted(root, start, last, | ||
100 | &intersection_set); | ||
101 | |||
102 | list_for_each_entry(interval, &intersection_set, link) { | ||
103 | if (pivot < interval->start) { | ||
104 | MAKE_NODE_AND_APPEND(tmp, pivot, interval->start - 1, | ||
105 | 1, flags, err, err_out, | ||
106 | diff_set); | ||
107 | pivot = interval->start; | ||
108 | } | ||
109 | |||
110 | /* | ||
111 | * Invariant: Set [start, pivot] is either in diff_set or root, | ||
112 | * but not in both. | ||
113 | */ | ||
114 | |||
115 | if (pivot > interval->last) { | ||
116 | continue; | ||
117 | } else if (pivot <= interval->last && | ||
118 | FLAGS_EQUAL(interval->flags, flags, | ||
119 | flag_mask)) { | ||
120 | pivot = interval->last + 1; | ||
121 | } | ||
122 | } | ||
123 | |||
124 | if (pivot <= last) | ||
125 | MAKE_NODE_AND_APPEND(tmp, pivot, last, 1, flags, err, err_out, | ||
126 | diff_set); | ||
127 | |||
128 | return 0; | ||
129 | |||
130 | err_out: | ||
131 | list_for_each_entry_safe(interval, tmp, diff_set, link) { | ||
132 | list_del(&interval->link); | ||
133 | kfree(interval); | ||
134 | } | ||
135 | |||
136 | return err; | ||
137 | } | ||
138 | |||
139 | void usnic_uiom_put_interval_set(struct list_head *intervals) | ||
140 | { | ||
141 | struct usnic_uiom_interval_node *interval, *tmp; | ||
142 | list_for_each_entry_safe(interval, tmp, intervals, link) | ||
143 | kfree(interval); | ||
144 | } | ||
145 | |||
146 | int usnic_uiom_insert_interval(struct rb_root *root, unsigned long start, | ||
147 | unsigned long last, int flags) | ||
148 | { | ||
149 | struct usnic_uiom_interval_node *interval, *tmp; | ||
150 | unsigned long istart, ilast; | ||
151 | int iref_cnt, iflags; | ||
152 | unsigned long lpivot = start; | ||
153 | int err = 0; | ||
154 | LIST_HEAD(to_add); | ||
155 | LIST_HEAD(intersection_set); | ||
156 | |||
157 | find_intervals_intersection_sorted(root, start, last, | ||
158 | &intersection_set); | ||
159 | |||
160 | list_for_each_entry(interval, &intersection_set, link) { | ||
161 | /* | ||
162 | * Invariant - lpivot is the left edge of next interval to be | ||
163 | * inserted | ||
164 | */ | ||
165 | istart = interval->start; | ||
166 | ilast = interval->last; | ||
167 | iref_cnt = interval->ref_cnt; | ||
168 | iflags = interval->flags; | ||
169 | |||
170 | if (istart < lpivot) { | ||
171 | MAKE_NODE_AND_APPEND(tmp, istart, lpivot - 1, iref_cnt, | ||
172 | iflags, err, err_out, &to_add); | ||
173 | } else if (istart > lpivot) { | ||
174 | MAKE_NODE_AND_APPEND(tmp, lpivot, istart - 1, 1, flags, | ||
175 | err, err_out, &to_add); | ||
176 | lpivot = istart; | ||
177 | } else { | ||
178 | lpivot = istart; | ||
179 | } | ||
180 | |||
181 | if (ilast > last) { | ||
182 | MAKE_NODE_AND_APPEND(tmp, lpivot, last, iref_cnt + 1, | ||
183 | iflags | flags, err, err_out, | ||
184 | &to_add); | ||
185 | MAKE_NODE_AND_APPEND(tmp, last + 1, ilast, iref_cnt, | ||
186 | iflags, err, err_out, &to_add); | ||
187 | } else { | ||
188 | MAKE_NODE_AND_APPEND(tmp, lpivot, ilast, iref_cnt + 1, | ||
189 | iflags | flags, err, err_out, | ||
190 | &to_add); | ||
191 | } | ||
192 | |||
193 | lpivot = ilast + 1; | ||
194 | } | ||
195 | |||
196 | if (lpivot <= last) | ||
197 | MAKE_NODE_AND_APPEND(tmp, lpivot, last, 1, flags, err, err_out, | ||
198 | &to_add); | ||
199 | |||
200 | list_for_each_entry_safe(interval, tmp, &intersection_set, link) { | ||
201 | usnic_uiom_interval_tree_remove(interval, root); | ||
202 | kfree(interval); | ||
203 | } | ||
204 | |||
205 | list_for_each_entry(interval, &to_add, link) | ||
206 | usnic_uiom_interval_tree_insert(interval, root); | ||
207 | |||
208 | return 0; | ||
209 | |||
210 | err_out: | ||
211 | list_for_each_entry_safe(interval, tmp, &to_add, link) | ||
212 | kfree(interval); | ||
213 | |||
214 | return err; | ||
215 | } | ||
216 | |||
217 | void usnic_uiom_remove_interval(struct rb_root *root, unsigned long start, | ||
218 | unsigned long last, struct list_head *removed) | ||
219 | { | ||
220 | struct usnic_uiom_interval_node *interval; | ||
221 | |||
222 | for (interval = usnic_uiom_interval_tree_iter_first(root, start, last); | ||
223 | interval; | ||
224 | interval = usnic_uiom_interval_tree_iter_next(interval, | ||
225 | start, | ||
226 | last)) { | ||
227 | if (--interval->ref_cnt == 0) | ||
228 | list_add_tail(&interval->link, removed); | ||
229 | } | ||
230 | |||
231 | list_for_each_entry(interval, removed, link) | ||
232 | usnic_uiom_interval_tree_remove(interval, root); | ||
233 | } | ||
234 | |||
235 | INTERVAL_TREE_DEFINE(struct usnic_uiom_interval_node, rb, | ||
236 | unsigned long, __subtree_last, | ||
237 | START, LAST, , usnic_uiom_interval_tree) | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_uiom_interval_tree.h b/drivers/infiniband/hw/usnic/usnic_uiom_interval_tree.h new file mode 100644 index 000000000000..030ba6e68a52 --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_uiom_interval_tree.h | |||
@@ -0,0 +1,74 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_UIOM_INTERVAL_TREE_H_ | ||
20 | #define USNIC_UIOM_INTERVAL_TREE_H_ | ||
21 | |||
22 | #include <linux/version.h> | ||
23 | #include <linux/rbtree.h> | ||
24 | |||
25 | struct usnic_uiom_interval_node { | ||
26 | struct rb_node rb; | ||
27 | struct list_head link; | ||
28 | unsigned long start; | ||
29 | unsigned long last; | ||
30 | unsigned long __subtree_last; | ||
31 | unsigned int ref_cnt; | ||
32 | int flags; | ||
33 | }; | ||
34 | |||
35 | extern void | ||
36 | usnic_uiom_interval_tree_insert(struct usnic_uiom_interval_node *node, | ||
37 | struct rb_root *root); | ||
38 | extern void | ||
39 | usnic_uiom_interval_tree_remove(struct usnic_uiom_interval_node *node, | ||
40 | struct rb_root *root); | ||
41 | extern struct usnic_uiom_interval_node * | ||
42 | usnic_uiom_interval_tree_iter_first(struct rb_root *root, | ||
43 | unsigned long start, | ||
44 | unsigned long last); | ||
45 | extern struct usnic_uiom_interval_node * | ||
46 | usnic_uiom_interval_tree_iter_next(struct usnic_uiom_interval_node *node, | ||
47 | unsigned long start, unsigned long last); | ||
48 | /* | ||
49 | * Inserts {start...last} into {root}. If there are overlaps, | ||
50 | * nodes will be broken up and merged | ||
51 | */ | ||
52 | int usnic_uiom_insert_interval(struct rb_root *root, | ||
53 | unsigned long start, unsigned long last, | ||
54 | int flags); | ||
55 | /* | ||
56 | * Removed {start...last} from {root}. The nodes removed are returned in | ||
57 | * 'removed.' The caller is responsibile for freeing memory of nodes in | ||
58 | * 'removed.' | ||
59 | */ | ||
60 | void usnic_uiom_remove_interval(struct rb_root *root, | ||
61 | unsigned long start, unsigned long last, | ||
62 | struct list_head *removed); | ||
63 | /* | ||
64 | * Returns {start...last} - {root} (relative complement of {start...last} in | ||
65 | * {root}) in diff_set sorted ascendingly | ||
66 | */ | ||
67 | int usnic_uiom_get_intervals_diff(unsigned long start, | ||
68 | unsigned long last, int flags, | ||
69 | int flag_mask, | ||
70 | struct rb_root *root, | ||
71 | struct list_head *diff_set); | ||
72 | /* Call this to free diff_set returned by usnic_uiom_get_intervals_diff */ | ||
73 | void usnic_uiom_put_interval_set(struct list_head *intervals); | ||
74 | #endif /* USNIC_UIOM_INTERVAL_TREE_H_ */ | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_vnic.c b/drivers/infiniband/hw/usnic/usnic_vnic.c new file mode 100644 index 000000000000..656b88c39eda --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_vnic.c | |||
@@ -0,0 +1,467 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | #include <linux/errno.h> | ||
19 | #include <linux/module.h> | ||
20 | #include <linux/pci.h> | ||
21 | |||
22 | #include "usnic_ib.h" | ||
23 | #include "vnic_resource.h" | ||
24 | #include "usnic_log.h" | ||
25 | #include "usnic_vnic.h" | ||
26 | |||
27 | struct usnic_vnic { | ||
28 | struct vnic_dev *vdev; | ||
29 | struct vnic_dev_bar bar[PCI_NUM_RESOURCES]; | ||
30 | struct usnic_vnic_res_chunk chunks[USNIC_VNIC_RES_TYPE_MAX]; | ||
31 | spinlock_t res_lock; | ||
32 | }; | ||
33 | |||
34 | static enum vnic_res_type _to_vnic_res_type(enum usnic_vnic_res_type res_type) | ||
35 | { | ||
36 | #define DEFINE_USNIC_VNIC_RES_AT(usnic_vnic_res_t, vnic_res_type, desc, val) \ | ||
37 | vnic_res_type, | ||
38 | #define DEFINE_USNIC_VNIC_RES(usnic_vnic_res_t, vnic_res_type, desc) \ | ||
39 | vnic_res_type, | ||
40 | static enum vnic_res_type usnic_vnic_type_2_vnic_type[] = { | ||
41 | USNIC_VNIC_RES_TYPES}; | ||
42 | #undef DEFINE_USNIC_VNIC_RES | ||
43 | #undef DEFINE_USNIC_VNIC_RES_AT | ||
44 | |||
45 | if (res_type >= USNIC_VNIC_RES_TYPE_MAX) | ||
46 | return RES_TYPE_MAX; | ||
47 | |||
48 | return usnic_vnic_type_2_vnic_type[res_type]; | ||
49 | } | ||
50 | |||
51 | const char *usnic_vnic_res_type_to_str(enum usnic_vnic_res_type res_type) | ||
52 | { | ||
53 | #define DEFINE_USNIC_VNIC_RES_AT(usnic_vnic_res_t, vnic_res_type, desc, val) \ | ||
54 | desc, | ||
55 | #define DEFINE_USNIC_VNIC_RES(usnic_vnic_res_t, vnic_res_type, desc) \ | ||
56 | desc, | ||
57 | static const char * const usnic_vnic_res_type_desc[] = { | ||
58 | USNIC_VNIC_RES_TYPES}; | ||
59 | #undef DEFINE_USNIC_VNIC_RES | ||
60 | #undef DEFINE_USNIC_VNIC_RES_AT | ||
61 | |||
62 | if (res_type >= USNIC_VNIC_RES_TYPE_MAX) | ||
63 | return "unknown"; | ||
64 | |||
65 | return usnic_vnic_res_type_desc[res_type]; | ||
66 | |||
67 | } | ||
68 | |||
69 | const char *usnic_vnic_pci_name(struct usnic_vnic *vnic) | ||
70 | { | ||
71 | return pci_name(usnic_vnic_get_pdev(vnic)); | ||
72 | } | ||
73 | |||
74 | int usnic_vnic_dump(struct usnic_vnic *vnic, char *buf, | ||
75 | int buf_sz, | ||
76 | void *hdr_obj, | ||
77 | int (*printtitle)(void *, char*, int), | ||
78 | int (*printcols)(char *, int), | ||
79 | int (*printrow)(void *, char *, int)) | ||
80 | { | ||
81 | struct usnic_vnic_res_chunk *chunk; | ||
82 | struct usnic_vnic_res *res; | ||
83 | struct vnic_dev_bar *bar0; | ||
84 | int i, j, offset; | ||
85 | |||
86 | offset = 0; | ||
87 | bar0 = usnic_vnic_get_bar(vnic, 0); | ||
88 | offset += scnprintf(buf + offset, buf_sz - offset, | ||
89 | "VF:%hu BAR0 bus_addr=%pa vaddr=0x%p size=%ld ", | ||
90 | usnic_vnic_get_index(vnic), | ||
91 | &bar0->bus_addr, | ||
92 | bar0->vaddr, bar0->len); | ||
93 | if (printtitle) | ||
94 | offset += printtitle(hdr_obj, buf + offset, buf_sz - offset); | ||
95 | offset += scnprintf(buf + offset, buf_sz - offset, "\n"); | ||
96 | offset += scnprintf(buf + offset, buf_sz - offset, | ||
97 | "|RES\t|CTRL_PIN\t\t|IN_USE\t"); | ||
98 | if (printcols) | ||
99 | offset += printcols(buf + offset, buf_sz - offset); | ||
100 | offset += scnprintf(buf + offset, buf_sz - offset, "\n"); | ||
101 | |||
102 | spin_lock(&vnic->res_lock); | ||
103 | for (i = 0; i < ARRAY_SIZE(vnic->chunks); i++) { | ||
104 | chunk = &vnic->chunks[i]; | ||
105 | for (j = 0; j < chunk->cnt; j++) { | ||
106 | res = chunk->res[j]; | ||
107 | offset += scnprintf(buf + offset, buf_sz - offset, | ||
108 | "|%s[%u]\t|0x%p\t|%u\t", | ||
109 | usnic_vnic_res_type_to_str(res->type), | ||
110 | res->vnic_idx, res->ctrl, !!res->owner); | ||
111 | if (printrow) { | ||
112 | offset += printrow(res->owner, buf + offset, | ||
113 | buf_sz - offset); | ||
114 | } | ||
115 | offset += scnprintf(buf + offset, buf_sz - offset, | ||
116 | "\n"); | ||
117 | } | ||
118 | } | ||
119 | spin_unlock(&vnic->res_lock); | ||
120 | return offset; | ||
121 | } | ||
122 | |||
123 | void usnic_vnic_res_spec_update(struct usnic_vnic_res_spec *spec, | ||
124 | enum usnic_vnic_res_type trgt_type, | ||
125 | u16 cnt) | ||
126 | { | ||
127 | int i; | ||
128 | |||
129 | for (i = 0; i < USNIC_VNIC_RES_TYPE_MAX; i++) { | ||
130 | if (spec->resources[i].type == trgt_type) { | ||
131 | spec->resources[i].cnt = cnt; | ||
132 | return; | ||
133 | } | ||
134 | } | ||
135 | |||
136 | WARN_ON(1); | ||
137 | } | ||
138 | |||
139 | int usnic_vnic_res_spec_satisfied(const struct usnic_vnic_res_spec *min_spec, | ||
140 | struct usnic_vnic_res_spec *res_spec) | ||
141 | { | ||
142 | int found, i, j; | ||
143 | |||
144 | for (i = 0; i < USNIC_VNIC_RES_TYPE_MAX; i++) { | ||
145 | found = 0; | ||
146 | |||
147 | for (j = 0; j < USNIC_VNIC_RES_TYPE_MAX; j++) { | ||
148 | if (res_spec->resources[i].type != | ||
149 | min_spec->resources[i].type) | ||
150 | continue; | ||
151 | found = 1; | ||
152 | if (min_spec->resources[i].cnt > | ||
153 | res_spec->resources[i].cnt) | ||
154 | return -EINVAL; | ||
155 | break; | ||
156 | } | ||
157 | |||
158 | if (!found) | ||
159 | return -EINVAL; | ||
160 | } | ||
161 | return 0; | ||
162 | } | ||
163 | |||
164 | int usnic_vnic_spec_dump(char *buf, int buf_sz, | ||
165 | struct usnic_vnic_res_spec *res_spec) | ||
166 | { | ||
167 | enum usnic_vnic_res_type res_type; | ||
168 | int res_cnt; | ||
169 | int i; | ||
170 | int offset = 0; | ||
171 | |||
172 | for (i = 0; i < USNIC_VNIC_RES_TYPE_MAX; i++) { | ||
173 | res_type = res_spec->resources[i].type; | ||
174 | res_cnt = res_spec->resources[i].cnt; | ||
175 | offset += scnprintf(buf + offset, buf_sz - offset, | ||
176 | "Res: %s Cnt: %d ", | ||
177 | usnic_vnic_res_type_to_str(res_type), | ||
178 | res_cnt); | ||
179 | } | ||
180 | |||
181 | return offset; | ||
182 | } | ||
183 | |||
184 | int usnic_vnic_check_room(struct usnic_vnic *vnic, | ||
185 | struct usnic_vnic_res_spec *res_spec) | ||
186 | { | ||
187 | int i; | ||
188 | enum usnic_vnic_res_type res_type; | ||
189 | int res_cnt; | ||
190 | |||
191 | for (i = 0; i < USNIC_VNIC_RES_TYPE_MAX; i++) { | ||
192 | res_type = res_spec->resources[i].type; | ||
193 | res_cnt = res_spec->resources[i].cnt; | ||
194 | |||
195 | if (res_type == USNIC_VNIC_RES_TYPE_EOL) | ||
196 | break; | ||
197 | |||
198 | if (res_cnt > usnic_vnic_res_free_cnt(vnic, res_type)) | ||
199 | return -EBUSY; | ||
200 | } | ||
201 | |||
202 | return 0; | ||
203 | } | ||
204 | |||
205 | int usnic_vnic_res_cnt(struct usnic_vnic *vnic, | ||
206 | enum usnic_vnic_res_type type) | ||
207 | { | ||
208 | return vnic->chunks[type].cnt; | ||
209 | } | ||
210 | |||
211 | int usnic_vnic_res_free_cnt(struct usnic_vnic *vnic, | ||
212 | enum usnic_vnic_res_type type) | ||
213 | { | ||
214 | return vnic->chunks[type].free_cnt; | ||
215 | } | ||
216 | |||
217 | struct usnic_vnic_res_chunk * | ||
218 | usnic_vnic_get_resources(struct usnic_vnic *vnic, enum usnic_vnic_res_type type, | ||
219 | int cnt, void *owner) | ||
220 | { | ||
221 | struct usnic_vnic_res_chunk *src, *ret; | ||
222 | struct usnic_vnic_res *res; | ||
223 | int i; | ||
224 | |||
225 | if (usnic_vnic_res_free_cnt(vnic, type) < cnt || cnt < 1 || !owner) | ||
226 | return ERR_PTR(-EINVAL); | ||
227 | |||
228 | ret = kzalloc(sizeof(*ret), GFP_ATOMIC); | ||
229 | if (!ret) { | ||
230 | usnic_err("Failed to allocate chunk for %s - Out of memory\n", | ||
231 | usnic_vnic_pci_name(vnic)); | ||
232 | return ERR_PTR(-ENOMEM); | ||
233 | } | ||
234 | |||
235 | ret->res = kzalloc(sizeof(*(ret->res))*cnt, GFP_ATOMIC); | ||
236 | if (!ret->res) { | ||
237 | usnic_err("Failed to allocate resources for %s. Out of memory\n", | ||
238 | usnic_vnic_pci_name(vnic)); | ||
239 | kfree(ret); | ||
240 | return ERR_PTR(-ENOMEM); | ||
241 | } | ||
242 | |||
243 | spin_lock(&vnic->res_lock); | ||
244 | src = &vnic->chunks[type]; | ||
245 | for (i = 0; i < src->cnt && ret->cnt < cnt; i++) { | ||
246 | res = src->res[i]; | ||
247 | if (!res->owner) { | ||
248 | src->free_cnt--; | ||
249 | res->owner = owner; | ||
250 | ret->res[ret->cnt++] = res; | ||
251 | } | ||
252 | } | ||
253 | |||
254 | spin_unlock(&vnic->res_lock); | ||
255 | ret->type = type; | ||
256 | ret->vnic = vnic; | ||
257 | WARN_ON(ret->cnt != cnt); | ||
258 | |||
259 | return ret; | ||
260 | } | ||
261 | |||
262 | void usnic_vnic_put_resources(struct usnic_vnic_res_chunk *chunk) | ||
263 | { | ||
264 | |||
265 | struct usnic_vnic_res *res; | ||
266 | int i; | ||
267 | struct usnic_vnic *vnic = chunk->vnic; | ||
268 | |||
269 | spin_lock(&vnic->res_lock); | ||
270 | while ((i = --chunk->cnt) >= 0) { | ||
271 | res = chunk->res[i]; | ||
272 | chunk->res[i] = NULL; | ||
273 | res->owner = NULL; | ||
274 | vnic->chunks[res->type].free_cnt++; | ||
275 | } | ||
276 | spin_unlock(&vnic->res_lock); | ||
277 | |||
278 | kfree(chunk->res); | ||
279 | kfree(chunk); | ||
280 | } | ||
281 | |||
282 | u16 usnic_vnic_get_index(struct usnic_vnic *vnic) | ||
283 | { | ||
284 | return usnic_vnic_get_pdev(vnic)->devfn - 1; | ||
285 | } | ||
286 | |||
287 | static int usnic_vnic_alloc_res_chunk(struct usnic_vnic *vnic, | ||
288 | enum usnic_vnic_res_type type, | ||
289 | struct usnic_vnic_res_chunk *chunk) | ||
290 | { | ||
291 | int cnt, err, i; | ||
292 | struct usnic_vnic_res *res; | ||
293 | |||
294 | cnt = vnic_dev_get_res_count(vnic->vdev, _to_vnic_res_type(type)); | ||
295 | if (cnt < 1) | ||
296 | return -EINVAL; | ||
297 | |||
298 | chunk->cnt = chunk->free_cnt = cnt; | ||
299 | chunk->res = kzalloc(sizeof(*(chunk->res))*cnt, GFP_KERNEL); | ||
300 | if (!chunk->res) | ||
301 | return -ENOMEM; | ||
302 | |||
303 | for (i = 0; i < cnt; i++) { | ||
304 | res = kzalloc(sizeof(*res), GFP_KERNEL); | ||
305 | if (!res) { | ||
306 | err = -ENOMEM; | ||
307 | goto fail; | ||
308 | } | ||
309 | res->type = type; | ||
310 | res->vnic_idx = i; | ||
311 | res->vnic = vnic; | ||
312 | res->ctrl = vnic_dev_get_res(vnic->vdev, | ||
313 | _to_vnic_res_type(type), i); | ||
314 | chunk->res[i] = res; | ||
315 | } | ||
316 | |||
317 | chunk->vnic = vnic; | ||
318 | return 0; | ||
319 | fail: | ||
320 | for (i--; i >= 0; i--) | ||
321 | kfree(chunk->res[i]); | ||
322 | kfree(chunk->res); | ||
323 | return err; | ||
324 | } | ||
325 | |||
326 | static void usnic_vnic_free_res_chunk(struct usnic_vnic_res_chunk *chunk) | ||
327 | { | ||
328 | int i; | ||
329 | for (i = 0; i < chunk->cnt; i++) | ||
330 | kfree(chunk->res[i]); | ||
331 | kfree(chunk->res); | ||
332 | } | ||
333 | |||
334 | static int usnic_vnic_discover_resources(struct pci_dev *pdev, | ||
335 | struct usnic_vnic *vnic) | ||
336 | { | ||
337 | enum usnic_vnic_res_type res_type; | ||
338 | int i; | ||
339 | int err = 0; | ||
340 | |||
341 | for (i = 0; i < ARRAY_SIZE(vnic->bar); i++) { | ||
342 | if (!(pci_resource_flags(pdev, i) & IORESOURCE_MEM)) | ||
343 | continue; | ||
344 | vnic->bar[i].len = pci_resource_len(pdev, i); | ||
345 | vnic->bar[i].vaddr = pci_iomap(pdev, i, vnic->bar[i].len); | ||
346 | if (!vnic->bar[i].vaddr) { | ||
347 | usnic_err("Cannot memory-map BAR %d, aborting\n", | ||
348 | i); | ||
349 | err = -ENODEV; | ||
350 | goto out_clean_bar; | ||
351 | } | ||
352 | vnic->bar[i].bus_addr = pci_resource_start(pdev, i); | ||
353 | } | ||
354 | |||
355 | vnic->vdev = vnic_dev_register(NULL, pdev, pdev, vnic->bar, | ||
356 | ARRAY_SIZE(vnic->bar)); | ||
357 | if (!vnic->vdev) { | ||
358 | usnic_err("Failed to register device %s\n", | ||
359 | pci_name(pdev)); | ||
360 | err = -EINVAL; | ||
361 | goto out_clean_bar; | ||
362 | } | ||
363 | |||
364 | for (res_type = USNIC_VNIC_RES_TYPE_EOL + 1; | ||
365 | res_type < USNIC_VNIC_RES_TYPE_MAX; res_type++) { | ||
366 | err = usnic_vnic_alloc_res_chunk(vnic, res_type, | ||
367 | &vnic->chunks[res_type]); | ||
368 | if (err) { | ||
369 | usnic_err("Failed to alloc res %s with err %d\n", | ||
370 | usnic_vnic_res_type_to_str(res_type), | ||
371 | err); | ||
372 | goto out_clean_chunks; | ||
373 | } | ||
374 | } | ||
375 | |||
376 | return 0; | ||
377 | |||
378 | out_clean_chunks: | ||
379 | for (res_type--; res_type > USNIC_VNIC_RES_TYPE_EOL; res_type--) | ||
380 | usnic_vnic_free_res_chunk(&vnic->chunks[res_type]); | ||
381 | vnic_dev_unregister(vnic->vdev); | ||
382 | out_clean_bar: | ||
383 | for (i = 0; i < ARRAY_SIZE(vnic->bar); i++) { | ||
384 | if (!(pci_resource_flags(pdev, i) & IORESOURCE_MEM)) | ||
385 | continue; | ||
386 | if (!vnic->bar[i].vaddr) | ||
387 | break; | ||
388 | |||
389 | iounmap(vnic->bar[i].vaddr); | ||
390 | } | ||
391 | |||
392 | return err; | ||
393 | } | ||
394 | |||
395 | struct pci_dev *usnic_vnic_get_pdev(struct usnic_vnic *vnic) | ||
396 | { | ||
397 | return vnic_dev_get_pdev(vnic->vdev); | ||
398 | } | ||
399 | |||
400 | struct vnic_dev_bar *usnic_vnic_get_bar(struct usnic_vnic *vnic, | ||
401 | int bar_num) | ||
402 | { | ||
403 | return (bar_num < ARRAY_SIZE(vnic->bar)) ? &vnic->bar[bar_num] : NULL; | ||
404 | } | ||
405 | |||
406 | static void usnic_vnic_release_resources(struct usnic_vnic *vnic) | ||
407 | { | ||
408 | int i; | ||
409 | struct pci_dev *pdev; | ||
410 | enum usnic_vnic_res_type res_type; | ||
411 | |||
412 | pdev = usnic_vnic_get_pdev(vnic); | ||
413 | |||
414 | for (res_type = USNIC_VNIC_RES_TYPE_EOL + 1; | ||
415 | res_type < USNIC_VNIC_RES_TYPE_MAX; res_type++) | ||
416 | usnic_vnic_free_res_chunk(&vnic->chunks[res_type]); | ||
417 | |||
418 | vnic_dev_unregister(vnic->vdev); | ||
419 | |||
420 | for (i = 0; i < ARRAY_SIZE(vnic->bar); i++) { | ||
421 | if (!(pci_resource_flags(pdev, i) & IORESOURCE_MEM)) | ||
422 | continue; | ||
423 | iounmap(vnic->bar[i].vaddr); | ||
424 | } | ||
425 | } | ||
426 | |||
427 | struct usnic_vnic *usnic_vnic_alloc(struct pci_dev *pdev) | ||
428 | { | ||
429 | struct usnic_vnic *vnic; | ||
430 | int err = 0; | ||
431 | |||
432 | if (!pci_is_enabled(pdev)) { | ||
433 | usnic_err("PCI dev %s is disabled\n", pci_name(pdev)); | ||
434 | return ERR_PTR(-EINVAL); | ||
435 | } | ||
436 | |||
437 | vnic = kzalloc(sizeof(*vnic), GFP_KERNEL); | ||
438 | if (!vnic) { | ||
439 | usnic_err("Failed to alloc vnic for %s - out of memory\n", | ||
440 | pci_name(pdev)); | ||
441 | return ERR_PTR(-ENOMEM); | ||
442 | } | ||
443 | |||
444 | spin_lock_init(&vnic->res_lock); | ||
445 | |||
446 | err = usnic_vnic_discover_resources(pdev, vnic); | ||
447 | if (err) { | ||
448 | usnic_err("Failed to discover %s resources with err %d\n", | ||
449 | pci_name(pdev), err); | ||
450 | goto out_free_vnic; | ||
451 | } | ||
452 | |||
453 | usnic_dbg("Allocated vnic for %s\n", usnic_vnic_pci_name(vnic)); | ||
454 | |||
455 | return vnic; | ||
456 | |||
457 | out_free_vnic: | ||
458 | kfree(vnic); | ||
459 | |||
460 | return ERR_PTR(err); | ||
461 | } | ||
462 | |||
463 | void usnic_vnic_free(struct usnic_vnic *vnic) | ||
464 | { | ||
465 | usnic_vnic_release_resources(vnic); | ||
466 | kfree(vnic); | ||
467 | } | ||
diff --git a/drivers/infiniband/hw/usnic/usnic_vnic.h b/drivers/infiniband/hw/usnic/usnic_vnic.h new file mode 100644 index 000000000000..14d931a8829d --- /dev/null +++ b/drivers/infiniband/hw/usnic/usnic_vnic.h | |||
@@ -0,0 +1,103 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved. | ||
3 | * | ||
4 | * This program is free software; you may redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License as published by | ||
6 | * the Free Software Foundation; version 2 of the License. | ||
7 | * | ||
8 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
9 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
10 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
11 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | ||
12 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | ||
13 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||
14 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
15 | * SOFTWARE. | ||
16 | * | ||
17 | */ | ||
18 | |||
19 | #ifndef USNIC_VNIC_H_ | ||
20 | #define USNIC_VNIC_H_ | ||
21 | |||
22 | #include <linux/pci.h> | ||
23 | |||
24 | #include "vnic_dev.h" | ||
25 | |||
26 | /* =USNIC_VNIC_RES_TYPE= =VNIC_RES= =DESC= */ | ||
27 | #define USNIC_VNIC_RES_TYPES \ | ||
28 | DEFINE_USNIC_VNIC_RES_AT(EOL, RES_TYPE_EOL, "EOL", 0) \ | ||
29 | DEFINE_USNIC_VNIC_RES(WQ, RES_TYPE_WQ, "WQ") \ | ||
30 | DEFINE_USNIC_VNIC_RES(RQ, RES_TYPE_RQ, "RQ") \ | ||
31 | DEFINE_USNIC_VNIC_RES(CQ, RES_TYPE_CQ, "CQ") \ | ||
32 | DEFINE_USNIC_VNIC_RES(INTR, RES_TYPE_INTR_CTRL, "INT") \ | ||
33 | DEFINE_USNIC_VNIC_RES(MAX, RES_TYPE_MAX, "MAX")\ | ||
34 | |||
35 | #define DEFINE_USNIC_VNIC_RES_AT(usnic_vnic_res_t, vnic_res_type, desc, val) \ | ||
36 | USNIC_VNIC_RES_TYPE_##usnic_vnic_res_t = val, | ||
37 | #define DEFINE_USNIC_VNIC_RES(usnic_vnic_res_t, vnic_res_type, desc) \ | ||
38 | USNIC_VNIC_RES_TYPE_##usnic_vnic_res_t, | ||
39 | enum usnic_vnic_res_type { | ||
40 | USNIC_VNIC_RES_TYPES | ||
41 | }; | ||
42 | #undef DEFINE_USNIC_VNIC_RES | ||
43 | #undef DEFINE_USNIC_VNIC_RES_AT | ||
44 | |||
45 | struct usnic_vnic_res { | ||
46 | enum usnic_vnic_res_type type; | ||
47 | unsigned int vnic_idx; | ||
48 | struct usnic_vnic *vnic; | ||
49 | void __iomem *ctrl; | ||
50 | void *owner; | ||
51 | }; | ||
52 | |||
53 | struct usnic_vnic_res_chunk { | ||
54 | enum usnic_vnic_res_type type; | ||
55 | int cnt; | ||
56 | int free_cnt; | ||
57 | struct usnic_vnic_res **res; | ||
58 | struct usnic_vnic *vnic; | ||
59 | }; | ||
60 | |||
61 | struct usnic_vnic_res_desc { | ||
62 | enum usnic_vnic_res_type type; | ||
63 | uint16_t cnt; | ||
64 | }; | ||
65 | |||
66 | struct usnic_vnic_res_spec { | ||
67 | struct usnic_vnic_res_desc resources[USNIC_VNIC_RES_TYPE_MAX]; | ||
68 | }; | ||
69 | |||
70 | const char *usnic_vnic_res_type_to_str(enum usnic_vnic_res_type res_type); | ||
71 | const char *usnic_vnic_pci_name(struct usnic_vnic *vnic); | ||
72 | int usnic_vnic_dump(struct usnic_vnic *vnic, char *buf, int buf_sz, | ||
73 | void *hdr_obj, | ||
74 | int (*printtitle)(void *, char*, int), | ||
75 | int (*printcols)(char *, int), | ||
76 | int (*printrow)(void *, char *, int)); | ||
77 | void usnic_vnic_res_spec_update(struct usnic_vnic_res_spec *spec, | ||
78 | enum usnic_vnic_res_type trgt_type, | ||
79 | u16 cnt); | ||
80 | int usnic_vnic_res_spec_satisfied(const struct usnic_vnic_res_spec *min_spec, | ||
81 | struct usnic_vnic_res_spec *res_spec); | ||
82 | int usnic_vnic_spec_dump(char *buf, int buf_sz, | ||
83 | struct usnic_vnic_res_spec *res_spec); | ||
84 | int usnic_vnic_check_room(struct usnic_vnic *vnic, | ||
85 | struct usnic_vnic_res_spec *res_spec); | ||
86 | int usnic_vnic_res_cnt(struct usnic_vnic *vnic, | ||
87 | enum usnic_vnic_res_type type); | ||
88 | int usnic_vnic_res_free_cnt(struct usnic_vnic *vnic, | ||
89 | enum usnic_vnic_res_type type); | ||
90 | struct usnic_vnic_res_chunk * | ||
91 | usnic_vnic_get_resources(struct usnic_vnic *vnic, | ||
92 | enum usnic_vnic_res_type type, | ||
93 | int cnt, | ||
94 | void *owner); | ||
95 | void usnic_vnic_put_resources(struct usnic_vnic_res_chunk *chunk); | ||
96 | struct pci_dev *usnic_vnic_get_pdev(struct usnic_vnic *vnic); | ||
97 | struct vnic_dev_bar *usnic_vnic_get_bar(struct usnic_vnic *vnic, | ||
98 | int bar_num); | ||
99 | struct usnic_vnic *usnic_vnic_alloc(struct pci_dev *pdev); | ||
100 | void usnic_vnic_free(struct usnic_vnic *vnic); | ||
101 | u16 usnic_vnic_get_index(struct usnic_vnic *vnic); | ||
102 | |||
103 | #endif /*!USNIC_VNIC_H_*/ | ||