Lines Matching +full:ipa +full:- +full:ap +full:- +full:to +full:- +full:modem

1 // SPDX-License-Identifier: GPL-2.0
3 /* Copyright (c) 2012-2018, The Linux Foundation. All rights reserved.
4 * Copyright (C) 2018-2023 Linaro Ltd.
15 #include <linux/dma-mapping.h>
17 #include "ipa.h"
28 * DOC: IPA Filter and Route Tables
30 * The IPA has tables defined in its local (IPA-resident) memory that define
32 * endian 64-bit "slot" that holds the address of a rule definition. (The
38 * by all IPA hardware (IPA v4.2 doesn't support hashed tables).
41 * an object (such as a route or filter table) in IPA-resident memory must
42 * 128-byte aligned. An object in system memory (such as a route or filter
43 * rule) must be at an 8-byte aligned address. We currently only place
46 * A rule consists of a contiguous block of 32-bit values terminated with
51 * Each filter rule is associated with an AP or modem TX endpoint, though
52 * not all TX endpoints support filtering. The first 64-bit slot in a
55 * address of a filter rule in the memory following the bitmap. Until IPA
56 * v5.0, the low-order bit (bit 0) in this bitmap represents a special
57 * global filter, which applies to all traffic. Otherwise the position of
61 * removed starting at IPA v5.0. For IPA v5.0+, the endpoint bitmap
62 * position defines the endpoint ID--i.e. if bit 1 is set in the endpoint
63 * bitmap, endpoint 1 has a filter rule. Older versions of IPA represent
73 * The AP initializes all entries in a filter table to refer to a "zero"
74 * rule. Once initialized, the modem and AP update the entries for
75 * endpoints they "own" directly. Currently the AP does not use the IPA
79 * bitmap as defined prior to IPA v5.0.
81 * IPA Filter Table
82 * ----------------------
84 * |--------------------|
85 * 1st endpoint | 0x000123456789abc0 | DMA address for modem endpoint 2 rule
86 * |--------------------|
87 * 2nd endpoint | 0x000123456789abf0 | DMA address for AP endpoint 5 rule
88 * |--------------------|
90 * |--------------------|
92 * |--------------------|
94 * ----------------------
96 * The set of available route rules is divided about equally between the AP
97 * and modem. The AP initializes all entries in a route table to refer to
98 * a "zero entry". Once initialized, the modem and AP are responsible for
100 * though the AP currently does not use the IPA routing functionality.
102 * IPA Route Table
103 * ----------------------
104 * 1st modem route | 0x0001234500001100 | DMA address for first route rule
105 * |--------------------|
106 * 2nd modem route | 0x0001234500001140 | DMA address for second route rule
107 * |--------------------|
109 * |--------------------|
110 * Last modem route| 0x0001234500002280 | DMA address for Nth route rule
111 * |--------------------|
112 * 1st AP route | 0x0001234500001100 | DMA address for route rule (N+1)
113 * |--------------------|
114 * 2nd AP route | 0x0001234500001140 | DMA address for next route rule
115 * |--------------------|
117 * |--------------------|
118 * Last AP route | 0x0001234500002280 | DMA address for last route rule
119 * ----------------------
122 /* Filter or route rules consist of a set of 32-bit values followed by a
123 * 32-bit all-zero rule list terminator. The "zero rule" is simply an
124 * all-zero rule followed by the list terminator.
132 * to filter or route rules. But the size of a table entry in ipa_table_validate_build()
133 * is 64 bits regardless of what the size of an AP DMA address in ipa_table_validate_build()
135 * code in ipa_table_init() uses a pointer to __le64 to in ipa_table_validate_build()
140 /* A "zero rule" is used to represent no filtering or no routing. in ipa_table_validate_build()
141 * It is a 64-bit block of zeroed memory. Code in ipa_table_init() in ipa_table_validate_build()
142 * assumes that it can be written using a pointer to __le64. in ipa_table_validate_build()
148 ipa_table_mem(struct ipa *ipa, bool filter, bool hashed, bool ipv6) in ipa_table_mem() argument
161 return ipa_mem_find(ipa, mem_id); in ipa_table_mem()
164 bool ipa_filtered_valid(struct ipa *ipa, u64 filtered) in ipa_filtered_valid() argument
166 struct device *dev = &ipa->pdev->dev; in ipa_filtered_valid()
176 if (count > ipa->filter_count) { in ipa_filtered_valid()
178 count, ipa->filter_count); in ipa_filtered_valid()
187 static dma_addr_t ipa_table_addr(struct ipa *ipa, bool filter_mask, u16 count) in ipa_table_addr() argument
194 WARN_ON(count > max_t(u32, ipa->filter_count, ipa->route_count)); in ipa_table_addr()
199 return ipa->table_addr + skip * sizeof(*ipa->table_virt); in ipa_table_addr()
205 struct ipa *ipa = container_of(trans->gsi, struct ipa, gsi); in ipa_table_reset_add() local
211 /* Nothing to do if the memory region is doesn't exist or is empty */ in ipa_table_reset_add()
212 mem = ipa_table_mem(ipa, filter, hashed, ipv6); in ipa_table_reset_add()
213 if (!mem || !mem->size) in ipa_table_reset_add()
219 offset = mem->offset + first * sizeof(__le64); in ipa_table_reset_add()
221 addr = ipa_table_addr(ipa, false, count); in ipa_table_reset_add()
226 /* Reset entries in a single filter table belonging to either the AP or
227 * modem to refer to the zero entry. The memory region supplied will be
228 * for the IPv4 and IPv6 non-hashed and hashed filter tables.
231 ipa_filter_reset_table(struct ipa *ipa, bool hashed, bool ipv6, bool modem) in ipa_filter_reset_table() argument
233 u64 ep_mask = ipa->filtered; in ipa_filter_reset_table()
237 trans = ipa_cmd_trans_alloc(ipa, hweight64(ep_mask)); in ipa_filter_reset_table()
239 dev_err(&ipa->pdev->dev, in ipa_filter_reset_table()
241 modem ? "modem" : "AP"); in ipa_filter_reset_table()
242 return -EBUSY; in ipa_filter_reset_table()
245 ee_id = modem ? GSI_EE_MODEM : GSI_EE_AP; in ipa_filter_reset_table()
252 endpoint = &ipa->endpoint[endpoint_id]; in ipa_filter_reset_table()
253 if (endpoint->ee_id != ee_id) in ipa_filter_reset_table()
264 /* Theoretically, each filter table could have more filter slots to
268 static int ipa_filter_reset(struct ipa *ipa, bool modem) in ipa_filter_reset() argument
272 ret = ipa_filter_reset_table(ipa, false, false, modem); in ipa_filter_reset()
276 ret = ipa_filter_reset_table(ipa, false, true, modem); in ipa_filter_reset()
277 if (ret || !ipa_table_hash_support(ipa)) in ipa_filter_reset()
280 ret = ipa_filter_reset_table(ipa, true, false, modem); in ipa_filter_reset()
284 return ipa_filter_reset_table(ipa, true, true, modem); in ipa_filter_reset()
287 /* The AP routes and modem routes are each contiguous within the
289 * won't exceed the per-transaction command limit.
291 static int ipa_route_reset(struct ipa *ipa, bool modem) in ipa_route_reset() argument
293 bool hash_support = ipa_table_hash_support(ipa); in ipa_route_reset()
294 u32 modem_route_count = ipa->modem_route_count; in ipa_route_reset()
299 trans = ipa_cmd_trans_alloc(ipa, hash_support ? 4 : 2); in ipa_route_reset()
301 dev_err(&ipa->pdev->dev, in ipa_route_reset()
303 modem ? "modem" : "AP"); in ipa_route_reset()
304 return -EBUSY; in ipa_route_reset()
307 if (modem) { in ipa_route_reset()
312 count = ipa->route_count - modem_route_count; in ipa_route_reset()
328 void ipa_table_reset(struct ipa *ipa, bool modem) in ipa_table_reset() argument
330 struct device *dev = &ipa->pdev->dev; in ipa_table_reset()
334 ee_name = modem ? "modem" : "AP"; in ipa_table_reset()
337 ret = ipa_filter_reset(ipa, modem); in ipa_table_reset()
342 ret = ipa_route_reset(ipa, modem); in ipa_table_reset()
348 int ipa_table_hash_flush(struct ipa *ipa) in ipa_table_hash_flush() argument
354 if (!ipa_table_hash_support(ipa)) in ipa_table_hash_flush()
357 trans = ipa_cmd_trans_alloc(ipa, 1); in ipa_table_hash_flush()
359 dev_err(&ipa->pdev->dev, "no transaction for hash flush\n"); in ipa_table_hash_flush()
360 return -EBUSY; in ipa_table_hash_flush()
363 if (ipa->version < IPA_VERSION_5_0) { in ipa_table_hash_flush()
364 reg = ipa_reg(ipa, FILT_ROUT_HASH_FLUSH); in ipa_table_hash_flush()
371 reg = ipa_reg(ipa, FILT_ROUT_CACHE_FLUSH); in ipa_table_hash_flush()
373 /* IPA v5.0+ uses a unified cache (both IPv4 and IPv6) */ in ipa_table_hash_flush()
387 struct ipa *ipa = container_of(trans->gsi, struct ipa, gsi); in ipa_table_init_add() local
406 /* The non-hashed region will exist (see ipa_table_mem_valid()) */ in ipa_table_init_add()
407 mem = ipa_table_mem(ipa, filter, false, ipv6); in ipa_table_init_add()
408 hash_mem = ipa_table_mem(ipa, filter, true, ipv6); in ipa_table_init_add()
409 hash_offset = hash_mem ? hash_mem->offset : 0; in ipa_table_init_add()
411 /* Compute the number of table entries to initialize */ in ipa_table_init_add()
415 * to hold the bitmap itself. The size of the hashed filter in ipa_table_init_add()
416 * table is either the same as the non-hashed one, or zero. in ipa_table_init_add()
418 count = 1 + hweight64(ipa->filtered); in ipa_table_init_add()
419 hash_count = hash_mem && hash_mem->size ? count : 0; in ipa_table_init_add()
424 count = mem->size / sizeof(__le64); in ipa_table_init_add()
425 hash_count = hash_mem ? hash_mem->size / sizeof(__le64) : 0; in ipa_table_init_add()
430 addr = ipa_table_addr(ipa, filter, count); in ipa_table_init_add()
431 hash_addr = ipa_table_addr(ipa, filter, hash_count); in ipa_table_init_add()
433 ipa_cmd_table_init_add(trans, opcode, size, mem->offset, addr, in ipa_table_init_add()
439 zero_offset = mem->offset + size; in ipa_table_init_add()
440 zero_size = mem->size - size; in ipa_table_init_add()
442 ipa->zero_addr, true); in ipa_table_init_add()
448 zero_size = hash_mem->size - hash_size; in ipa_table_init_add()
450 ipa->zero_addr, true); in ipa_table_init_add()
453 int ipa_table_setup(struct ipa *ipa) in ipa_table_setup() argument
458 * - IPv4: in ipa_table_setup()
459 * - One for route table initialization (non-hashed and hashed) in ipa_table_setup()
460 * - One for filter table initialization (non-hashed and hashed) in ipa_table_setup()
461 * - One to zero unused entries in the non-hashed filter table in ipa_table_setup()
462 * - One to zero unused entries in the hashed filter table in ipa_table_setup()
463 * - IPv6: in ipa_table_setup()
464 * - One for route table initialization (non-hashed and hashed) in ipa_table_setup()
465 * - One for filter table initialization (non-hashed and hashed) in ipa_table_setup()
466 * - One to zero unused entries in the non-hashed filter table in ipa_table_setup()
467 * - One to zero unused entries in the hashed filter table in ipa_table_setup()
470 trans = ipa_cmd_trans_alloc(ipa, 8); in ipa_table_setup()
472 dev_err(&ipa->pdev->dev, "no transaction for table setup\n"); in ipa_table_setup()
473 return -EBUSY; in ipa_table_setup()
487 * ipa_filter_tuple_zero() - Zero an endpoint's hashed filter tuple
490 * Endpoint must be for the AP (not modem) and support filtering. Updates
495 u32 endpoint_id = endpoint->endpoint_id; in ipa_filter_tuple_zero()
496 struct ipa *ipa = endpoint->ipa; in ipa_filter_tuple_zero() local
501 if (ipa->version < IPA_VERSION_5_0) { in ipa_filter_tuple_zero()
502 reg = ipa_reg(ipa, ENDP_FILTER_ROUTER_HSH_CFG); in ipa_filter_tuple_zero()
505 val = ioread32(endpoint->ipa->reg_virt + offset); in ipa_filter_tuple_zero()
507 /* Zero all filter-related fields, preserving the rest */ in ipa_filter_tuple_zero()
510 /* IPA v5.0 separates filter and router cache configuration */ in ipa_filter_tuple_zero()
511 reg = ipa_reg(ipa, ENDP_FILTER_CACHE_CFG); in ipa_filter_tuple_zero()
514 /* Zero all filter-related fields */ in ipa_filter_tuple_zero()
518 iowrite32(val, endpoint->ipa->reg_virt + offset); in ipa_filter_tuple_zero()
522 static void ipa_filter_config(struct ipa *ipa, bool modem) in ipa_filter_config() argument
524 enum gsi_ee_id ee_id = modem ? GSI_EE_MODEM : GSI_EE_AP; in ipa_filter_config()
525 u64 ep_mask = ipa->filtered; in ipa_filter_config()
527 if (!ipa_table_hash_support(ipa)) in ipa_filter_config()
536 endpoint = &ipa->endpoint[endpoint_id]; in ipa_filter_config()
537 if (endpoint->ee_id == ee_id) in ipa_filter_config()
542 static bool ipa_route_id_modem(struct ipa *ipa, u32 route_id) in ipa_route_id_modem() argument
544 return route_id < ipa->modem_route_count; in ipa_route_id_modem()
548 * ipa_route_tuple_zero() - Zero a hashed route table entry tuple
549 * @ipa: IPA pointer
554 static void ipa_route_tuple_zero(struct ipa *ipa, u32 route_id) in ipa_route_tuple_zero() argument
560 if (ipa->version < IPA_VERSION_5_0) { in ipa_route_tuple_zero()
561 reg = ipa_reg(ipa, ENDP_FILTER_ROUTER_HSH_CFG); in ipa_route_tuple_zero()
564 val = ioread32(ipa->reg_virt + offset); in ipa_route_tuple_zero()
566 /* Zero all route-related fields, preserving the rest */ in ipa_route_tuple_zero()
569 /* IPA v5.0 separates filter and router cache configuration */ in ipa_route_tuple_zero()
570 reg = ipa_reg(ipa, ENDP_ROUTER_CACHE_CFG); in ipa_route_tuple_zero()
573 /* Zero all route-related fields */ in ipa_route_tuple_zero()
577 iowrite32(val, ipa->reg_virt + offset); in ipa_route_tuple_zero()
581 static void ipa_route_config(struct ipa *ipa, bool modem) in ipa_route_config() argument
585 if (!ipa_table_hash_support(ipa)) in ipa_route_config()
588 for (route_id = 0; route_id < ipa->route_count; route_id++) in ipa_route_config()
589 if (ipa_route_id_modem(ipa, route_id) == modem) in ipa_route_config()
590 ipa_route_tuple_zero(ipa, route_id); in ipa_route_config()
594 void ipa_table_config(struct ipa *ipa) in ipa_table_config() argument
596 ipa_filter_config(ipa, false); in ipa_table_config()
597 ipa_filter_config(ipa, true); in ipa_table_config()
598 ipa_route_config(ipa, false); in ipa_table_config()
599 ipa_route_config(ipa, true); in ipa_table_config()
602 /* Verify the sizes of all IPA table filter or routing table memory regions
605 bool ipa_table_mem_valid(struct ipa *ipa, bool filter) in ipa_table_mem_valid() argument
607 bool hash_support = ipa_table_hash_support(ipa); in ipa_table_mem_valid()
613 /* IPv4 and IPv6 non-hashed tables are expected to be defined and in ipa_table_mem_valid()
617 mem_ipv4 = ipa_table_mem(ipa, filter, false, false); in ipa_table_mem_valid()
621 mem_ipv6 = ipa_table_mem(ipa, filter, false, true); in ipa_table_mem_valid()
625 if (mem_ipv4->size != mem_ipv6->size) in ipa_table_mem_valid()
629 count = mem_ipv4->size / sizeof(__le64); in ipa_table_mem_valid()
633 ipa->filter_count = count - 1; /* Filter map in first entry */ in ipa_table_mem_valid()
635 ipa->route_count = count; in ipa_table_mem_valid()
638 if (!ipa_cmd_table_init_valid(ipa, mem_ipv4, !filter)) in ipa_table_mem_valid()
643 /* Filter tables must able to hold the endpoint bitmap plus in ipa_table_mem_valid()
646 if (count < 1 + hweight64(ipa->filtered)) in ipa_table_mem_valid()
649 /* Routing tables must be able to hold all modem entries, in ipa_table_mem_valid()
650 * plus at least one entry for the AP. in ipa_table_mem_valid()
652 if (count < ipa->modem_route_count + 1) in ipa_table_mem_valid()
656 /* If hashing is supported, hashed tables are expected to be defined, in ipa_table_mem_valid()
657 * and have the same size as non-hashed tables. If hashing is not in ipa_table_mem_valid()
658 * supported, hashed tables are expected to have zero size (or not in ipa_table_mem_valid()
661 mem_hashed = ipa_table_mem(ipa, filter, true, false); in ipa_table_mem_valid()
663 if (!mem_hashed || mem_hashed->size != mem_ipv4->size) in ipa_table_mem_valid()
666 if (mem_hashed && mem_hashed->size) in ipa_table_mem_valid()
671 mem_hashed = ipa_table_mem(ipa, filter, true, true); in ipa_table_mem_valid()
673 if (!mem_hashed || mem_hashed->size != mem_ipv6->size) in ipa_table_mem_valid()
676 if (mem_hashed && mem_hashed->size) in ipa_table_mem_valid()
684 * route table data. This is used when initializing or resetting the IPA
688 * endpoints contain entries in the table. In addition to that first entry,
693 * reset) its entries are made to refer to the zero rule.
696 * routing there is also a 64-bit "zero rule" that means no routing, and
697 * when a route table is initialized or reset, its entries are made to refer
698 * to the zero rule. The zero rule is shared for route and filter tables.
700 * +-------------------+
701 * --> | zero rule |
702 * / |-------------------|
704 * |\ |-------------------|
705 * | ---- zero rule address | \
706 * |\ |-------------------| |
707 * | ---- zero rule address | | Max IPA filter count
708 * | |-------------------| > or IPA route count,
710 * \ |-------------------| |
711 * ---- zero rule address | /
712 * +-------------------+
714 int ipa_table_init(struct ipa *ipa) in ipa_table_init() argument
716 struct device *dev = &ipa->pdev->dev; in ipa_table_init()
725 count = max_t(u32, ipa->filter_count, ipa->route_count); in ipa_table_init()
727 /* The IPA hardware requires route and filter table rules to be in ipa_table_init()
728 * aligned on a 128-byte boundary. We put the "zero rule" at the in ipa_table_init()
730 * by dma_alloc_coherent() is guaranteed to be a power-of-2 number in ipa_table_init()
736 return -ENOMEM; in ipa_table_init()
738 ipa->table_virt = virt; in ipa_table_init()
739 ipa->table_addr = addr; in ipa_table_init()
745 * need to be converted to the hardware representation by shifting in ipa_table_init()
746 * it left one position. Prior to IPA v5.0, bit 0 repesents global in ipa_table_init()
747 * filtering, which is possible but not used. IPA v5.0+ eliminated in ipa_table_init()
750 if (ipa->version < IPA_VERSION_5_0) in ipa_table_init()
751 *virt++ = cpu_to_le64(ipa->filtered << 1); in ipa_table_init()
753 *virt++ = cpu_to_le64(ipa->filtered); in ipa_table_init()
757 while (count--) in ipa_table_init()
763 void ipa_table_exit(struct ipa *ipa) in ipa_table_exit() argument
765 u32 count = max_t(u32, 1 + ipa->filter_count, ipa->route_count); in ipa_table_exit()
766 struct device *dev = &ipa->pdev->dev; in ipa_table_exit()
771 dma_free_coherent(dev, size, ipa->table_virt, ipa->table_addr); in ipa_table_exit()
772 ipa->table_addr = 0; in ipa_table_exit()
773 ipa->table_virt = NULL; in ipa_table_exit()