1*2ed98337SAymeric Wibo /*-
2*2ed98337SAymeric Wibo * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3*2ed98337SAymeric Wibo *
4*2ed98337SAymeric Wibo * Copyright (c) 2022 Scott Long
5*2ed98337SAymeric Wibo * All rights reserved.
6*2ed98337SAymeric Wibo *
7*2ed98337SAymeric Wibo * Redistribution and use in source and binary forms, with or without
8*2ed98337SAymeric Wibo * modification, are permitted provided that the following conditions
9*2ed98337SAymeric Wibo * are met:
10*2ed98337SAymeric Wibo * 1. Redistributions of source code must retain the above copyright
11*2ed98337SAymeric Wibo * notice, this list of conditions and the following disclaimer.
12*2ed98337SAymeric Wibo * 2. Redistributions in binary form must reproduce the above copyright
13*2ed98337SAymeric Wibo * notice, this list of conditions and the following disclaimer in the
14*2ed98337SAymeric Wibo * documentation and/or other materials provided with the distribution.
15*2ed98337SAymeric Wibo *
16*2ed98337SAymeric Wibo * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17*2ed98337SAymeric Wibo * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18*2ed98337SAymeric Wibo * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19*2ed98337SAymeric Wibo * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20*2ed98337SAymeric Wibo * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21*2ed98337SAymeric Wibo * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22*2ed98337SAymeric Wibo * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23*2ed98337SAymeric Wibo * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24*2ed98337SAymeric Wibo * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25*2ed98337SAymeric Wibo * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26*2ed98337SAymeric Wibo * SUCH DAMAGE.
27*2ed98337SAymeric Wibo *
28*2ed98337SAymeric Wibo * Thunderbolt 3 / Native Host Interface driver variables
29*2ed98337SAymeric Wibo *
30*2ed98337SAymeric Wibo * $FreeBSD$
31*2ed98337SAymeric Wibo */
32*2ed98337SAymeric Wibo
33*2ed98337SAymeric Wibo #ifndef _NHI_VAR
34*2ed98337SAymeric Wibo #define _NHI_VAR
35*2ed98337SAymeric Wibo
36*2ed98337SAymeric Wibo MALLOC_DECLARE(M_NHI);
37*2ed98337SAymeric Wibo
38*2ed98337SAymeric Wibo #define NHI_MSIX_MAX 32
39*2ed98337SAymeric Wibo #define NHI_RING0_TX_DEPTH 16
40*2ed98337SAymeric Wibo #define NHI_RING0_RX_DEPTH 16
41*2ed98337SAymeric Wibo #define NHI_DEFAULT_NUM_RINGS 1
42*2ed98337SAymeric Wibo #define NHI_MAX_NUM_RINGS 32 /* XXX 2? */
43*2ed98337SAymeric Wibo #define NHI_RING0_FRAME_SIZE 256
44*2ed98337SAymeric Wibo #define NHI_MAILBOX_TIMEOUT 15
45*2ed98337SAymeric Wibo
46*2ed98337SAymeric Wibo #define NHI_CMD_TIMEOUT 3 /* 3 seconds */
47*2ed98337SAymeric Wibo
48*2ed98337SAymeric Wibo struct nhi_softc;
49*2ed98337SAymeric Wibo struct nhi_ring_pair;
50*2ed98337SAymeric Wibo struct nhi_intr_tracker;
51*2ed98337SAymeric Wibo struct nhi_cmd_frame;
52*2ed98337SAymeric Wibo struct hcm_softc;
53*2ed98337SAymeric Wibo struct router_softc;
54*2ed98337SAymeric Wibo
55*2ed98337SAymeric Wibo struct nhi_cmd_frame {
56*2ed98337SAymeric Wibo TAILQ_ENTRY(nhi_cmd_frame) cm_link;
57*2ed98337SAymeric Wibo uint32_t *data;
58*2ed98337SAymeric Wibo bus_addr_t data_busaddr;
59*2ed98337SAymeric Wibo u_int req_len;
60*2ed98337SAymeric Wibo uint16_t flags;
61*2ed98337SAymeric Wibo #define CMD_MAPPED (1 << 0)
62*2ed98337SAymeric Wibo #define CMD_POLLED (1 << 1)
63*2ed98337SAymeric Wibo #define CMD_REQ_COMPLETE (1 << 2)
64*2ed98337SAymeric Wibo #define CMD_RESP_COMPLETE (1 << 3)
65*2ed98337SAymeric Wibo #define CMD_RESP_OVERRUN (1 << 4)
66*2ed98337SAymeric Wibo uint16_t retries;
67*2ed98337SAymeric Wibo uint16_t pdf;
68*2ed98337SAymeric Wibo uint16_t idx;
69*2ed98337SAymeric Wibo
70*2ed98337SAymeric Wibo void *context;
71*2ed98337SAymeric Wibo u_int timeout;
72*2ed98337SAymeric Wibo
73*2ed98337SAymeric Wibo uint32_t *resp_buffer;
74*2ed98337SAymeric Wibo u_int resp_len;
75*2ed98337SAymeric Wibo };
76*2ed98337SAymeric Wibo
77*2ed98337SAymeric Wibo #define NHI_RING_NAMELEN 16
78*2ed98337SAymeric Wibo struct nhi_ring_pair {
79*2ed98337SAymeric Wibo struct nhi_softc *sc;
80*2ed98337SAymeric Wibo
81*2ed98337SAymeric Wibo union nhi_ring_desc *tx_ring;
82*2ed98337SAymeric Wibo union nhi_ring_desc *rx_ring;
83*2ed98337SAymeric Wibo
84*2ed98337SAymeric Wibo uint16_t tx_pi;
85*2ed98337SAymeric Wibo uint16_t tx_ci;
86*2ed98337SAymeric Wibo uint16_t rx_pi;
87*2ed98337SAymeric Wibo uint16_t rx_ci;
88*2ed98337SAymeric Wibo
89*2ed98337SAymeric Wibo uint16_t rx_pici_reg;
90*2ed98337SAymeric Wibo uint16_t tx_pici_reg;
91*2ed98337SAymeric Wibo
92*2ed98337SAymeric Wibo struct nhi_cmd_frame **rx_cmd_ring;
93*2ed98337SAymeric Wibo struct nhi_cmd_frame **tx_cmd_ring;
94*2ed98337SAymeric Wibo
95*2ed98337SAymeric Wibo struct mtx mtx;
96*2ed98337SAymeric Wibo char name[NHI_RING_NAMELEN];
97*2ed98337SAymeric Wibo struct nhi_intr_tracker *tracker;
98*2ed98337SAymeric Wibo SLIST_ENTRY(nhi_ring_pair) ring_link;
99*2ed98337SAymeric Wibo
100*2ed98337SAymeric Wibo TAILQ_HEAD(, nhi_cmd_frame) tx_head;
101*2ed98337SAymeric Wibo TAILQ_HEAD(, nhi_cmd_frame) rx_head;
102*2ed98337SAymeric Wibo
103*2ed98337SAymeric Wibo uint16_t tx_ring_depth;
104*2ed98337SAymeric Wibo uint16_t tx_ring_mask;
105*2ed98337SAymeric Wibo uint16_t rx_ring_depth;
106*2ed98337SAymeric Wibo uint16_t rx_ring_mask;
107*2ed98337SAymeric Wibo uint16_t rx_buffer_size;
108*2ed98337SAymeric Wibo u_char ring_num;
109*2ed98337SAymeric Wibo
110*2ed98337SAymeric Wibo bus_dma_tag_t ring_dmat;
111*2ed98337SAymeric Wibo bus_dmamap_t ring_map;
112*2ed98337SAymeric Wibo void *ring;
113*2ed98337SAymeric Wibo bus_addr_t tx_ring_busaddr;
114*2ed98337SAymeric Wibo bus_addr_t rx_ring_busaddr;
115*2ed98337SAymeric Wibo
116*2ed98337SAymeric Wibo bus_dma_tag_t frames_dmat;
117*2ed98337SAymeric Wibo bus_dmamap_t frames_map;
118*2ed98337SAymeric Wibo void *frames;
119*2ed98337SAymeric Wibo bus_addr_t tx_frames_busaddr;
120*2ed98337SAymeric Wibo bus_addr_t rx_frames_busaddr;
121*2ed98337SAymeric Wibo };
122*2ed98337SAymeric Wibo
123*2ed98337SAymeric Wibo /* PDF-indexed array of dispatch routines for interrupts */
124*2ed98337SAymeric Wibo typedef void (nhi_ring_cb_t)(void *, union nhi_ring_desc *,
125*2ed98337SAymeric Wibo struct nhi_cmd_frame *);
126*2ed98337SAymeric Wibo struct nhi_pdf_dispatch {
127*2ed98337SAymeric Wibo nhi_ring_cb_t *cb;
128*2ed98337SAymeric Wibo void *context;
129*2ed98337SAymeric Wibo };
130*2ed98337SAymeric Wibo
131*2ed98337SAymeric Wibo struct nhi_intr_tracker {
132*2ed98337SAymeric Wibo struct nhi_softc *sc;
133*2ed98337SAymeric Wibo struct nhi_ring_pair *ring;
134*2ed98337SAymeric Wibo struct nhi_pdf_dispatch txpdf[16];
135*2ed98337SAymeric Wibo struct nhi_pdf_dispatch rxpdf[16];
136*2ed98337SAymeric Wibo u_int vector;
137*2ed98337SAymeric Wibo };
138*2ed98337SAymeric Wibo
139*2ed98337SAymeric Wibo struct nhi_softc {
140*2ed98337SAymeric Wibo device_t dev;
141*2ed98337SAymeric Wibo device_t ufp;
142*2ed98337SAymeric Wibo u_int debug;
143*2ed98337SAymeric Wibo u_int hwflags;
144*2ed98337SAymeric Wibo #define NHI_TYPE_UNKNOWN 0x00
145*2ed98337SAymeric Wibo #define NHI_TYPE_AR 0x01 /* Alpine Ridge */
146*2ed98337SAymeric Wibo #define NHI_TYPE_TR 0x02 /* Titan Ridge */
147*2ed98337SAymeric Wibo #define NHI_TYPE_ICL 0x03 /* IceLake */
148*2ed98337SAymeric Wibo #define NHI_TYPE_MR 0x04 /* Maple Ridge */
149*2ed98337SAymeric Wibo #define NHI_TYPE_ADL 0x05 /* AlderLake */
150*2ed98337SAymeric Wibo #define NHI_TYPE_USB4 0x0f
151*2ed98337SAymeric Wibo #define NHI_TYPE_MASK 0x0f
152*2ed98337SAymeric Wibo #define NHI_MBOX_BUSY 0x10
153*2ed98337SAymeric Wibo u_int caps;
154*2ed98337SAymeric Wibo #define NHI_CAP_ICM 0x01
155*2ed98337SAymeric Wibo #define NHI_CAP_HCM 0x02
156*2ed98337SAymeric Wibo #define NHI_USE_ICM(sc) ((sc)->caps & NHI_CAP_ICM)
157*2ed98337SAymeric Wibo #define NHI_USE_HCM(sc) ((sc)->caps & NHI_CAP_HCM)
158*2ed98337SAymeric Wibo struct hcm_softc *hcm;
159*2ed98337SAymeric Wibo struct router_softc *root_rsc;
160*2ed98337SAymeric Wibo
161*2ed98337SAymeric Wibo struct nhi_ring_pair *ring0;
162*2ed98337SAymeric Wibo struct nhi_intr_tracker *intr_trackers;
163*2ed98337SAymeric Wibo
164*2ed98337SAymeric Wibo uint16_t path_count;
165*2ed98337SAymeric Wibo uint16_t max_ring_count;
166*2ed98337SAymeric Wibo
167*2ed98337SAymeric Wibo struct mtx nhi_mtx;
168*2ed98337SAymeric Wibo SLIST_HEAD(, nhi_ring_pair) ring_list;
169*2ed98337SAymeric Wibo
170*2ed98337SAymeric Wibo int msix_count;
171*2ed98337SAymeric Wibo struct resource *irqs[NHI_MSIX_MAX];
172*2ed98337SAymeric Wibo void *intrhand[NHI_MSIX_MAX];
173*2ed98337SAymeric Wibo int irq_rid[NHI_MSIX_MAX];
174*2ed98337SAymeric Wibo struct resource *irq_pba;
175*2ed98337SAymeric Wibo int irq_pba_rid;
176*2ed98337SAymeric Wibo struct resource *irq_table;
177*2ed98337SAymeric Wibo int irq_table_rid;
178*2ed98337SAymeric Wibo
179*2ed98337SAymeric Wibo struct resource *regs_resource;
180*2ed98337SAymeric Wibo bus_space_handle_t regs_bhandle;
181*2ed98337SAymeric Wibo bus_space_tag_t regs_btag;
182*2ed98337SAymeric Wibo int regs_rid;
183*2ed98337SAymeric Wibo
184*2ed98337SAymeric Wibo bus_dma_tag_t parent_dmat;
185*2ed98337SAymeric Wibo
186*2ed98337SAymeric Wibo bus_dma_tag_t ring0_dmat;
187*2ed98337SAymeric Wibo bus_dmamap_t ring0_map;
188*2ed98337SAymeric Wibo void *ring0_frames;
189*2ed98337SAymeric Wibo bus_addr_t ring0_frames_busaddr;
190*2ed98337SAymeric Wibo struct nhi_cmd_frame *ring0_cmds;
191*2ed98337SAymeric Wibo
192*2ed98337SAymeric Wibo struct sysctl_ctx_list *sysctl_ctx;
193*2ed98337SAymeric Wibo struct sysctl_oid *sysctl_tree;
194*2ed98337SAymeric Wibo
195*2ed98337SAymeric Wibo struct intr_config_hook ich;
196*2ed98337SAymeric Wibo
197*2ed98337SAymeric Wibo uint8_t force_hcm;
198*2ed98337SAymeric Wibo #define NHI_FORCE_HCM_DEFAULT 0x00
199*2ed98337SAymeric Wibo #define NHI_FORCE_HCM_ON 0x01
200*2ed98337SAymeric Wibo #define NHI_FORCE_HCM_OFF 0x02
201*2ed98337SAymeric Wibo
202*2ed98337SAymeric Wibo uint8_t uuid[16];
203*2ed98337SAymeric Wibo uint8_t lc_uuid[16];
204*2ed98337SAymeric Wibo };
205*2ed98337SAymeric Wibo
206*2ed98337SAymeric Wibo struct nhi_dispatch {
207*2ed98337SAymeric Wibo uint8_t pdf;
208*2ed98337SAymeric Wibo nhi_ring_cb_t *cb;
209*2ed98337SAymeric Wibo void *context;
210*2ed98337SAymeric Wibo };
211*2ed98337SAymeric Wibo
212*2ed98337SAymeric Wibo #define NHI_IS_AR(sc) (((sc)->hwflags & NHI_TYPE_MASK) == NHI_TYPE_AR)
213*2ed98337SAymeric Wibo #define NHI_IS_TR(sc) (((sc)->hwflags & NHI_TYPE_MASK) == NHI_TYPE_TR)
214*2ed98337SAymeric Wibo #define NHI_IS_ICL(sc) (((sc)->hwflags & NHI_TYPE_MASK) == NHI_TYPE_ICL)
215*2ed98337SAymeric Wibo #define NHI_IS_USB4(sc) (((sc)->hwflags & NHI_TYPE_MASK) == NHI_TYPE_USB4)
216*2ed98337SAymeric Wibo
217*2ed98337SAymeric Wibo int nhi_pci_configure_interrupts(struct nhi_softc *sc);
218*2ed98337SAymeric Wibo void nhi_pci_enable_interrupt(struct nhi_ring_pair *r);
219*2ed98337SAymeric Wibo void nhi_pci_disable_interrupts(struct nhi_softc *sc);
220*2ed98337SAymeric Wibo int nhi_pci_get_uuid(struct nhi_softc *sc);
221*2ed98337SAymeric Wibo int nhi_read_lc_mailbox(struct nhi_softc *, u_int reg, uint32_t *val);
222*2ed98337SAymeric Wibo int nhi_write_lc_mailbox(struct nhi_softc *, u_int reg, uint32_t val);
223*2ed98337SAymeric Wibo
224*2ed98337SAymeric Wibo void nhi_get_tunables(struct nhi_softc *);
225*2ed98337SAymeric Wibo int nhi_attach(struct nhi_softc *);
226*2ed98337SAymeric Wibo int nhi_detach(struct nhi_softc *);
227*2ed98337SAymeric Wibo
228*2ed98337SAymeric Wibo struct nhi_cmd_frame * nhi_alloc_tx_frame(struct nhi_ring_pair *);
229*2ed98337SAymeric Wibo void nhi_free_tx_frame(struct nhi_ring_pair *, struct nhi_cmd_frame *);
230*2ed98337SAymeric Wibo
231*2ed98337SAymeric Wibo int nhi_inmail_cmd(struct nhi_softc *, uint32_t, uint32_t);
232*2ed98337SAymeric Wibo int nhi_outmail_cmd(struct nhi_softc *, uint32_t *);
233*2ed98337SAymeric Wibo
234*2ed98337SAymeric Wibo int nhi_tx_schedule(struct nhi_ring_pair *, struct nhi_cmd_frame *);
235*2ed98337SAymeric Wibo int nhi_tx_synchronous(struct nhi_ring_pair *, struct nhi_cmd_frame *);
236*2ed98337SAymeric Wibo void nhi_intr(void *);
237*2ed98337SAymeric Wibo
238*2ed98337SAymeric Wibo int nhi_register_pdf(struct nhi_ring_pair *, struct nhi_dispatch *,
239*2ed98337SAymeric Wibo struct nhi_dispatch *);
240*2ed98337SAymeric Wibo int nhi_deregister_pdf(struct nhi_ring_pair *, struct nhi_dispatch *,
241*2ed98337SAymeric Wibo struct nhi_dispatch *);
242*2ed98337SAymeric Wibo
243*2ed98337SAymeric Wibo /* Low level read/write MMIO registers */
244*2ed98337SAymeric Wibo static __inline uint32_t
nhi_read_reg(struct nhi_softc * sc,u_int offset)245*2ed98337SAymeric Wibo nhi_read_reg(struct nhi_softc *sc, u_int offset)
246*2ed98337SAymeric Wibo {
247*2ed98337SAymeric Wibo return (le32toh(bus_space_read_4(sc->regs_btag, sc->regs_bhandle,
248*2ed98337SAymeric Wibo offset)));
249*2ed98337SAymeric Wibo }
250*2ed98337SAymeric Wibo
251*2ed98337SAymeric Wibo static __inline void
nhi_write_reg(struct nhi_softc * sc,u_int offset,uint32_t val)252*2ed98337SAymeric Wibo nhi_write_reg(struct nhi_softc *sc, u_int offset, uint32_t val)
253*2ed98337SAymeric Wibo {
254*2ed98337SAymeric Wibo bus_space_write_4(sc->regs_btag, sc->regs_bhandle, offset,
255*2ed98337SAymeric Wibo htole32(val));
256*2ed98337SAymeric Wibo }
257*2ed98337SAymeric Wibo
258*2ed98337SAymeric Wibo static __inline struct nhi_cmd_frame *
nhi_alloc_tx_frame_locked(struct nhi_ring_pair * r)259*2ed98337SAymeric Wibo nhi_alloc_tx_frame_locked(struct nhi_ring_pair *r)
260*2ed98337SAymeric Wibo {
261*2ed98337SAymeric Wibo struct nhi_cmd_frame *cmd;
262*2ed98337SAymeric Wibo
263*2ed98337SAymeric Wibo if ((cmd = TAILQ_FIRST(&r->tx_head)) != NULL)
264*2ed98337SAymeric Wibo TAILQ_REMOVE(&r->tx_head, cmd, cm_link);
265*2ed98337SAymeric Wibo return (cmd);
266*2ed98337SAymeric Wibo }
267*2ed98337SAymeric Wibo
268*2ed98337SAymeric Wibo static __inline void
nhi_free_tx_frame_locked(struct nhi_ring_pair * r,struct nhi_cmd_frame * cmd)269*2ed98337SAymeric Wibo nhi_free_tx_frame_locked(struct nhi_ring_pair *r, struct nhi_cmd_frame *cmd)
270*2ed98337SAymeric Wibo {
271*2ed98337SAymeric Wibo /* Clear all flags except for MAPPED */
272*2ed98337SAymeric Wibo cmd->flags &= CMD_MAPPED;
273*2ed98337SAymeric Wibo cmd->resp_buffer = NULL;
274*2ed98337SAymeric Wibo TAILQ_INSERT_TAIL(&r->tx_head, cmd, cm_link);
275*2ed98337SAymeric Wibo }
276*2ed98337SAymeric Wibo
277*2ed98337SAymeric Wibo #endif /* _NHI_VAR */
278