blob: 6ee4cfbdb4a849632473d81cb8726d12d1eac168 [file] [log] [blame]
Ramon Fried663686d2019-04-27 11:15:21 +03001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * PCI Endpoint uclass
4 *
5 * Based on Linux PCI-EP driver written by
6 * Kishon Vijay Abraham I <kishon@ti.com>
7 *
8 * Copyright (c) 2019
9 * Written by Ramon Fried <ramon.fried@gmail.com>
10 */
11
Patrick Delaunay81313352021-04-27 11:02:19 +020012#define LOG_CATEGORY UCLASS_PCI_EP
13
Ramon Fried663686d2019-04-27 11:15:21 +030014#include <common.h>
15#include <dm.h>
16#include <errno.h>
Simon Glass3ba929a2020-10-30 21:38:53 -060017#include <asm/global_data.h>
Ramon Fried663686d2019-04-27 11:15:21 +030018#include <linux/log2.h>
19#include <pci_ep.h>
20
21DECLARE_GLOBAL_DATA_PTR;
22
23int pci_ep_write_header(struct udevice *dev, uint fn, struct pci_ep_header *hdr)
24{
25 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
26
27 if (!ops->write_header)
28 return -ENOSYS;
29
30 return ops->write_header(dev, fn, hdr);
31}
32
33int pci_ep_read_header(struct udevice *dev, uint fn, struct pci_ep_header *hdr)
34{
35 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
36
37 if (!ops->read_header)
38 return -ENOSYS;
39
40 return ops->read_header(dev, fn, hdr);
41}
42
43int pci_ep_set_bar(struct udevice *dev, uint func_no, struct pci_bar *ep_bar)
44{
45 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
46 int flags = ep_bar->flags;
47
48 /* Some basic bar validity checks */
Ramon Fried77990812019-07-31 11:04:26 +030049 if (ep_bar->barno > BAR_5 || ep_bar->barno < BAR_0)
Ramon Fried663686d2019-04-27 11:15:21 +030050 return -EINVAL;
51
52 if ((ep_bar->barno == BAR_5 &&
53 (flags & PCI_BASE_ADDRESS_MEM_TYPE_64)) ||
54 ((flags & PCI_BASE_ADDRESS_SPACE_IO) &&
55 (flags & PCI_BASE_ADDRESS_IO_MASK)) ||
56 (upper_32_bits(ep_bar->size) &&
57 !(flags & PCI_BASE_ADDRESS_MEM_TYPE_64)))
58 return -EINVAL;
59
60 if (!ops->set_bar)
61 return -ENOSYS;
62
63 return ops->set_bar(dev, func_no, ep_bar);
64}
65
66int pci_ep_read_bar(struct udevice *dev, uint func_no, struct pci_bar *ep_bar,
67 enum pci_barno barno)
68{
69 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
70
71 /* Some basic bar validity checks */
72 if (barno > BAR_5 || barno < BAR_0)
73 return -EINVAL;
74
75 if (!ops->read_bar)
76 return -ENOSYS;
77
78 return ops->read_bar(dev, func_no, ep_bar, barno);
79}
80
81int pci_ep_clear_bar(struct udevice *dev, uint func_num, enum pci_barno bar)
82{
83 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
84
85 if (!ops->clear_bar)
86 return -ENOSYS;
87
88 return ops->clear_bar(dev, func_num, bar);
89}
90
91int pci_ep_map_addr(struct udevice *dev, uint func_no, phys_addr_t addr,
92 u64 pci_addr, size_t size)
93{
94 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
95
96 if (!ops->map_addr)
97 return -ENOSYS;
98
99 return ops->map_addr(dev, func_no, addr, pci_addr, size);
100}
101
102int pci_ep_unmap_addr(struct udevice *dev, uint func_no, phys_addr_t addr)
103{
104 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
105
106 if (!ops->unmap_addr)
107 return -ENOSYS;
108
109 return ops->unmap_addr(dev, func_no, addr);
110}
111
112int pci_ep_set_msi(struct udevice *dev, uint func_no, uint interrupts)
113{
114 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
115 uint encode_int;
116
117 if (interrupts > 32)
118 return -EINVAL;
119
120 if (!ops->set_msi)
121 return -ENOSYS;
122
123 /* MSI spec permits allocation of
124 * only 1, 2, 4, 8, 16, 32 interrupts
125 */
126 encode_int = order_base_2(interrupts);
127
128 return ops->set_msi(dev, func_no, encode_int);
129}
130
131int pci_ep_get_msi(struct udevice *dev, uint func_no)
132{
133 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
134 int interrupt;
135
136 if (!ops->get_msi)
137 return -ENOSYS;
138
139 interrupt = ops->get_msi(dev, func_no);
140
141 if (interrupt < 0)
142 return 0;
143
144 /* Translate back from order base 2*/
145 interrupt = 1 << interrupt;
146
147 return interrupt;
148}
149
150int pci_ep_set_msix(struct udevice *dev, uint func_no, uint interrupts)
151{
152 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
153
154 if (interrupts < 1 || interrupts > 2048)
155 return -EINVAL;
156
157 if (!ops->set_msix)
158 return -ENOSYS;
159
160 return ops->set_msix(dev, func_no, interrupts - 1);
161}
162
163int pci_ep_get_msix(struct udevice *dev, uint func_no)
164{
165 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
166 int interrupt;
167
168 if (!ops->get_msix)
169 return -ENOSYS;
170
171 interrupt = ops->get_msix(dev, func_no);
172
173 if (interrupt < 0)
174 return 0;
175
176 return interrupt + 1;
177}
178
179int pci_ep_raise_irq(struct udevice *dev, uint func_no,
180 enum pci_ep_irq_type type, uint interrupt_num)
181{
182 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
183
184 if (!ops->raise_irq)
185 return -ENOSYS;
186
187 return ops->raise_irq(dev, func_no, type, interrupt_num);
188}
189
190int pci_ep_start(struct udevice *dev)
191{
192 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
193
194 if (!ops->start)
195 return -ENOSYS;
196
197 return ops->start(dev);
198}
199
200int pci_ep_stop(struct udevice *dev)
201{
202 struct pci_ep_ops *ops = pci_ep_get_ops(dev);
203
204 if (!ops->stop)
205 return -ENOSYS;
206
207 return ops->stop(dev);
208}
209
210UCLASS_DRIVER(pci_ep) = {
211 .id = UCLASS_PCI_EP,
212 .name = "pci_ep",
213 .flags = DM_UC_FLAG_SEQ_ALIAS,
214};
Xiaowei Bao97434b42020-07-09 23:31:34 +0800215
Ovidiu Panaita45c6e12020-11-28 10:43:11 +0200216int pci_ep_init(void)
Xiaowei Bao97434b42020-07-09 23:31:34 +0800217{
218 struct udevice *dev;
219
220 for (uclass_first_device_check(UCLASS_PCI_EP, &dev);
221 dev;
222 uclass_next_device_check(&dev)) {
223 ;
224 }
Ovidiu Panaita45c6e12020-11-28 10:43:11 +0200225
226 return 0;
Xiaowei Bao97434b42020-07-09 23:31:34 +0800227}