mirror of
https://github.com/FEX-Emu/linux.git
synced 2025-01-09 11:00:52 +00:00
Merge git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net-next
Pull networking updates from David Miller: "Highlights: 1) Fix the iwlwifi regression, from Johannes Berg. 2) Support BSS coloring and 802.11 encapsulation offloading in hardware, from John Crispin. 3) Fix some potential Spectre issues in qtnfmac, from Sergey Matyukevich. 4) Add TTL decrement action to openvswitch, from Matteo Croce. 5) Allow paralleization through flow_action setup by not taking the RTNL mutex, from Vlad Buslov. 6) A lot of zero-length array to flexible-array conversions, from Gustavo A. R. Silva. 7) Align XDP statistics names across several drivers for consistency, from Lorenzo Bianconi. 8) Add various pieces of infrastructure for offloading conntrack, and make use of it in mlx5 driver, from Paul Blakey. 9) Allow using listening sockets in BPF sockmap, from Jakub Sitnicki. 10) Lots of parallelization improvements during configuration changes in mlxsw driver, from Ido Schimmel. 11) Add support to devlink for generic packet traps, which report packets dropped during ACL processing. And use them in mlxsw driver. From Jiri Pirko. 12) Support bcmgenet on ACPI, from Jeremy Linton. 13) Make BPF compatible with RT, from Thomas Gleixnet, Alexei Starovoitov, and your's truly. 14) Support XDP meta-data in virtio_net, from Yuya Kusakabe. 15) Fix sysfs permissions when network devices change namespaces, from Christian Brauner. 16) Add a flags element to ethtool_ops so that drivers can more simply indicate which coalescing parameters they actually support, and therefore the generic layer can validate the user's ethtool request. Use this in all drivers, from Jakub Kicinski. 17) Offload FIFO qdisc in mlxsw, from Petr Machata. 18) Support UDP sockets in sockmap, from Lorenz Bauer. 19) Fix stretch ACK bugs in several TCP congestion control modules, from Pengcheng Yang. 20) Support virtual functiosn in octeontx2 driver, from Tomasz Duszynski. 21) Add region operations for devlink and use it in ice driver to dump NVM contents, from Jacob Keller. 22) Add support for hw offload of MACSEC, from Antoine Tenart. 23) Add support for BPF programs that can be attached to LSM hooks, from KP Singh. 24) Support for multiple paths, path managers, and counters in MPTCP. From Peter Krystad, Paolo Abeni, Florian Westphal, Davide Caratti, and others. 25) More progress on adding the netlink interface to ethtool, from Michal Kubecek" * git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net-next: (2121 commits) net: ipv6: rpl_iptunnel: Fix potential memory leak in rpl_do_srh_inline cxgb4/chcr: nic-tls stats in ethtool net: dsa: fix oops while probing Marvell DSA switches net/bpfilter: remove superfluous testing message net: macb: Fix handling of fixed-link node net: dsa: ksz: Select KSZ protocol tag netdevsim: dev: Fix memory leak in nsim_dev_take_snapshot_write net: stmmac: add EHL 2.5Gbps PCI info and PCI ID net: stmmac: add EHL PSE0 & PSE1 1Gbps PCI info and PCI ID net: stmmac: create dwmac-intel.c to contain all Intel platform net: dsa: bcm_sf2: Support specifying VLAN tag egress rule net: dsa: bcm_sf2: Add support for matching VLAN TCI net: dsa: bcm_sf2: Move writing of CFP_DATA(5) into slicing functions net: dsa: bcm_sf2: Check earlier for FLOW_EXT and FLOW_MAC_EXT net: dsa: bcm_sf2: Disable learning for ASP port net: dsa: b53: Deny enslaving port 7 for 7278 into a bridge net: dsa: b53: Prevent tagged VLAN on port 7 for 7278 net: dsa: b53: Restore VLAN entries upon (re)configuration net: dsa: bcm_sf2: Fix overflow checks hv_netvsc: Remove unnecessary round_up for recv_completion_cnt ...
This commit is contained in:
commit
29d9f30d4c
@ -67,7 +67,8 @@ two flavors of JITs, the newer eBPF JIT currently supported on:
|
||||
- sparc64
|
||||
- mips64
|
||||
- s390x
|
||||
- riscv
|
||||
- riscv64
|
||||
- riscv32
|
||||
|
||||
And the older cBPF JIT supported on the following archs:
|
||||
|
||||
|
@ -20,11 +20,11 @@ Reporting bugs
|
||||
Q: How do I report bugs for BPF kernel code?
|
||||
--------------------------------------------
|
||||
A: Since all BPF kernel development as well as bpftool and iproute2 BPF
|
||||
loader development happens through the netdev kernel mailing list,
|
||||
loader development happens through the bpf kernel mailing list,
|
||||
please report any found issues around BPF to the following mailing
|
||||
list:
|
||||
|
||||
netdev@vger.kernel.org
|
||||
bpf@vger.kernel.org
|
||||
|
||||
This may also include issues related to XDP, BPF tracing, etc.
|
||||
|
||||
@ -46,17 +46,12 @@ Submitting patches
|
||||
|
||||
Q: To which mailing list do I need to submit my BPF patches?
|
||||
------------------------------------------------------------
|
||||
A: Please submit your BPF patches to the netdev kernel mailing list:
|
||||
A: Please submit your BPF patches to the bpf kernel mailing list:
|
||||
|
||||
netdev@vger.kernel.org
|
||||
|
||||
Historically, BPF came out of networking and has always been maintained
|
||||
by the kernel networking community. Although these days BPF touches
|
||||
many other subsystems as well, the patches are still routed mainly
|
||||
through the networking community.
|
||||
bpf@vger.kernel.org
|
||||
|
||||
In case your patch has changes in various different subsystems (e.g.
|
||||
tracing, security, etc), make sure to Cc the related kernel mailing
|
||||
networking, tracing, security, etc), make sure to Cc the related kernel mailing
|
||||
lists and maintainers from there as well, so they are able to review
|
||||
the changes and provide their Acked-by's to the patches.
|
||||
|
||||
@ -168,7 +163,7 @@ a BPF point of view.
|
||||
Be aware that this is not a final verdict that the patch will
|
||||
automatically get accepted into net or net-next trees eventually:
|
||||
|
||||
On the netdev kernel mailing list reviews can come in at any point
|
||||
On the bpf kernel mailing list reviews can come in at any point
|
||||
in time. If discussions around a patch conclude that they cannot
|
||||
get included as-is, we will either apply a follow-up fix or drop
|
||||
them from the trees entirely. Therefore, we also reserve to rebase
|
||||
@ -494,15 +489,15 @@ A: You need cmake and gcc-c++ as build requisites for LLVM. Once you have
|
||||
that set up, proceed with building the latest LLVM and clang version
|
||||
from the git repositories::
|
||||
|
||||
$ git clone http://llvm.org/git/llvm.git
|
||||
$ cd llvm/tools
|
||||
$ git clone --depth 1 http://llvm.org/git/clang.git
|
||||
$ cd ..; mkdir build; cd build
|
||||
$ cmake .. -DLLVM_TARGETS_TO_BUILD="BPF;X86" \
|
||||
$ git clone https://github.com/llvm/llvm-project.git
|
||||
$ mkdir -p llvm-project/llvm/build/install
|
||||
$ cd llvm-project/llvm/build
|
||||
$ cmake .. -G "Ninja" -DLLVM_TARGETS_TO_BUILD="BPF;X86" \
|
||||
-DLLVM_ENABLE_PROJECTS="clang" \
|
||||
-DBUILD_SHARED_LIBS=OFF \
|
||||
-DCMAKE_BUILD_TYPE=Release \
|
||||
-DLLVM_BUILD_RUNTIME=OFF
|
||||
$ make -j $(getconf _NPROCESSORS_ONLN)
|
||||
$ ninja
|
||||
|
||||
The built binaries can then be found in the build/bin/ directory, where
|
||||
you can point the PATH variable to.
|
||||
|
142
Documentation/bpf/bpf_lsm.rst
Normal file
142
Documentation/bpf/bpf_lsm.rst
Normal file
@ -0,0 +1,142 @@
|
||||
.. SPDX-License-Identifier: GPL-2.0+
|
||||
.. Copyright (C) 2020 Google LLC.
|
||||
|
||||
================
|
||||
LSM BPF Programs
|
||||
================
|
||||
|
||||
These BPF programs allow runtime instrumentation of the LSM hooks by privileged
|
||||
users to implement system-wide MAC (Mandatory Access Control) and Audit
|
||||
policies using eBPF.
|
||||
|
||||
Structure
|
||||
---------
|
||||
|
||||
The example shows an eBPF program that can be attached to the ``file_mprotect``
|
||||
LSM hook:
|
||||
|
||||
.. c:function:: int file_mprotect(struct vm_area_struct *vma, unsigned long reqprot, unsigned long prot);
|
||||
|
||||
Other LSM hooks which can be instrumented can be found in
|
||||
``include/linux/lsm_hooks.h``.
|
||||
|
||||
eBPF programs that use :doc:`/bpf/btf` do not need to include kernel headers
|
||||
for accessing information from the attached eBPF program's context. They can
|
||||
simply declare the structures in the eBPF program and only specify the fields
|
||||
that need to be accessed.
|
||||
|
||||
.. code-block:: c
|
||||
|
||||
struct mm_struct {
|
||||
unsigned long start_brk, brk, start_stack;
|
||||
} __attribute__((preserve_access_index));
|
||||
|
||||
struct vm_area_struct {
|
||||
unsigned long start_brk, brk, start_stack;
|
||||
unsigned long vm_start, vm_end;
|
||||
struct mm_struct *vm_mm;
|
||||
} __attribute__((preserve_access_index));
|
||||
|
||||
|
||||
.. note:: The order of the fields is irrelevant.
|
||||
|
||||
This can be further simplified (if one has access to the BTF information at
|
||||
build time) by generating the ``vmlinux.h`` with:
|
||||
|
||||
.. code-block:: console
|
||||
|
||||
# bpftool btf dump file <path-to-btf-vmlinux> format c > vmlinux.h
|
||||
|
||||
.. note:: ``path-to-btf-vmlinux`` can be ``/sys/kernel/btf/vmlinux`` if the
|
||||
build environment matches the environment the BPF programs are
|
||||
deployed in.
|
||||
|
||||
The ``vmlinux.h`` can then simply be included in the BPF programs without
|
||||
requiring the definition of the types.
|
||||
|
||||
The eBPF programs can be declared using the``BPF_PROG``
|
||||
macros defined in `tools/lib/bpf/bpf_tracing.h`_. In this
|
||||
example:
|
||||
|
||||
* ``"lsm/file_mprotect"`` indicates the LSM hook that the program must
|
||||
be attached to
|
||||
* ``mprotect_audit`` is the name of the eBPF program
|
||||
|
||||
.. code-block:: c
|
||||
|
||||
SEC("lsm/file_mprotect")
|
||||
int BPF_PROG(mprotect_audit, struct vm_area_struct *vma,
|
||||
unsigned long reqprot, unsigned long prot, int ret)
|
||||
{
|
||||
/* ret is the return value from the previous BPF program
|
||||
* or 0 if it's the first hook.
|
||||
*/
|
||||
if (ret != 0)
|
||||
return ret;
|
||||
|
||||
int is_heap;
|
||||
|
||||
is_heap = (vma->vm_start >= vma->vm_mm->start_brk &&
|
||||
vma->vm_end <= vma->vm_mm->brk);
|
||||
|
||||
/* Return an -EPERM or write information to the perf events buffer
|
||||
* for auditing
|
||||
*/
|
||||
if (is_heap)
|
||||
return -EPERM;
|
||||
}
|
||||
|
||||
The ``__attribute__((preserve_access_index))`` is a clang feature that allows
|
||||
the BPF verifier to update the offsets for the access at runtime using the
|
||||
:doc:`/bpf/btf` information. Since the BPF verifier is aware of the types, it
|
||||
also validates all the accesses made to the various types in the eBPF program.
|
||||
|
||||
Loading
|
||||
-------
|
||||
|
||||
eBPF programs can be loaded with the :manpage:`bpf(2)` syscall's
|
||||
``BPF_PROG_LOAD`` operation:
|
||||
|
||||
.. code-block:: c
|
||||
|
||||
struct bpf_object *obj;
|
||||
|
||||
obj = bpf_object__open("./my_prog.o");
|
||||
bpf_object__load(obj);
|
||||
|
||||
This can be simplified by using a skeleton header generated by ``bpftool``:
|
||||
|
||||
.. code-block:: console
|
||||
|
||||
# bpftool gen skeleton my_prog.o > my_prog.skel.h
|
||||
|
||||
and the program can be loaded by including ``my_prog.skel.h`` and using
|
||||
the generated helper, ``my_prog__open_and_load``.
|
||||
|
||||
Attachment to LSM Hooks
|
||||
-----------------------
|
||||
|
||||
The LSM allows attachment of eBPF programs as LSM hooks using :manpage:`bpf(2)`
|
||||
syscall's ``BPF_RAW_TRACEPOINT_OPEN`` operation or more simply by
|
||||
using the libbpf helper ``bpf_program__attach_lsm``.
|
||||
|
||||
The program can be detached from the LSM hook by *destroying* the ``link``
|
||||
link returned by ``bpf_program__attach_lsm`` using ``bpf_link__destroy``.
|
||||
|
||||
One can also use the helpers generated in ``my_prog.skel.h`` i.e.
|
||||
``my_prog__attach`` for attachment and ``my_prog__destroy`` for cleaning up.
|
||||
|
||||
Examples
|
||||
--------
|
||||
|
||||
An example eBPF program can be found in
|
||||
`tools/testing/selftests/bpf/progs/lsm.c`_ and the corresponding
|
||||
userspace code in `tools/testing/selftests/bpf/prog_tests/test_lsm.c`_
|
||||
|
||||
.. Links
|
||||
.. _tools/lib/bpf/bpf_tracing.h:
|
||||
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git/tree/tools/lib/bpf/bpf_tracing.h
|
||||
.. _tools/testing/selftests/bpf/progs/lsm.c:
|
||||
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git/tree/tools/testing/selftests/bpf/progs/lsm.c
|
||||
.. _tools/testing/selftests/bpf/prog_tests/test_lsm.c:
|
||||
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git/tree/tools/testing/selftests/bpf/prog_tests/test_lsm.c
|
213
Documentation/bpf/drgn.rst
Normal file
213
Documentation/bpf/drgn.rst
Normal file
@ -0,0 +1,213 @@
|
||||
.. SPDX-License-Identifier: (LGPL-2.1 OR BSD-2-Clause)
|
||||
|
||||
==============
|
||||
BPF drgn tools
|
||||
==============
|
||||
|
||||
drgn scripts is a convenient and easy to use mechanism to retrieve arbitrary
|
||||
kernel data structures. drgn is not relying on kernel UAPI to read the data.
|
||||
Instead it's reading directly from ``/proc/kcore`` or vmcore and pretty prints
|
||||
the data based on DWARF debug information from vmlinux.
|
||||
|
||||
This document describes BPF related drgn tools.
|
||||
|
||||
See `drgn/tools`_ for all tools available at the moment and `drgn/doc`_ for
|
||||
more details on drgn itself.
|
||||
|
||||
bpf_inspect.py
|
||||
--------------
|
||||
|
||||
Description
|
||||
===========
|
||||
|
||||
`bpf_inspect.py`_ is a tool intended to inspect BPF programs and maps. It can
|
||||
iterate over all programs and maps in the system and print basic information
|
||||
about these objects, including id, type and name.
|
||||
|
||||
The main use-case `bpf_inspect.py`_ covers is to show BPF programs of types
|
||||
``BPF_PROG_TYPE_EXT`` and ``BPF_PROG_TYPE_TRACING`` attached to other BPF
|
||||
programs via ``freplace``/``fentry``/``fexit`` mechanisms, since there is no
|
||||
user-space API to get this information.
|
||||
|
||||
Getting started
|
||||
===============
|
||||
|
||||
List BPF programs (full names are obtained from BTF)::
|
||||
|
||||
% sudo bpf_inspect.py prog
|
||||
27: BPF_PROG_TYPE_TRACEPOINT tracepoint__tcp__tcp_send_reset
|
||||
4632: BPF_PROG_TYPE_CGROUP_SOCK_ADDR tw_ipt_bind
|
||||
49464: BPF_PROG_TYPE_RAW_TRACEPOINT raw_tracepoint__sched_process_exit
|
||||
|
||||
List BPF maps::
|
||||
|
||||
% sudo bpf_inspect.py map
|
||||
2577: BPF_MAP_TYPE_HASH tw_ipt_vips
|
||||
4050: BPF_MAP_TYPE_STACK_TRACE stack_traces
|
||||
4069: BPF_MAP_TYPE_PERCPU_ARRAY ned_dctcp_cntr
|
||||
|
||||
Find BPF programs attached to BPF program ``test_pkt_access``::
|
||||
|
||||
% sudo bpf_inspect.py p | grep test_pkt_access
|
||||
650: BPF_PROG_TYPE_SCHED_CLS test_pkt_access
|
||||
654: BPF_PROG_TYPE_TRACING test_main linked:[650->25: BPF_TRAMP_FEXIT test_pkt_access->test_pkt_access()]
|
||||
655: BPF_PROG_TYPE_TRACING test_subprog1 linked:[650->29: BPF_TRAMP_FEXIT test_pkt_access->test_pkt_access_subprog1()]
|
||||
656: BPF_PROG_TYPE_TRACING test_subprog2 linked:[650->31: BPF_TRAMP_FEXIT test_pkt_access->test_pkt_access_subprog2()]
|
||||
657: BPF_PROG_TYPE_TRACING test_subprog3 linked:[650->21: BPF_TRAMP_FEXIT test_pkt_access->test_pkt_access_subprog3()]
|
||||
658: BPF_PROG_TYPE_EXT new_get_skb_len linked:[650->16: BPF_TRAMP_REPLACE test_pkt_access->get_skb_len()]
|
||||
659: BPF_PROG_TYPE_EXT new_get_skb_ifindex linked:[650->23: BPF_TRAMP_REPLACE test_pkt_access->get_skb_ifindex()]
|
||||
660: BPF_PROG_TYPE_EXT new_get_constant linked:[650->19: BPF_TRAMP_REPLACE test_pkt_access->get_constant()]
|
||||
|
||||
It can be seen that there is a program ``test_pkt_access``, id 650 and there
|
||||
are multiple other tracing and ext programs attached to functions in
|
||||
``test_pkt_access``.
|
||||
|
||||
For example the line::
|
||||
|
||||
658: BPF_PROG_TYPE_EXT new_get_skb_len linked:[650->16: BPF_TRAMP_REPLACE test_pkt_access->get_skb_len()]
|
||||
|
||||
, means that BPF program id 658, type ``BPF_PROG_TYPE_EXT``, name
|
||||
``new_get_skb_len`` replaces (``BPF_TRAMP_REPLACE``) function ``get_skb_len()``
|
||||
that has BTF id 16 in BPF program id 650, name ``test_pkt_access``.
|
||||
|
||||
Getting help:
|
||||
|
||||
.. code-block:: none
|
||||
|
||||
% sudo bpf_inspect.py
|
||||
usage: bpf_inspect.py [-h] {prog,p,map,m} ...
|
||||
|
||||
drgn script to list BPF programs or maps and their properties
|
||||
unavailable via kernel API.
|
||||
|
||||
See https://github.com/osandov/drgn/ for more details on drgn.
|
||||
|
||||
optional arguments:
|
||||
-h, --help show this help message and exit
|
||||
|
||||
subcommands:
|
||||
{prog,p,map,m}
|
||||
prog (p) list BPF programs
|
||||
map (m) list BPF maps
|
||||
|
||||
Customization
|
||||
=============
|
||||
|
||||
The script is intended to be customized by developers to print relevant
|
||||
information about BPF programs, maps and other objects.
|
||||
|
||||
For example, to print ``struct bpf_prog_aux`` for BPF program id 53077:
|
||||
|
||||
.. code-block:: none
|
||||
|
||||
% git diff
|
||||
diff --git a/tools/bpf_inspect.py b/tools/bpf_inspect.py
|
||||
index 650e228..aea2357 100755
|
||||
--- a/tools/bpf_inspect.py
|
||||
+++ b/tools/bpf_inspect.py
|
||||
@@ -112,7 +112,9 @@ def list_bpf_progs(args):
|
||||
if linked:
|
||||
linked = f" linked:[{linked}]"
|
||||
|
||||
- print(f"{id_:>6}: {type_:32} {name:32} {linked}")
|
||||
+ if id_ == 53077:
|
||||
+ print(f"{id_:>6}: {type_:32} {name:32}")
|
||||
+ print(f"{bpf_prog.aux}")
|
||||
|
||||
|
||||
def list_bpf_maps(args):
|
||||
|
||||
It produces the output::
|
||||
|
||||
% sudo bpf_inspect.py p
|
||||
53077: BPF_PROG_TYPE_XDP tw_xdp_policer
|
||||
*(struct bpf_prog_aux *)0xffff8893fad4b400 = {
|
||||
.refcnt = (atomic64_t){
|
||||
.counter = (long)58,
|
||||
},
|
||||
.used_map_cnt = (u32)1,
|
||||
.max_ctx_offset = (u32)8,
|
||||
.max_pkt_offset = (u32)15,
|
||||
.max_tp_access = (u32)0,
|
||||
.stack_depth = (u32)8,
|
||||
.id = (u32)53077,
|
||||
.func_cnt = (u32)0,
|
||||
.func_idx = (u32)0,
|
||||
.attach_btf_id = (u32)0,
|
||||
.linked_prog = (struct bpf_prog *)0x0,
|
||||
.verifier_zext = (bool)0,
|
||||
.offload_requested = (bool)0,
|
||||
.attach_btf_trace = (bool)0,
|
||||
.func_proto_unreliable = (bool)0,
|
||||
.trampoline_prog_type = (enum bpf_tramp_prog_type)BPF_TRAMP_FENTRY,
|
||||
.trampoline = (struct bpf_trampoline *)0x0,
|
||||
.tramp_hlist = (struct hlist_node){
|
||||
.next = (struct hlist_node *)0x0,
|
||||
.pprev = (struct hlist_node **)0x0,
|
||||
},
|
||||
.attach_func_proto = (const struct btf_type *)0x0,
|
||||
.attach_func_name = (const char *)0x0,
|
||||
.func = (struct bpf_prog **)0x0,
|
||||
.jit_data = (void *)0x0,
|
||||
.poke_tab = (struct bpf_jit_poke_descriptor *)0x0,
|
||||
.size_poke_tab = (u32)0,
|
||||
.ksym_tnode = (struct latch_tree_node){
|
||||
.node = (struct rb_node [2]){
|
||||
{
|
||||
.__rb_parent_color = (unsigned long)18446612956263126665,
|
||||
.rb_right = (struct rb_node *)0x0,
|
||||
.rb_left = (struct rb_node *)0xffff88a0be3d0088,
|
||||
},
|
||||
{
|
||||
.__rb_parent_color = (unsigned long)18446612956263126689,
|
||||
.rb_right = (struct rb_node *)0x0,
|
||||
.rb_left = (struct rb_node *)0xffff88a0be3d00a0,
|
||||
},
|
||||
},
|
||||
},
|
||||
.ksym_lnode = (struct list_head){
|
||||
.next = (struct list_head *)0xffff88bf481830b8,
|
||||
.prev = (struct list_head *)0xffff888309f536b8,
|
||||
},
|
||||
.ops = (const struct bpf_prog_ops *)xdp_prog_ops+0x0 = 0xffffffff820fa350,
|
||||
.used_maps = (struct bpf_map **)0xffff889ff795de98,
|
||||
.prog = (struct bpf_prog *)0xffffc9000cf2d000,
|
||||
.user = (struct user_struct *)root_user+0x0 = 0xffffffff82444820,
|
||||
.load_time = (u64)2408348759285319,
|
||||
.cgroup_storage = (struct bpf_map *[2]){},
|
||||
.name = (char [16])"tw_xdp_policer",
|
||||
.security = (void *)0xffff889ff795d548,
|
||||
.offload = (struct bpf_prog_offload *)0x0,
|
||||
.btf = (struct btf *)0xffff8890ce6d0580,
|
||||
.func_info = (struct bpf_func_info *)0xffff889ff795d240,
|
||||
.func_info_aux = (struct bpf_func_info_aux *)0xffff889ff795de20,
|
||||
.linfo = (struct bpf_line_info *)0xffff888a707afc00,
|
||||
.jited_linfo = (void **)0xffff8893fad48600,
|
||||
.func_info_cnt = (u32)1,
|
||||
.nr_linfo = (u32)37,
|
||||
.linfo_idx = (u32)0,
|
||||
.num_exentries = (u32)0,
|
||||
.extable = (struct exception_table_entry *)0xffffffffa032d950,
|
||||
.stats = (struct bpf_prog_stats *)0x603fe3a1f6d0,
|
||||
.work = (struct work_struct){
|
||||
.data = (atomic_long_t){
|
||||
.counter = (long)0,
|
||||
},
|
||||
.entry = (struct list_head){
|
||||
.next = (struct list_head *)0x0,
|
||||
.prev = (struct list_head *)0x0,
|
||||
},
|
||||
.func = (work_func_t)0x0,
|
||||
},
|
||||
.rcu = (struct callback_head){
|
||||
.next = (struct callback_head *)0x0,
|
||||
.func = (void (*)(struct callback_head *))0x0,
|
||||
},
|
||||
}
|
||||
|
||||
|
||||
.. Links
|
||||
.. _drgn/doc: https://drgn.readthedocs.io/en/latest/
|
||||
.. _drgn/tools: https://github.com/osandov/drgn/tree/master/tools
|
||||
.. _bpf_inspect.py:
|
||||
https://github.com/osandov/drgn/blob/master/tools/bpf_inspect.py
|
@ -45,14 +45,16 @@ Program types
|
||||
prog_cgroup_sockopt
|
||||
prog_cgroup_sysctl
|
||||
prog_flow_dissector
|
||||
bpf_lsm
|
||||
|
||||
|
||||
Testing BPF
|
||||
===========
|
||||
Testing and debugging BPF
|
||||
=========================
|
||||
|
||||
.. toctree::
|
||||
:maxdepth: 1
|
||||
|
||||
drgn
|
||||
s390
|
||||
|
||||
|
||||
|
116
Documentation/devicetree/bindings/net/dsa/ocelot.txt
Normal file
116
Documentation/devicetree/bindings/net/dsa/ocelot.txt
Normal file
@ -0,0 +1,116 @@
|
||||
Microchip Ocelot switch driver family
|
||||
=====================================
|
||||
|
||||
Felix
|
||||
-----
|
||||
|
||||
The VSC9959 core is currently the only switch supported by the driver, and is
|
||||
found in the NXP LS1028A. It is a PCI device, part of the larger ENETC root
|
||||
complex. As a result, the ethernet-switch node is a sub-node of the PCIe root
|
||||
complex node and its "reg" property conforms to the parent node bindings:
|
||||
|
||||
* reg: Specifies PCIe Device Number and Function Number of the endpoint device,
|
||||
in this case for the Ethernet L2Switch it is PF5 (of device 0, bus 0).
|
||||
|
||||
It does not require a "compatible" string.
|
||||
|
||||
The interrupt line is used to signal availability of PTP TX timestamps and for
|
||||
TSN frame preemption.
|
||||
|
||||
For the external switch ports, depending on board configuration, "phy-mode" and
|
||||
"phy-handle" are populated by board specific device tree instances. Ports 4 and
|
||||
5 are fixed as internal ports in the NXP LS1028A instantiation.
|
||||
|
||||
The CPU port property ("ethernet") configures the feature called "NPI port" in
|
||||
the Ocelot hardware core. The CPU port in Ocelot is a set of queues, which are
|
||||
connected, in the Node Processor Interface (NPI) mode, to an Ethernet port.
|
||||
By default, in fsl-ls1028a.dtsi, the NPI port is assigned to the internal
|
||||
2.5Gbps port@4, but can be moved to the 1Gbps port@5, depending on the specific
|
||||
use case. Moving the NPI port to an external switch port is hardware possible,
|
||||
but there is no platform support for the Linux system on the LS1028A chip to
|
||||
operate as an entire slave DSA chip. NPI functionality (and therefore DSA
|
||||
tagging) is supported on a single port at a time.
|
||||
|
||||
Any port can be disabled (and in fsl-ls1028a.dtsi, they are indeed all disabled
|
||||
by default, and should be enabled on a per-board basis). But if any external
|
||||
switch port is enabled at all, the ENETC PF2 (enetc_port2) should be enabled as
|
||||
well, regardless of whether it is configured as the DSA master or not. This is
|
||||
because the Felix PHYLINK implementation accesses the MAC PCS registers, which
|
||||
in hardware truly belong to the ENETC port #2 and not to Felix.
|
||||
|
||||
Supported PHY interface types (appropriate SerDes protocol setting changes are
|
||||
needed in the RCW binary):
|
||||
|
||||
* phy_mode = "internal": on ports 4 and 5
|
||||
* phy_mode = "sgmii": on ports 0, 1, 2, 3
|
||||
* phy_mode = "qsgmii": on ports 0, 1, 2, 3
|
||||
* phy_mode = "usxgmii": on ports 0, 1, 2, 3
|
||||
* phy_mode = "2500base-x": on ports 0, 1, 2, 3
|
||||
|
||||
For the rest of the device tree binding definitions, which are standard DSA and
|
||||
PCI, refer to the following documents:
|
||||
|
||||
Documentation/devicetree/bindings/net/dsa/dsa.txt
|
||||
Documentation/devicetree/bindings/pci/pci.txt
|
||||
|
||||
Example:
|
||||
|
||||
&soc {
|
||||
pcie@1f0000000 { /* Integrated Endpoint Root Complex */
|
||||
ethernet-switch@0,5 {
|
||||
reg = <0x000500 0 0 0 0>;
|
||||
/* IEP INT_B */
|
||||
interrupts = <GIC_SPI 95 IRQ_TYPE_LEVEL_HIGH>;
|
||||
|
||||
ports {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
/* External ports */
|
||||
port@0 {
|
||||
reg = <0>;
|
||||
label = "swp0";
|
||||
};
|
||||
|
||||
port@1 {
|
||||
reg = <1>;
|
||||
label = "swp1";
|
||||
};
|
||||
|
||||
port@2 {
|
||||
reg = <2>;
|
||||
label = "swp2";
|
||||
};
|
||||
|
||||
port@3 {
|
||||
reg = <3>;
|
||||
label = "swp3";
|
||||
};
|
||||
|
||||
/* Tagging CPU port */
|
||||
port@4 {
|
||||
reg = <4>;
|
||||
ethernet = <&enetc_port2>;
|
||||
phy-mode = "internal";
|
||||
|
||||
fixed-link {
|
||||
speed = <2500>;
|
||||
full-duplex;
|
||||
};
|
||||
};
|
||||
|
||||
/* Non-tagging CPU port */
|
||||
port@5 {
|
||||
reg = <5>;
|
||||
phy-mode = "internal";
|
||||
status = "disabled";
|
||||
|
||||
fixed-link {
|
||||
speed = <1000>;
|
||||
full-duplex;
|
||||
};
|
||||
};
|
||||
};
|
||||
};
|
||||
};
|
||||
};
|
65
Documentation/devicetree/bindings/net/marvell,mvusb.yaml
Normal file
65
Documentation/devicetree/bindings/net/marvell,mvusb.yaml
Normal file
@ -0,0 +1,65 @@
|
||||
# SPDX-License-Identifier: GPL-2.0
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/net/marvell,mvusb.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: Marvell USB to MDIO Controller
|
||||
|
||||
maintainers:
|
||||
- Tobias Waldekranz <tobias@waldekranz.com>
|
||||
|
||||
description: |+
|
||||
This controller is mounted on development boards for Marvell's Link Street
|
||||
family of Ethernet switches. It allows you to configure the switch's registers
|
||||
using the standard MDIO interface.
|
||||
|
||||
Since the device is connected over USB, there is no strict requirement of
|
||||
having a device tree representation of the device. But in order to use it with
|
||||
the mv88e6xxx driver, you need a device tree node in which to place the switch
|
||||
definition.
|
||||
|
||||
allOf:
|
||||
- $ref: "mdio.yaml#"
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
const: usb1286,1fa4
|
||||
reg:
|
||||
maxItems: 1
|
||||
description: The USB port number on the host controller
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- reg
|
||||
- "#address-cells"
|
||||
- "#size-cells"
|
||||
|
||||
examples:
|
||||
- |
|
||||
/* USB host controller */
|
||||
&usb1 {
|
||||
mvusb: mdio@1 {
|
||||
compatible = "usb1286,1fa4";
|
||||
reg = <1>;
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
};
|
||||
};
|
||||
|
||||
/* MV88E6390X devboard */
|
||||
&mvusb {
|
||||
switch@0 {
|
||||
compatible = "marvell,mv88e6190";
|
||||
status = "ok";
|
||||
reg = <0x0>;
|
||||
|
||||
ports {
|
||||
/* Port definitions */
|
||||
};
|
||||
|
||||
mdio {
|
||||
/* PHY definitions */
|
||||
};
|
||||
};
|
||||
};
|
198
Documentation/devicetree/bindings/net/qcom,ipa.yaml
Normal file
198
Documentation/devicetree/bindings/net/qcom,ipa.yaml
Normal file
@ -0,0 +1,198 @@
|
||||
# SPDX-License-Identifier: GPL-2.0 OR BSD-2-Clause
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/net/qcom,ipa.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: Qualcomm IP Accelerator (IPA)
|
||||
|
||||
maintainers:
|
||||
- Alex Elder <elder@kernel.org>
|
||||
|
||||
description:
|
||||
This binding describes the Qualcomm IPA. The IPA is capable of offloading
|
||||
certain network processing tasks (e.g. filtering, routing, and NAT) from
|
||||
the main processor.
|
||||
|
||||
The IPA sits between multiple independent "execution environments,"
|
||||
including the Application Processor (AP) and the modem. The IPA presents
|
||||
a Generic Software Interface (GSI) to each execution environment.
|
||||
The GSI is an integral part of the IPA, but it is logically isolated
|
||||
and has a distinct interrupt and a separately-defined address space.
|
||||
|
||||
See also soc/qcom/qcom,smp2p.txt and interconnect/interconnect.txt.
|
||||
|
||||
- |
|
||||
-------- ---------
|
||||
| | | |
|
||||
| AP +<---. .----+ Modem |
|
||||
| +--. | | .->+ |
|
||||
| | | | | | | |
|
||||
-------- | | | | ---------
|
||||
v | v |
|
||||
--+-+---+-+--
|
||||
| GSI |
|
||||
|-----------|
|
||||
| |
|
||||
| IPA |
|
||||
| |
|
||||
-------------
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
const: "qcom,sdm845-ipa"
|
||||
|
||||
reg:
|
||||
items:
|
||||
- description: IPA registers
|
||||
- description: IPA shared memory
|
||||
- description: GSI registers
|
||||
|
||||
reg-names:
|
||||
items:
|
||||
- const: ipa-reg
|
||||
- const: ipa-shared
|
||||
- const: gsi
|
||||
|
||||
clocks:
|
||||
maxItems: 1
|
||||
|
||||
clock-names:
|
||||
const: core
|
||||
|
||||
interrupts:
|
||||
items:
|
||||
- description: IPA interrupt (hardware IRQ)
|
||||
- description: GSI interrupt (hardware IRQ)
|
||||
- description: Modem clock query interrupt (smp2p interrupt)
|
||||
- description: Modem setup ready interrupt (smp2p interrupt)
|
||||
|
||||
interrupt-names:
|
||||
items:
|
||||
- const: ipa
|
||||
- const: gsi
|
||||
- const: ipa-clock-query
|
||||
- const: ipa-setup-ready
|
||||
|
||||
interconnects:
|
||||
items:
|
||||
- description: Interconnect path between IPA and main memory
|
||||
- description: Interconnect path between IPA and internal memory
|
||||
- description: Interconnect path between IPA and the AP subsystem
|
||||
|
||||
interconnect-names:
|
||||
items:
|
||||
- const: memory
|
||||
- const: imem
|
||||
- const: config
|
||||
|
||||
qcom,smem-states:
|
||||
allOf:
|
||||
- $ref: /schemas/types.yaml#/definitions/phandle-array
|
||||
description: State bits used in by the AP to signal the modem.
|
||||
items:
|
||||
- description: Whether the "ipa-clock-enabled" state bit is valid
|
||||
- description: Whether the IPA clock is enabled (if valid)
|
||||
|
||||
qcom,smem-state-names:
|
||||
allOf:
|
||||
- $ref: /schemas/types.yaml#/definitions/string-array
|
||||
description: The names of the state bits used for SMP2P output
|
||||
items:
|
||||
- const: ipa-clock-enabled-valid
|
||||
- const: ipa-clock-enabled
|
||||
|
||||
modem-init:
|
||||
type: boolean
|
||||
description:
|
||||
If present, it indicates that the modem is responsible for
|
||||
performing early IPA initialization, including loading and
|
||||
validating firwmare used by the GSI.
|
||||
|
||||
modem-remoteproc:
|
||||
$ref: /schemas/types.yaml#definitions/phandle
|
||||
description:
|
||||
This defines the phandle to the remoteproc node representing
|
||||
the modem subsystem. This is requied so the IPA driver can
|
||||
receive and act on notifications of modem up/down events.
|
||||
|
||||
memory-region:
|
||||
$ref: /schemas/types.yaml#/definitions/phandle-array
|
||||
maxItems: 1
|
||||
description:
|
||||
If present, a phandle for a reserved memory area that holds
|
||||
the firmware passed to Trust Zone for authentication. Required
|
||||
when Trust Zone (not the modem) performs early initialization.
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- reg
|
||||
- clocks
|
||||
- interrupts
|
||||
- interconnects
|
||||
- qcom,smem-states
|
||||
- modem-remoteproc
|
||||
|
||||
oneOf:
|
||||
- required:
|
||||
- modem-init
|
||||
- required:
|
||||
- memory-region
|
||||
|
||||
examples:
|
||||
- |
|
||||
#include <dt-bindings/interrupt-controller/irq.h>
|
||||
#include <dt-bindings/clock/qcom,rpmh.h>
|
||||
#include <dt-bindings/interconnect/qcom,sdm845.h>
|
||||
|
||||
smp2p-mpss {
|
||||
compatible = "qcom,smp2p";
|
||||
ipa_smp2p_out: ipa-ap-to-modem {
|
||||
qcom,entry-name = "ipa";
|
||||
#qcom,smem-state-cells = <1>;
|
||||
};
|
||||
|
||||
ipa_smp2p_in: ipa-modem-to-ap {
|
||||
qcom,entry-name = "ipa";
|
||||
interrupt-controller;
|
||||
#interrupt-cells = <2>;
|
||||
};
|
||||
};
|
||||
ipa@1e40000 {
|
||||
compatible = "qcom,sdm845-ipa";
|
||||
|
||||
modem-init;
|
||||
modem-remoteproc = <&mss_pil>;
|
||||
|
||||
reg = <0 0x1e40000 0 0x7000>,
|
||||
<0 0x1e47000 0 0x2000>,
|
||||
<0 0x1e04000 0 0x2c000>;
|
||||
reg-names = "ipa-reg",
|
||||
"ipa-shared",
|
||||
"gsi";
|
||||
|
||||
interrupts-extended = <&intc 0 311 IRQ_TYPE_EDGE_RISING>,
|
||||
<&intc 0 432 IRQ_TYPE_LEVEL_HIGH>,
|
||||
<&ipa_smp2p_in 0 IRQ_TYPE_EDGE_RISING>,
|
||||
<&ipa_smp2p_in 1 IRQ_TYPE_EDGE_RISING>;
|
||||
interrupt-names = "ipa",
|
||||
"gsi",
|
||||
"ipa-clock-query",
|
||||
"ipa-setup-ready";
|
||||
|
||||
clocks = <&rpmhcc RPMH_IPA_CLK>;
|
||||
clock-names = "core";
|
||||
|
||||
interconnects =
|
||||
<&rsc_hlos MASTER_IPA &rsc_hlos SLAVE_EBI1>,
|
||||
<&rsc_hlos MASTER_IPA &rsc_hlos SLAVE_IMEM>,
|
||||
<&rsc_hlos MASTER_APPSS_PROC &rsc_hlos SLAVE_IPA_CFG>;
|
||||
interconnect-names = "memory",
|
||||
"imem",
|
||||
"config";
|
||||
|
||||
qcom,smem-states = <&ipa_smp2p_out 0>,
|
||||
<&ipa_smp2p_out 1>;
|
||||
qcom,smem-state-names = "ipa-clock-enabled-valid",
|
||||
"ipa-clock-enabled";
|
||||
};
|
53
Documentation/devicetree/bindings/net/qcom,ipq8064-mdio.yaml
Normal file
53
Documentation/devicetree/bindings/net/qcom,ipq8064-mdio.yaml
Normal file
@ -0,0 +1,53 @@
|
||||
# SPDX-License-Identifier: GPL-2.0-only OR BSD-2-Clause
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/net/qcom,ipq8064-mdio.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: Qualcomm ipq806x MDIO bus controller
|
||||
|
||||
maintainers:
|
||||
- Ansuel Smith <ansuelsmth@gmail.com>
|
||||
|
||||
description:
|
||||
The ipq806x soc have a MDIO dedicated controller that is
|
||||
used to communicate with the gmac phy connected.
|
||||
|
||||
allOf:
|
||||
- $ref: "mdio.yaml#"
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
const: qcom,ipq8064-mdio
|
||||
|
||||
reg:
|
||||
maxItems: 1
|
||||
|
||||
clocks:
|
||||
maxItems: 1
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- reg
|
||||
- clocks
|
||||
- "#address-cells"
|
||||
- "#size-cells"
|
||||
|
||||
examples:
|
||||
- |
|
||||
#include <dt-bindings/clock/qcom,gcc-ipq806x.h>
|
||||
|
||||
mdio0: mdio@37000000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
compatible = "qcom,ipq8064-mdio";
|
||||
reg = <0x37000000 0x200000>;
|
||||
|
||||
clocks = <&gcc GMAC_CORE1_CLK>;
|
||||
|
||||
switch@10 {
|
||||
compatible = "qca,qca8337";
|
||||
/* ... */
|
||||
};
|
||||
};
|
@ -31,6 +31,7 @@ Optional properties for compatible string qcom,wcn399x-bt:
|
||||
|
||||
- max-speed: see Documentation/devicetree/bindings/serial/slave-device.txt
|
||||
- firmware-name: specify the name of nvm firmware to load
|
||||
- clocks: clock provided to the controller
|
||||
|
||||
Examples:
|
||||
|
||||
@ -57,5 +58,6 @@ serial@898000 {
|
||||
vddch0-supply = <&vreg_l25a_3p3>;
|
||||
max-speed = <3200000>;
|
||||
firmware-name = "crnv21.bin";
|
||||
clocks = <&rpmhcc RPMH_RF_CLK2>;
|
||||
};
|
||||
};
|
||||
|
225
Documentation/devicetree/bindings/net/ti,k3-am654-cpsw-nuss.yaml
Normal file
225
Documentation/devicetree/bindings/net/ti,k3-am654-cpsw-nuss.yaml
Normal file
@ -0,0 +1,225 @@
|
||||
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/net/ti,k3-am654-cpsw-nuss.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: The TI AM654x/J721E SoC Gigabit Ethernet MAC (Media Access Controller) Device Tree Bindings
|
||||
|
||||
maintainers:
|
||||
- Grygorii Strashko <grygorii.strashko@ti.com>
|
||||
- Sekhar Nori <nsekhar@ti.com>
|
||||
|
||||
description:
|
||||
The TI AM654x/J721E SoC Gigabit Ethernet MAC (CPSW2G NUSS) has two ports
|
||||
(one external) and provides Ethernet packet communication for the device.
|
||||
CPSW2G NUSS features - the Reduced Gigabit Media Independent Interface (RGMII),
|
||||
Reduced Media Independent Interface (RMII), the Management Data
|
||||
Input/Output (MDIO) interface for physical layer device (PHY) management,
|
||||
new version of Common Platform Time Sync (CPTS), updated Address Lookup
|
||||
Engine (ALE).
|
||||
One external Ethernet port (port 1) with selectable RGMII/RMII interfaces and
|
||||
an internal Communications Port Programming Interface (CPPI5) (Host port 0).
|
||||
Host Port 0 CPPI Packet Streaming Interface interface supports 8 TX channels
|
||||
and one RX channels and operating by TI AM654x/J721E NAVSS Unified DMA
|
||||
Peripheral Root Complex (UDMA-P) controller.
|
||||
The CPSW2G NUSS is integrated into device MCU domain named MCU_CPSW0.
|
||||
|
||||
Additional features
|
||||
priority level Quality Of Service (QOS) support (802.1p)
|
||||
Support for Audio/Video Bridging (P802.1Qav/D6.0)
|
||||
Support for IEEE 1588 Clock Synchronization (2008 Annex D, Annex E and Annex F)
|
||||
Flow Control (802.3x) Support
|
||||
Time Sensitive Network Support
|
||||
IEEE P902.3br/D2.0 Interspersing Express Traffic
|
||||
IEEE 802.1Qbv/D2.2 Enhancements for Scheduled Traffic
|
||||
Configurable number of addresses plus VLANs
|
||||
Configurable number of classifier/policers
|
||||
VLAN support, 802.1Q compliant, Auto add port VLAN for untagged frames on
|
||||
ingress, Auto VLAN removal on egress and auto pad to minimum frame size.
|
||||
RX/TX csum offload
|
||||
|
||||
Specifications can be found at
|
||||
http://www.ti.com/lit/ug/spruid7e/spruid7e.pdf
|
||||
http://www.ti.com/lit/ug/spruil1a/spruil1a.pdf
|
||||
|
||||
properties:
|
||||
"#address-cells": true
|
||||
"#size-cells": true
|
||||
|
||||
compatible:
|
||||
oneOf:
|
||||
- const: ti,am654-cpsw-nuss
|
||||
- const: ti,j721e-cpsw-nuss
|
||||
|
||||
reg:
|
||||
maxItems: 1
|
||||
description:
|
||||
The physical base address and size of full the CPSW2G NUSS IO range
|
||||
|
||||
reg-names:
|
||||
items:
|
||||
- const: cpsw_nuss
|
||||
|
||||
ranges: true
|
||||
|
||||
dma-coherent: true
|
||||
|
||||
clocks:
|
||||
description: CPSW2G NUSS functional clock
|
||||
|
||||
clock-names:
|
||||
items:
|
||||
- const: fck
|
||||
|
||||
power-domains:
|
||||
maxItems: 1
|
||||
|
||||
dmas:
|
||||
maxItems: 9
|
||||
|
||||
dma-names:
|
||||
items:
|
||||
- const: tx0
|
||||
- const: tx1
|
||||
- const: tx2
|
||||
- const: tx3
|
||||
- const: tx4
|
||||
- const: tx5
|
||||
- const: tx6
|
||||
- const: tx7
|
||||
- const: rx
|
||||
|
||||
ethernet-ports:
|
||||
type: object
|
||||
properties:
|
||||
'#address-cells':
|
||||
const: 1
|
||||
'#size-cells':
|
||||
const: 0
|
||||
|
||||
patternProperties:
|
||||
port@1:
|
||||
type: object
|
||||
description: CPSW2G NUSS external ports
|
||||
|
||||
allOf:
|
||||
- $ref: ethernet-controller.yaml#
|
||||
|
||||
properties:
|
||||
reg:
|
||||
items:
|
||||
- const: 1
|
||||
description: CPSW port number
|
||||
|
||||
phys:
|
||||
maxItems: 1
|
||||
description: phandle on phy-gmii-sel PHY
|
||||
|
||||
label:
|
||||
description: label associated with this port
|
||||
|
||||
ti,mac-only:
|
||||
$ref: /schemas/types.yaml#definitions/flag
|
||||
description:
|
||||
Specifies the port works in mac-only mode.
|
||||
|
||||
ti,syscon-efuse:
|
||||
$ref: /schemas/types.yaml#definitions/phandle-array
|
||||
description:
|
||||
Phandle to the system control device node which provides access
|
||||
to efuse IO range with MAC addresses
|
||||
|
||||
required:
|
||||
- reg
|
||||
- phys
|
||||
|
||||
additionalProperties: false
|
||||
|
||||
patternProperties:
|
||||
"^mdio@[0-9a-f]+$":
|
||||
type: object
|
||||
allOf:
|
||||
- $ref: "ti,davinci-mdio.yaml#"
|
||||
description:
|
||||
CPSW MDIO bus.
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- reg
|
||||
- reg-names
|
||||
- ranges
|
||||
- clocks
|
||||
- clock-names
|
||||
- power-domains
|
||||
- dmas
|
||||
- dma-names
|
||||
- '#address-cells'
|
||||
- '#size-cells'
|
||||
|
||||
additionalProperties: false
|
||||
|
||||
examples:
|
||||
- |
|
||||
#include <dt-bindings/pinctrl/k3.h>
|
||||
#include <dt-bindings/soc/ti,sci_pm_domain.h>
|
||||
#include <dt-bindings/net/ti-dp83867.h>
|
||||
|
||||
mcu_cpsw: ethernet@46000000 {
|
||||
compatible = "ti,am654-cpsw-nuss";
|
||||
#address-cells = <2>;
|
||||
#size-cells = <2>;
|
||||
reg = <0x0 0x46000000 0x0 0x200000>;
|
||||
reg-names = "cpsw_nuss";
|
||||
ranges = <0x0 0x0 0x46000000 0x0 0x200000>;
|
||||
dma-coherent;
|
||||
clocks = <&k3_clks 5 10>;
|
||||
clock-names = "fck";
|
||||
power-domains = <&k3_pds 5 TI_SCI_PD_EXCLUSIVE>;
|
||||
pinctrl-names = "default";
|
||||
pinctrl-0 = <&mcu_cpsw_pins_default &mcu_mdio_pins_default>;
|
||||
|
||||
dmas = <&mcu_udmap 0xf000>,
|
||||
<&mcu_udmap 0xf001>,
|
||||
<&mcu_udmap 0xf002>,
|
||||
<&mcu_udmap 0xf003>,
|
||||
<&mcu_udmap 0xf004>,
|
||||
<&mcu_udmap 0xf005>,
|
||||
<&mcu_udmap 0xf006>,
|
||||
<&mcu_udmap 0xf007>,
|
||||
<&mcu_udmap 0x7000>;
|
||||
dma-names = "tx0", "tx1", "tx2", "tx3", "tx4", "tx5", "tx6", "tx7",
|
||||
"rx";
|
||||
|
||||
ethernet-ports {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
cpsw_port1: port@1 {
|
||||
reg = <1>;
|
||||
ti,mac-only;
|
||||
label = "port1";
|
||||
ti,syscon-efuse = <&mcu_conf 0x200>;
|
||||
phys = <&phy_gmii_sel 1>;
|
||||
|
||||
phy-mode = "rgmii-rxid";
|
||||
phy-handle = <&phy0>;
|
||||
};
|
||||
};
|
||||
|
||||
davinci_mdio: mdio@f00 {
|
||||
compatible = "ti,cpsw-mdio","ti,davinci_mdio";
|
||||
reg = <0x0 0xf00 0x0 0x100>;
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
clocks = <&k3_clks 5 10>;
|
||||
clock-names = "fck";
|
||||
bus_freq = <1000000>;
|
||||
|
||||
phy0: ethernet-phy@0 {
|
||||
reg = <0>;
|
||||
ti,rx-internal-delay = <DP83867_RGMIIDCTL_2_00_NS>;
|
||||
ti,fifo-depth = <DP83867_PHYCR_FIFO_DEPTH_4_B_NIB>;
|
||||
};
|
||||
};
|
||||
};
|
@ -4,17 +4,27 @@ This node provides properties for configuring the MediaTek mt76xx wireless
|
||||
device. The node is expected to be specified as a child node of the PCI
|
||||
controller to which the wireless chip is connected.
|
||||
|
||||
Alternatively, it can specify the wireless part of the MT7628/MT7688 SoC.
|
||||
For SoC, use the compatible string "mediatek,mt7628-wmac" and the following
|
||||
properties:
|
||||
Alternatively, it can specify the wireless part of the MT7628/MT7688 or
|
||||
MT7622 SoC. For SoC, use the following compatible strings:
|
||||
|
||||
compatible:
|
||||
- "mediatek,mt7628-wmac" for MT7628/MT7688
|
||||
- "mediatek,mt7622-wmac" for MT7622
|
||||
|
||||
properties:
|
||||
- reg: Address and length of the register set for the device.
|
||||
- interrupts: Main device interrupt
|
||||
|
||||
MT7622 specific properties:
|
||||
- power-domains: phandle to the power domain that the WMAC is part of
|
||||
- mediatek,infracfg: phandle to the infrastructure bus fabric syscon node
|
||||
|
||||
Optional properties:
|
||||
|
||||
- ieee80211-freq-limit: See ieee80211.txt
|
||||
- mediatek,mtd-eeprom: Specify a MTD partition + offset containing EEPROM data
|
||||
- big-endian: if the radio eeprom partition is written in big-endian, specify
|
||||
this property
|
||||
|
||||
The MAC address can as well be set with corresponding optional properties
|
||||
defined in net/ethernet.txt.
|
||||
@ -31,6 +41,7 @@ Optional nodes:
|
||||
reg = <0x0000 0 0 0 0>;
|
||||
ieee80211-freq-limit = <5000000 6000000>;
|
||||
mediatek,mtd-eeprom = <&factory 0x8000>;
|
||||
big-endian;
|
||||
|
||||
led {
|
||||
led-sources = <2>;
|
||||
@ -50,3 +61,15 @@ wmac: wmac@10300000 {
|
||||
|
||||
mediatek,mtd-eeprom = <&factory 0x0000>;
|
||||
};
|
||||
|
||||
MT7622 example:
|
||||
|
||||
wmac: wmac@18000000 {
|
||||
compatible = "mediatek,mt7622-wmac";
|
||||
reg = <0 0x18000000 0 0x100000>;
|
||||
interrupts = <GIC_SPI 211 IRQ_TYPE_LEVEL_LOW>;
|
||||
|
||||
mediatek,infracfg = <&infracfg>;
|
||||
|
||||
power-domains = <&scpsys MT7622_POWER_DOMAIN_WB>;
|
||||
};
|
||||
|
@ -91,6 +91,11 @@ Optional properties:
|
||||
- qcom,msa-fixed-perm: Boolean context flag to disable SCM call for statically
|
||||
mapped msa region.
|
||||
|
||||
- qcom,coexist-support : should contain eithr "0" or "1" to indicate coex
|
||||
support by the hardware.
|
||||
- qcom,coexist-gpio-pin : gpio pin number information to support coex
|
||||
which will be used by wifi firmware.
|
||||
|
||||
Example (to supply PCI based wifi block details):
|
||||
|
||||
In this example, the node is defined as child node of the PCI controller.
|
||||
@ -159,6 +164,8 @@ wifi0: wifi@a000000 {
|
||||
qcom,msi_addr = <0x0b006040>;
|
||||
qcom,msi_base = <0x40>;
|
||||
qcom,ath10k-pre-calibration-data = [ 01 02 03 ... ];
|
||||
qcom,coexist-support = <1>;
|
||||
qcom,coexist-gpio-pin = <0x33>;
|
||||
};
|
||||
|
||||
Example (to supply wcn3990 SoC wifi block details):
|
||||
|
@ -9,11 +9,12 @@ Required properties:
|
||||
- spi-max-frequency : Maximum SPI clocking speed of device in Hz
|
||||
- interrupts : Should contain interrupt line
|
||||
- vio-supply : phandle to regulator providing VIO
|
||||
- ti,power-gpio : GPIO connected to chip's PMEN pin
|
||||
|
||||
Optional properties:
|
||||
- ti,wl1251-has-eeprom : boolean, the wl1251 has an eeprom connected, which
|
||||
provides configuration data (calibration, MAC, ...)
|
||||
- ti,power-gpio : GPIO connected to chip's PMEN pin if operated in
|
||||
SPI mode
|
||||
- Please consult Documentation/devicetree/bindings/spi/spi-bus.txt
|
||||
for optional SPI connection related properties,
|
||||
|
||||
|
45
Documentation/devicetree/bindings/ptp/ptp-idt82p33.yaml
Normal file
45
Documentation/devicetree/bindings/ptp/ptp-idt82p33.yaml
Normal file
@ -0,0 +1,45 @@
|
||||
# SPDX-License-Identifier: (GPL-2.0 OR BSD-2-Clause)
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/ptp/ptp-idt82p33.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: IDT 82P33 PTP Clock Device Tree Bindings
|
||||
|
||||
description: |
|
||||
IDT 82P33XXX Synchronization Management Unit (SMU) based PTP clock
|
||||
|
||||
maintainers:
|
||||
- Min Li <min.li.xe@renesas.com>
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- idt,82p33810
|
||||
- idt,82p33813
|
||||
- idt,82p33814
|
||||
- idt,82p33831
|
||||
- idt,82p33910
|
||||
- idt,82p33913
|
||||
- idt,82p33914
|
||||
- idt,82p33931
|
||||
|
||||
reg:
|
||||
maxItems: 1
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- reg
|
||||
|
||||
additionalProperties: false
|
||||
|
||||
examples:
|
||||
- |
|
||||
i2c {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
phc@51 {
|
||||
compatible = "idt,82p33810";
|
||||
reg = <0x51>;
|
||||
};
|
||||
};
|
@ -1,37 +1,40 @@
|
||||
.. SPDX-License-Identifier: GPL-2.0
|
||||
|
||||
Netdev private dataroom for 6lowpan interfaces:
|
||||
==============================================
|
||||
Netdev private dataroom for 6lowpan interfaces
|
||||
==============================================
|
||||
|
||||
All 6lowpan able net devices, means all interfaces with ARPHRD_6LOWPAN,
|
||||
must have "struct lowpan_priv" placed at beginning of netdev_priv.
|
||||
|
||||
The priv_size of each interface should be calculate by:
|
||||
The priv_size of each interface should be calculate by::
|
||||
|
||||
dev->priv_size = LOWPAN_PRIV_SIZE(LL_6LOWPAN_PRIV_DATA);
|
||||
|
||||
Where LL_PRIV_6LOWPAN_DATA is sizeof linklayer 6lowpan private data struct.
|
||||
To access the LL_PRIV_6LOWPAN_DATA structure you can cast:
|
||||
To access the LL_PRIV_6LOWPAN_DATA structure you can cast::
|
||||
|
||||
lowpan_priv(dev)-priv;
|
||||
|
||||
to your LL_6LOWPAN_PRIV_DATA structure.
|
||||
|
||||
Before registering the lowpan netdev interface you must run:
|
||||
Before registering the lowpan netdev interface you must run::
|
||||
|
||||
lowpan_netdev_setup(dev, LOWPAN_LLTYPE_FOOBAR);
|
||||
|
||||
wheres LOWPAN_LLTYPE_FOOBAR is a define for your 6LoWPAN linklayer type of
|
||||
enum lowpan_lltypes.
|
||||
|
||||
Example to evaluate the private usually you can do:
|
||||
Example to evaluate the private usually you can do::
|
||||
|
||||
static inline struct lowpan_priv_foobar *
|
||||
lowpan_foobar_priv(struct net_device *dev)
|
||||
{
|
||||
static inline struct lowpan_priv_foobar *
|
||||
lowpan_foobar_priv(struct net_device *dev)
|
||||
{
|
||||
return (struct lowpan_priv_foobar *)lowpan_priv(dev)->priv;
|
||||
}
|
||||
}
|
||||
|
||||
switch (dev->type) {
|
||||
case ARPHRD_6LOWPAN:
|
||||
switch (dev->type) {
|
||||
case ARPHRD_6LOWPAN:
|
||||
lowpan_priv = lowpan_priv(dev);
|
||||
/* do great stuff which is ARPHRD_6LOWPAN related */
|
||||
switch (lowpan_priv->lltype) {
|
||||
@ -42,8 +45,8 @@ case ARPHRD_6LOWPAN:
|
||||
...
|
||||
}
|
||||
break;
|
||||
...
|
||||
}
|
||||
...
|
||||
}
|
||||
|
||||
In case of generic 6lowpan branch ("net/6lowpan") you can remove the check
|
||||
on ARPHRD_6LOWPAN, because you can be sure that these function are called
|
52
Documentation/networking/bareudp.rst
Normal file
52
Documentation/networking/bareudp.rst
Normal file
@ -0,0 +1,52 @@
|
||||
.. SPDX-License-Identifier: GPL-2.0
|
||||
|
||||
========================================
|
||||
Bare UDP Tunnelling Module Documentation
|
||||
========================================
|
||||
|
||||
There are various L3 encapsulation standards using UDP being discussed to
|
||||
leverage the UDP based load balancing capability of different networks.
|
||||
MPLSoUDP (__ https://tools.ietf.org/html/rfc7510) is one among them.
|
||||
|
||||
The Bareudp tunnel module provides a generic L3 encapsulation tunnelling
|
||||
support for tunnelling different L3 protocols like MPLS, IP, NSH etc. inside
|
||||
a UDP tunnel.
|
||||
|
||||
Special Handling
|
||||
----------------
|
||||
The bareudp device supports special handling for MPLS & IP as they can have
|
||||
multiple ethertypes.
|
||||
MPLS procotcol can have ethertypes ETH_P_MPLS_UC (unicast) & ETH_P_MPLS_MC (multicast).
|
||||
IP protocol can have ethertypes ETH_P_IP (v4) & ETH_P_IPV6 (v6).
|
||||
This special handling can be enabled only for ethertypes ETH_P_IP & ETH_P_MPLS_UC
|
||||
with a flag called multiproto mode.
|
||||
|
||||
Usage
|
||||
------
|
||||
|
||||
1) Device creation & deletion
|
||||
|
||||
a) ip link add dev bareudp0 type bareudp dstport 6635 ethertype 0x8847.
|
||||
|
||||
This creates a bareudp tunnel device which tunnels L3 traffic with ethertype
|
||||
0x8847 (MPLS traffic). The destination port of the UDP header will be set to
|
||||
6635.The device will listen on UDP port 6635 to receive traffic.
|
||||
|
||||
b) ip link delete bareudp0
|
||||
|
||||
2) Device creation with multiple proto mode enabled
|
||||
|
||||
There are two ways to create a bareudp device for MPLS & IP with multiproto mode
|
||||
enabled.
|
||||
|
||||
a) ip link add dev bareudp0 type bareudp dstport 6635 ethertype 0x8847 multiproto
|
||||
|
||||
b) ip link add dev bareudp0 type bareudp dstport 6635 ethertype mpls
|
||||
|
||||
3) Device Usage
|
||||
|
||||
The bareudp device could be used along with OVS or flower filter in TC.
|
||||
The OVS or TC flower layer must set the tunnel information in SKB dst field before
|
||||
sending packet buffer to the bareudp device for transmission. On reception the
|
||||
bareudp device extracts and stores the tunnel information in SKB dst field before
|
||||
passing the packet buffer to the network stack.
|
@ -101,7 +101,7 @@ Enabling the driver and kconfig options
|
||||
**External options** ( Choose if the corresponding mlx5 feature is required )
|
||||
|
||||
- CONFIG_PTP_1588_CLOCK: When chosen, mlx5 ptp support will be enabled
|
||||
- CONFIG_VXLAN: When chosen, mlx5 vxaln support will be enabled.
|
||||
- CONFIG_VXLAN: When chosen, mlx5 vxlan support will be enabled.
|
||||
- CONFIG_MLXFW: When chosen, mlx5 firmware flashing support will be enabled (via devlink and ethtool).
|
||||
|
||||
Devlink info
|
||||
|
@ -32,7 +32,8 @@ is also supported.
|
||||
DesignWare(R) Cores Ethernet MAC 10/100/1000 Universal version 3.70a
|
||||
(and older) and DesignWare(R) Cores Ethernet Quality-of-Service version 4.0
|
||||
(and upper) have been used for developing this driver as well as
|
||||
DesignWare(R) Cores XGMAC - 10G Ethernet MAC.
|
||||
DesignWare(R) Cores XGMAC - 10G Ethernet MAC and DesignWare(R) Cores
|
||||
Enterprise MAC - 100G Ethernet MAC.
|
||||
|
||||
This driver supports both the platform bus and PCI.
|
||||
|
||||
@ -48,6 +49,8 @@ Cores Ethernet Controllers and corresponding minimum and maximum versions:
|
||||
+-------------------------------+--------------+--------------+--------------+
|
||||
| XGMAC - 10G Ethernet MAC | 2.10a | N/A | XGMAC2+ |
|
||||
+-------------------------------+--------------+--------------+--------------+
|
||||
| XLGMAC - 100G Ethernet MAC | 2.00a | N/A | XLGMAC2+ |
|
||||
+-------------------------------+--------------+--------------+--------------+
|
||||
|
||||
For questions related to hardware requirements, refer to the documentation
|
||||
supplied with your Ethernet adapter. All hardware requirements listed apply
|
||||
@ -57,7 +60,7 @@ Feature List
|
||||
============
|
||||
|
||||
The following features are available in this driver:
|
||||
- GMII/MII/RGMII/SGMII/RMII/XGMII Interface
|
||||
- GMII/MII/RGMII/SGMII/RMII/XGMII/XLGMII Interface
|
||||
- Half-Duplex / Full-Duplex Operation
|
||||
- Energy Efficient Ethernet (EEE)
|
||||
- IEEE 802.3x PAUSE Packets (Flow Control)
|
||||
|
@ -51,6 +51,9 @@ The ``bnxt_en`` driver reports the following versions
|
||||
* - Name
|
||||
- Type
|
||||
- Description
|
||||
* - ``board.id``
|
||||
- fixed
|
||||
- Part number identifying the board design
|
||||
* - ``asic.id``
|
||||
- fixed
|
||||
- ASIC design identifier
|
||||
@ -63,12 +66,15 @@ The ``bnxt_en`` driver reports the following versions
|
||||
* - ``fw``
|
||||
- stored, running
|
||||
- Overall board firmware version
|
||||
* - ``fw.app``
|
||||
- stored, running
|
||||
- Data path firmware version
|
||||
* - ``fw.mgmt``
|
||||
- stored, running
|
||||
- Management firmware version
|
||||
- NIC hardware resource management firmware version
|
||||
* - ``fw.mgmt.api``
|
||||
- running
|
||||
- Minimum firmware interface spec version supported between driver and firmware
|
||||
* - ``fw.nsci``
|
||||
- stored, running
|
||||
- General platform management firmware version
|
||||
* - ``fw.roce``
|
||||
- stored, running
|
||||
- RoCE management firmware version
|
||||
|
93
Documentation/networking/devlink/devlink-flash.rst
Normal file
93
Documentation/networking/devlink/devlink-flash.rst
Normal file
@ -0,0 +1,93 @@
|
||||
.. SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
|
||||
|
||||
.. _devlink_flash:
|
||||
|
||||
=============
|
||||
Devlink Flash
|
||||
=============
|
||||
|
||||
The ``devlink-flash`` API allows updating device firmware. It replaces the
|
||||
older ``ethtool-flash`` mechanism, and doesn't require taking any
|
||||
networking locks in the kernel to perform the flash update. Example use::
|
||||
|
||||
$ devlink dev flash pci/0000:05:00.0 file flash-boot.bin
|
||||
|
||||
Note that the file name is a path relative to the firmware loading path
|
||||
(usually ``/lib/firmware/``). Drivers may send status updates to inform
|
||||
user space about the progress of the update operation.
|
||||
|
||||
Firmware Loading
|
||||
================
|
||||
|
||||
Devices which require firmware to operate usually store it in non-volatile
|
||||
memory on the board, e.g. flash. Some devices store only basic firmware on
|
||||
the board, and the driver loads the rest from disk during probing.
|
||||
``devlink-info`` allows users to query firmware information (loaded
|
||||
components and versions).
|
||||
|
||||
In other cases the device can both store the image on the board, load from
|
||||
disk, or automatically flash a new image from disk. The ``fw_load_policy``
|
||||
devlink parameter can be used to control this behavior
|
||||
(:ref:`Documentation/networking/devlink/devlink-params.rst <devlink_params_generic>`).
|
||||
|
||||
On-disk firmware files are usually stored in ``/lib/firmware/``.
|
||||
|
||||
Firmware Version Management
|
||||
===========================
|
||||
|
||||
Drivers are expected to implement ``devlink-flash`` and ``devlink-info``
|
||||
functionality, which together allow for implementing vendor-independent
|
||||
automated firmware update facilities.
|
||||
|
||||
``devlink-info`` exposes the ``driver`` name and three version groups
|
||||
(``fixed``, ``running``, ``stored``).
|
||||
|
||||
The ``driver`` attribute and ``fixed`` group identify the specific device
|
||||
design, e.g. for looking up applicable firmware updates. This is why
|
||||
``serial_number`` is not part of the ``fixed`` versions (even though it
|
||||
is fixed) - ``fixed`` versions should identify the design, not a single
|
||||
device.
|
||||
|
||||
``running`` and ``stored`` firmware versions identify the firmware running
|
||||
on the device, and firmware which will be activated after reboot or device
|
||||
reset.
|
||||
|
||||
The firmware update agent is supposed to be able to follow this simple
|
||||
algorithm to update firmware contents, regardless of the device vendor:
|
||||
|
||||
.. code-block:: sh
|
||||
|
||||
# Get unique HW design identifier
|
||||
$hw_id = devlink-dev-info['fixed']
|
||||
|
||||
# Find out which FW flash we want to use for this NIC
|
||||
$want_flash_vers = some-db-backed.lookup($hw_id, 'flash')
|
||||
|
||||
# Update flash if necessary
|
||||
if $want_flash_vers != devlink-dev-info['stored']:
|
||||
$file = some-db-backed.download($hw_id, 'flash')
|
||||
devlink-dev-flash($file)
|
||||
|
||||
# Find out the expected overall firmware versions
|
||||
$want_fw_vers = some-db-backed.lookup($hw_id, 'all')
|
||||
|
||||
# Update on-disk file if necessary
|
||||
if $want_fw_vers != devlink-dev-info['running']:
|
||||
$file = some-db-backed.download($hw_id, 'disk')
|
||||
write($file, '/lib/firmware/')
|
||||
|
||||
# Try device reset, if available
|
||||
if $want_fw_vers != devlink-dev-info['running']:
|
||||
devlink-reset()
|
||||
|
||||
# Reboot, if reset wasn't enough
|
||||
if $want_fw_vers != devlink-dev-info['running']:
|
||||
reboot()
|
||||
|
||||
Note that each reference to ``devlink-dev-info`` in this pseudo-code
|
||||
is expected to fetch up-to-date information from the kernel.
|
||||
|
||||
For the convenience of identifying firmware files some vendors add
|
||||
``bundle_id`` information to the firmware versions. This meta-version covers
|
||||
multiple per-component versions and can be used e.g. in firmware file names
|
||||
(all component versions could get rather long.)
|
@ -5,34 +5,119 @@ Devlink Info
|
||||
============
|
||||
|
||||
The ``devlink-info`` mechanism enables device drivers to report device
|
||||
information in a generic fashion. It is extensible, and enables exporting
|
||||
even device or driver specific information.
|
||||
(hardware and firmware) information in a standard, extensible fashion.
|
||||
|
||||
devlink supports representing the following types of versions
|
||||
The original motivation for the ``devlink-info`` API was twofold:
|
||||
|
||||
.. list-table:: List of version types
|
||||
- making it possible to automate device and firmware management in a fleet
|
||||
of machines in a vendor-independent fashion (see also
|
||||
:ref:`Documentation/networking/devlink/devlink-flash.rst <devlink_flash>`);
|
||||
- name the per component FW versions (as opposed to the crowded ethtool
|
||||
version string).
|
||||
|
||||
``devlink-info`` supports reporting multiple types of objects. Reporting driver
|
||||
versions is generally discouraged - here, and via any other Linux API.
|
||||
|
||||
.. list-table:: List of top level info objects
|
||||
:widths: 5 95
|
||||
|
||||
* - Type
|
||||
* - Name
|
||||
- Description
|
||||
* - ``driver``
|
||||
- Name of the currently used device driver, also available through sysfs.
|
||||
|
||||
* - ``serial_number``
|
||||
- Serial number of the device.
|
||||
|
||||
This is usually the serial number of the ASIC, also often available
|
||||
in PCI config space of the device in the *Device Serial Number*
|
||||
capability.
|
||||
|
||||
The serial number should be unique per physical device.
|
||||
Sometimes the serial number of the device is only 48 bits long (the
|
||||
length of the Ethernet MAC address), and since PCI DSN is 64 bits long
|
||||
devices pad or encode additional information into the serial number.
|
||||
One example is adding port ID or PCI interface ID in the extra two bytes.
|
||||
Drivers should make sure to strip or normalize any such padding
|
||||
or interface ID, and report only the part of the serial number
|
||||
which uniquely identifies the hardware. In other words serial number
|
||||
reported for two ports of the same device or on two hosts of
|
||||
a multi-host device should be identical.
|
||||
|
||||
.. note:: ``devlink-info`` API should be extended with a new field
|
||||
if devices want to report board/product serial number (often
|
||||
reported in PCI *Vital Product Data* capability).
|
||||
|
||||
* - ``fixed``
|
||||
- Represents fixed versions, which cannot change. For example,
|
||||
- Group for hardware identifiers, and versions of components
|
||||
which are not field-updatable.
|
||||
|
||||
Versions in this section identify the device design. For example,
|
||||
component identifiers or the board version reported in the PCI VPD.
|
||||
Data in ``devlink-info`` should be broken into the smallest logical
|
||||
components, e.g. PCI VPD may concatenate various information
|
||||
to form the Part Number string, while in ``devlink-info`` all parts
|
||||
should be reported as separate items.
|
||||
|
||||
This group must not contain any frequently changing identifiers,
|
||||
such as serial numbers. See
|
||||
:ref:`Documentation/networking/devlink/devlink-flash.rst <devlink_flash>`
|
||||
to understand why.
|
||||
|
||||
* - ``running``
|
||||
- Represents the version of the currently running component. For
|
||||
example the running version of firmware. These versions generally
|
||||
only update after a reboot.
|
||||
- Group for information about currently running software/firmware.
|
||||
These versions often only update after a reboot, sometimes device reset.
|
||||
|
||||
* - ``stored``
|
||||
- Represents the version of a component as stored, such as after a
|
||||
flash update. Stored values should update to reflect changes in the
|
||||
flash even if a reboot has not yet occurred.
|
||||
- Group for software/firmware versions in device flash.
|
||||
|
||||
Stored values must update to reflect changes in the flash even
|
||||
if reboot has not yet occurred. If device is not capable of updating
|
||||
``stored`` versions when new software is flashed, it must not report
|
||||
them.
|
||||
|
||||
Each version can be reported at most once in each version group. Firmware
|
||||
components stored on the flash should feature in both the ``running`` and
|
||||
``stored`` sections, if device is capable of reporting ``stored`` versions
|
||||
(see :ref:`Documentation/networking/devlink/devlink-flash.rst <devlink_flash>`).
|
||||
In case software/firmware components are loaded from the disk (e.g.
|
||||
``/lib/firmware``) only the running version should be reported via
|
||||
the kernel API.
|
||||
|
||||
Generic Versions
|
||||
================
|
||||
|
||||
It is expected that drivers use the following generic names for exporting
|
||||
version information. Other information may be exposed using driver-specific
|
||||
names, but these should be documented in the driver-specific file.
|
||||
version information. If a generic name for a given component doesn't exist yet,
|
||||
driver authors should consult existing driver-specific versions and attempt
|
||||
reuse. As last resort, if a component is truly unique, using driver-specific
|
||||
names is allowed, but these should be documented in the driver-specific file.
|
||||
|
||||
All versions should try to use the following terminology:
|
||||
|
||||
.. list-table:: List of common version suffixes
|
||||
:widths: 10 90
|
||||
|
||||
* - Name
|
||||
- Description
|
||||
* - ``id``, ``revision``
|
||||
- Identifiers of designs and revision, mostly used for hardware versions.
|
||||
|
||||
* - ``api``
|
||||
- Version of API between components. API items are usually of limited
|
||||
value to the user, and can be inferred from other versions by the vendor,
|
||||
so adding API versions is generally discouraged as noise.
|
||||
|
||||
* - ``bundle_id``
|
||||
- Identifier of a distribution package which was flashed onto the device.
|
||||
This is an attribute of a firmware package which covers multiple versions
|
||||
for ease of managing firmware images (see
|
||||
:ref:`Documentation/networking/devlink/devlink-flash.rst <devlink_flash>`).
|
||||
|
||||
``bundle_id`` can appear in both ``running`` and ``stored`` versions,
|
||||
but it must not be reported if any of the components covered by the
|
||||
``bundle_id`` was changed and no longer matches the version from
|
||||
the bundle.
|
||||
|
||||
board.id
|
||||
--------
|
||||
@ -52,7 +137,7 @@ ASIC design identifier.
|
||||
asic.rev
|
||||
--------
|
||||
|
||||
ASIC design revision.
|
||||
ASIC design revision/stepping.
|
||||
|
||||
board.manufacture
|
||||
-----------------
|
||||
@ -72,6 +157,12 @@ Control unit firmware version. This firmware is responsible for house
|
||||
keeping tasks, PHY control etc. but not the packet-by-packet data path
|
||||
operation.
|
||||
|
||||
fw.mgmt.api
|
||||
-----------
|
||||
|
||||
Firmware interface specification version of the software interfaces between
|
||||
driver and firmware.
|
||||
|
||||
fw.app
|
||||
------
|
||||
|
||||
@ -91,10 +182,31 @@ Network Controller Sideband Interface.
|
||||
fw.psid
|
||||
-------
|
||||
|
||||
Unique identifier of the firmware parameter set.
|
||||
Unique identifier of the firmware parameter set. These are usually
|
||||
parameters of a particular board, defined at manufacturing time.
|
||||
|
||||
fw.roce
|
||||
-------
|
||||
|
||||
RoCE firmware version which is responsible for handling roce
|
||||
management.
|
||||
|
||||
fw.bundle_id
|
||||
------------
|
||||
|
||||
Unique identifier of the entire firmware bundle.
|
||||
|
||||
Future work
|
||||
===========
|
||||
|
||||
The following extensions could be useful:
|
||||
|
||||
- product serial number - NIC boards often get labeled with a board serial
|
||||
number rather than ASIC serial number; it'd be useful to add board serial
|
||||
numbers to the API if they can be retrieved from the device;
|
||||
|
||||
- on-disk firmware file names - drivers list the file names of firmware they
|
||||
may need to load onto devices via the ``MODULE_FIRMWARE()`` macro. These,
|
||||
however, are per module, rather than per device. It'd be useful to list
|
||||
the names of firmware files the driver will try to load for a given device,
|
||||
in order of priority.
|
||||
|
@ -41,6 +41,8 @@ In order for ``driverinit`` parameters to take effect, the driver must
|
||||
support reloading via the ``devlink-reload`` command. This command will
|
||||
request a reload of the device driver.
|
||||
|
||||
.. _devlink_params_generic:
|
||||
|
||||
Generic configuration parameters
|
||||
================================
|
||||
The following is a list of generic configuration parameters that drivers may
|
||||
|
@ -20,6 +20,11 @@ address regions that are otherwise inaccessible to the user.
|
||||
Regions may also be used to provide an additional way to debug complex error
|
||||
states, but see also :doc:`devlink-health`
|
||||
|
||||
Regions may optionally support capturing a snapshot on demand via the
|
||||
``DEVLINK_CMD_REGION_NEW`` netlink message. A driver wishing to allow
|
||||
requested snapshots must implement the ``.snapshot`` callback for the region
|
||||
in its ``devlink_region_ops`` structure.
|
||||
|
||||
example usage
|
||||
-------------
|
||||
|
||||
@ -29,8 +34,7 @@ example usage
|
||||
$ devlink region show [ DEV/REGION ]
|
||||
$ devlink region del DEV/REGION snapshot SNAPSHOT_ID
|
||||
$ devlink region dump DEV/REGION [ snapshot SNAPSHOT_ID ]
|
||||
$ devlink region read DEV/REGION [ snapshot SNAPSHOT_ID ]
|
||||
address ADDRESS length length
|
||||
$ devlink region read DEV/REGION [ snapshot SNAPSHOT_ID ] address ADDRESS length length
|
||||
|
||||
# Show all of the exposed regions with region sizes:
|
||||
$ devlink region show
|
||||
@ -40,6 +44,9 @@ example usage
|
||||
# Delete a snapshot using:
|
||||
$ devlink region del pci/0000:00:05.0/cr-space snapshot 1
|
||||
|
||||
# Request an immediate snapshot, if supported by the region
|
||||
$ devlink region new pci/0000:00:05.0/cr-space snapshot 5
|
||||
|
||||
# Dump a snapshot:
|
||||
$ devlink region dump pci/0000:00:05.0/fw-health snapshot 1
|
||||
0000000000000000 0014 95dc 0014 9514 0035 1670 0034 db30
|
||||
@ -48,8 +55,7 @@ example usage
|
||||
0000000000000030 bada cce5 bada cce5 bada cce5 bada cce5
|
||||
|
||||
# Read a specific part of a snapshot:
|
||||
$ devlink region read pci/0000:00:05.0/fw-health snapshot 1 address 0
|
||||
length 16
|
||||
$ devlink region read pci/0000:00:05.0/fw-health snapshot 1 address 0 length 16
|
||||
0000000000000000 0014 95dc 0014 9514 0035 1670 0034 db30
|
||||
|
||||
As regions are likely very device or driver specific, no generic regions are
|
||||
|
@ -238,6 +238,12 @@ be added to the following table:
|
||||
- ``drop``
|
||||
- Traps NVE packets that the device decided to drop because their overlay
|
||||
source MAC is multicast
|
||||
* - ``ingress_flow_action_drop``
|
||||
- ``drop``
|
||||
- Traps packets dropped during processing of ingress flow action drop
|
||||
* - ``egress_flow_action_drop``
|
||||
- ``drop``
|
||||
- Traps packets dropped during processing of egress flow action drop
|
||||
|
||||
Driver-specific Packet Traps
|
||||
============================
|
||||
@ -277,6 +283,35 @@ narrow. The description of these groups must be added to the following table:
|
||||
* - ``tunnel_drops``
|
||||
- Contains packet traps for packets that were dropped by the device during
|
||||
tunnel encapsulation / decapsulation
|
||||
* - ``acl_drops``
|
||||
- Contains packet traps for packets that were dropped by the device during
|
||||
ACL processing
|
||||
|
||||
Packet Trap Policers
|
||||
====================
|
||||
|
||||
As previously explained, the underlying device can trap certain packets to the
|
||||
CPU for processing. In most cases, the underlying device is capable of handling
|
||||
packet rates that are several orders of magnitude higher compared to those that
|
||||
can be handled by the CPU.
|
||||
|
||||
Therefore, in order to prevent the underlying device from overwhelming the CPU,
|
||||
devices usually include packet trap policers that are able to police the
|
||||
trapped packets to rates that can be handled by the CPU.
|
||||
|
||||
The ``devlink-trap`` mechanism allows capable device drivers to register their
|
||||
supported packet trap policers with ``devlink``. The device driver can choose
|
||||
to associate these policers with supported packet trap groups (see
|
||||
:ref:`Generic-Packet-Trap-Groups`) during its initialization, thereby exposing
|
||||
its default control plane policy to user space.
|
||||
|
||||
Device drivers should allow user space to change the parameters of the policers
|
||||
(e.g., rate, burst size) as well as the association between the policers and
|
||||
trap groups by implementing the relevant callbacks.
|
||||
|
||||
If possible, device drivers should implement a callback that allows user space
|
||||
to retrieve the number of packets that were dropped by the policer because its
|
||||
configured policy was violated.
|
||||
|
||||
Testing
|
||||
=======
|
||||
|
96
Documentation/networking/devlink/ice.rst
Normal file
96
Documentation/networking/devlink/ice.rst
Normal file
@ -0,0 +1,96 @@
|
||||
.. SPDX-License-Identifier: GPL-2.0
|
||||
|
||||
===================
|
||||
ice devlink support
|
||||
===================
|
||||
|
||||
This document describes the devlink features implemented by the ``ice``
|
||||
device driver.
|
||||
|
||||
Info versions
|
||||
=============
|
||||
|
||||
The ``ice`` driver reports the following versions
|
||||
|
||||
.. list-table:: devlink info versions implemented
|
||||
:widths: 5 5 5 90
|
||||
|
||||
* - Name
|
||||
- Type
|
||||
- Example
|
||||
- Description
|
||||
* - ``board.id``
|
||||
- fixed
|
||||
- K65390-000
|
||||
- The Product Board Assembly (PBA) identifier of the board.
|
||||
* - ``fw.mgmt``
|
||||
- running
|
||||
- 2.1.7
|
||||
- 3-digit version number of the management firmware that controls the
|
||||
PHY, link, etc.
|
||||
* - ``fw.mgmt.api``
|
||||
- running
|
||||
- 1.5
|
||||
- 2-digit version number of the API exported over the AdminQ by the
|
||||
management firmware. Used by the driver to identify what commands
|
||||
are supported.
|
||||
* - ``fw.mgmt.build``
|
||||
- running
|
||||
- 0x305d955f
|
||||
- Unique identifier of the source for the management firmware.
|
||||
* - ``fw.undi``
|
||||
- running
|
||||
- 1.2581.0
|
||||
- Version of the Option ROM containing the UEFI driver. The version is
|
||||
reported in ``major.minor.patch`` format. The major version is
|
||||
incremented whenever a major breaking change occurs, or when the
|
||||
minor version would overflow. The minor version is incremented for
|
||||
non-breaking changes and reset to 1 when the major version is
|
||||
incremented. The patch version is normally 0 but is incremented when
|
||||
a fix is delivered as a patch against an older base Option ROM.
|
||||
* - ``fw.psid.api``
|
||||
- running
|
||||
- 0.80
|
||||
- Version defining the format of the flash contents.
|
||||
* - ``fw.bundle_id``
|
||||
- running
|
||||
- 0x80002ec0
|
||||
- Unique identifier of the firmware image file that was loaded onto
|
||||
the device. Also referred to as the EETRACK identifier of the NVM.
|
||||
* - ``fw.app.name``
|
||||
- running
|
||||
- ICE OS Default Package
|
||||
- The name of the DDP package that is active in the device. The DDP
|
||||
package is loaded by the driver during initialization. Each varation
|
||||
of DDP package shall have a unique name.
|
||||
* - ``fw.app``
|
||||
- running
|
||||
- 1.3.1.0
|
||||
- The version of the DDP package that is active in the device. Note
|
||||
that both the name (as reported by ``fw.app.name``) and version are
|
||||
required to uniquely identify the package.
|
||||
|
||||
Regions
|
||||
=======
|
||||
|
||||
The ``ice`` driver enables access to the contents of the Non Volatile Memory
|
||||
flash chip via the ``nvm-flash`` region.
|
||||
|
||||
Users can request an immediate capture of a snapshot via the
|
||||
``DEVLINK_CMD_REGION_NEW``
|
||||
|
||||
.. code:: shell
|
||||
|
||||
$ devlink region new pci/0000:01:00.0/nvm-flash snapshot 1
|
||||
$ devlink region dump pci/0000:01:00.0/nvm-flash snapshot 1
|
||||
|
||||
$ devlink region dump pci/0000:01:00.0/nvm-flash snapshot 1
|
||||
0000000000000000 0014 95dc 0014 9514 0035 1670 0034 db30
|
||||
0000000000000010 0000 0000 ffff ff04 0029 8c00 0028 8cc8
|
||||
0000000000000020 0016 0bb8 0016 1720 0000 0000 c00f 3ffc
|
||||
0000000000000030 bada cce5 bada cce5 bada cce5 bada cce5
|
||||
|
||||
$ devlink region read pci/0000:01:00.0/nvm-flash snapshot 1 address 0 length 16
|
||||
0000000000000000 0014 95dc 0014 9514 0035 1670 0034 db30
|
||||
|
||||
$ devlink region delete pci/0000:01:00.0/nvm-flash snapshot 1
|
@ -16,6 +16,7 @@ general.
|
||||
devlink-dpipe
|
||||
devlink-health
|
||||
devlink-info
|
||||
devlink-flash
|
||||
devlink-params
|
||||
devlink-region
|
||||
devlink-resource
|
||||
@ -32,6 +33,7 @@ parameters, info versions, and other features it supports.
|
||||
|
||||
bnxt
|
||||
ionic
|
||||
ice
|
||||
mlx4
|
||||
mlx5
|
||||
mlxsw
|
||||
|
@ -37,6 +37,12 @@ parameters.
|
||||
* ``smfs`` Software managed flow steering. In SMFS mode, the HW
|
||||
steering entities are created and manage through the driver without
|
||||
firmware intervention.
|
||||
* - ``fdb_large_groups``
|
||||
- u32
|
||||
- driverinit
|
||||
- Control the number of large groups (size > 1) in the FDB table.
|
||||
|
||||
* The default value is 15, and the range is between 1 and 1024.
|
||||
|
||||
The ``mlx5`` driver supports reloading via ``DEVLINK_CMD_RELOAD``
|
||||
|
||||
|
@ -189,6 +189,21 @@ Userspace to kernel:
|
||||
``ETHTOOL_MSG_DEBUG_SET`` set debugging settings
|
||||
``ETHTOOL_MSG_WOL_GET`` get wake-on-lan settings
|
||||
``ETHTOOL_MSG_WOL_SET`` set wake-on-lan settings
|
||||
``ETHTOOL_MSG_FEATURES_GET`` get device features
|
||||
``ETHTOOL_MSG_FEATURES_SET`` set device features
|
||||
``ETHTOOL_MSG_PRIVFLAGS_GET`` get private flags
|
||||
``ETHTOOL_MSG_PRIVFLAGS_SET`` set private flags
|
||||
``ETHTOOL_MSG_RINGS_GET`` get ring sizes
|
||||
``ETHTOOL_MSG_RINGS_SET`` set ring sizes
|
||||
``ETHTOOL_MSG_CHANNELS_GET`` get channel counts
|
||||
``ETHTOOL_MSG_CHANNELS_SET`` set channel counts
|
||||
``ETHTOOL_MSG_COALESCE_GET`` get coalescing parameters
|
||||
``ETHTOOL_MSG_COALESCE_SET`` set coalescing parameters
|
||||
``ETHTOOL_MSG_PAUSE_GET`` get pause parameters
|
||||
``ETHTOOL_MSG_PAUSE_SET`` set pause parameters
|
||||
``ETHTOOL_MSG_EEE_GET`` get EEE settings
|
||||
``ETHTOOL_MSG_EEE_SET`` set EEE settings
|
||||
``ETHTOOL_MSG_TSINFO_GET`` get timestamping info
|
||||
===================================== ================================
|
||||
|
||||
Kernel to userspace:
|
||||
@ -204,6 +219,22 @@ Kernel to userspace:
|
||||
``ETHTOOL_MSG_DEBUG_NTF`` debugging settings notification
|
||||
``ETHTOOL_MSG_WOL_GET_REPLY`` wake-on-lan settings
|
||||
``ETHTOOL_MSG_WOL_NTF`` wake-on-lan settings notification
|
||||
``ETHTOOL_MSG_FEATURES_GET_REPLY`` device features
|
||||
``ETHTOOL_MSG_FEATURES_SET_REPLY`` optional reply to FEATURES_SET
|
||||
``ETHTOOL_MSG_FEATURES_NTF`` netdev features notification
|
||||
``ETHTOOL_MSG_PRIVFLAGS_GET_REPLY`` private flags
|
||||
``ETHTOOL_MSG_PRIVFLAGS_NTF`` private flags
|
||||
``ETHTOOL_MSG_RINGS_GET_REPLY`` ring sizes
|
||||
``ETHTOOL_MSG_RINGS_NTF`` ring sizes
|
||||
``ETHTOOL_MSG_CHANNELS_GET_REPLY`` channel counts
|
||||
``ETHTOOL_MSG_CHANNELS_NTF`` channel counts
|
||||
``ETHTOOL_MSG_COALESCE_GET_REPLY`` coalescing parameters
|
||||
``ETHTOOL_MSG_COALESCE_NTF`` coalescing parameters
|
||||
``ETHTOOL_MSG_PAUSE_GET_REPLY`` pause parameters
|
||||
``ETHTOOL_MSG_PAUSE_NTF`` pause parameters
|
||||
``ETHTOOL_MSG_EEE_GET_REPLY`` EEE settings
|
||||
``ETHTOOL_MSG_EEE_NTF`` EEE settings
|
||||
``ETHTOOL_MSG_TSINFO_GET_REPLY`` timestamping info
|
||||
===================================== =================================
|
||||
|
||||
``GET`` requests are sent by userspace applications to retrieve device
|
||||
@ -521,6 +552,410 @@ Request contents:
|
||||
``WAKE_MAGICSECURE`` mode.
|
||||
|
||||
|
||||
FEATURES_GET
|
||||
============
|
||||
|
||||
Gets netdev features like ``ETHTOOL_GFEATURES`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
==================================== ====== ==========================
|
||||
``ETHTOOL_A_FEATURES_HEADER`` nested request header
|
||||
==================================== ====== ==========================
|
||||
|
||||
Kernel response contents:
|
||||
|
||||
==================================== ====== ==========================
|
||||
``ETHTOOL_A_FEATURES_HEADER`` nested reply header
|
||||
``ETHTOOL_A_FEATURES_HW`` bitset dev->hw_features
|
||||
``ETHTOOL_A_FEATURES_WANTED`` bitset dev->wanted_features
|
||||
``ETHTOOL_A_FEATURES_ACTIVE`` bitset dev->features
|
||||
``ETHTOOL_A_FEATURES_NOCHANGE`` bitset NETIF_F_NEVER_CHANGE
|
||||
==================================== ====== ==========================
|
||||
|
||||
Bitmaps in kernel response have the same meaning as bitmaps used in ioctl
|
||||
interference but attribute names are different (they are based on
|
||||
corresponding members of struct net_device). Legacy "flags" are not provided,
|
||||
if userspace needs them (most likely only ethtool for backward compatibility),
|
||||
it can calculate their values from related feature bits itself.
|
||||
ETHA_FEATURES_HW uses mask consisting of all features recognized by kernel (to
|
||||
provide all names when using verbose bitmap format), the other three use no
|
||||
mask (simple bit lists).
|
||||
|
||||
|
||||
FEATURES_SET
|
||||
============
|
||||
|
||||
Request to set netdev features like ``ETHTOOL_SFEATURES`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
==================================== ====== ==========================
|
||||
``ETHTOOL_A_FEATURES_HEADER`` nested request header
|
||||
``ETHTOOL_A_FEATURES_WANTED`` bitset requested features
|
||||
==================================== ====== ==========================
|
||||
|
||||
Kernel response contents:
|
||||
|
||||
==================================== ====== ==========================
|
||||
``ETHTOOL_A_FEATURES_HEADER`` nested reply header
|
||||
``ETHTOOL_A_FEATURES_WANTED`` bitset diff wanted vs. result
|
||||
``ETHTOOL_A_FEATURES_ACTIVE`` bitset diff old vs. new active
|
||||
==================================== ====== ==========================
|
||||
|
||||
Request constains only one bitset which can be either value/mask pair (request
|
||||
to change specific feature bits and leave the rest) or only a value (request
|
||||
to set all features to specified set).
|
||||
|
||||
As request is subject to netdev_change_features() sanity checks, optional
|
||||
kernel reply (can be suppressed by ``ETHTOOL_FLAG_OMIT_REPLY`` flag in request
|
||||
header) informs client about the actual result. ``ETHTOOL_A_FEATURES_WANTED``
|
||||
reports the difference between client request and actual result: mask consists
|
||||
of bits which differ between requested features and result (dev->features
|
||||
after the operation), value consists of values of these bits in the request
|
||||
(i.e. negated values from resulting features). ``ETHTOOL_A_FEATURES_ACTIVE``
|
||||
reports the difference between old and new dev->features: mask consists of
|
||||
bits which have changed, values are their values in new dev->features (after
|
||||
the operation).
|
||||
|
||||
``ETHTOOL_MSG_FEATURES_NTF`` notification is sent not only if device features
|
||||
are modified using ``ETHTOOL_MSG_FEATURES_SET`` request or on of ethtool ioctl
|
||||
request but also each time features are modified with netdev_update_features()
|
||||
or netdev_change_features().
|
||||
|
||||
|
||||
PRIVFLAGS_GET
|
||||
=============
|
||||
|
||||
Gets private flags like ``ETHTOOL_GPFLAGS`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
==================================== ====== ==========================
|
||||
``ETHTOOL_A_PRIVFLAGS_HEADER`` nested request header
|
||||
==================================== ====== ==========================
|
||||
|
||||
Kernel response contents:
|
||||
|
||||
==================================== ====== ==========================
|
||||
``ETHTOOL_A_PRIVFLAGS_HEADER`` nested reply header
|
||||
``ETHTOOL_A_PRIVFLAGS_FLAGS`` bitset private flags
|
||||
==================================== ====== ==========================
|
||||
|
||||
``ETHTOOL_A_PRIVFLAGS_FLAGS`` is a bitset with values of device private flags.
|
||||
These flags are defined by driver, their number and names (and also meaning)
|
||||
are device dependent. For compact bitset format, names can be retrieved as
|
||||
``ETH_SS_PRIV_FLAGS`` string set. If verbose bitset format is requested,
|
||||
response uses all private flags supported by the device as mask so that client
|
||||
gets the full information without having to fetch the string set with names.
|
||||
|
||||
|
||||
PRIVFLAGS_SET
|
||||
=============
|
||||
|
||||
Sets or modifies values of device private flags like ``ETHTOOL_SPFLAGS``
|
||||
ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
==================================== ====== ==========================
|
||||
``ETHTOOL_A_PRIVFLAGS_HEADER`` nested request header
|
||||
``ETHTOOL_A_PRIVFLAGS_FLAGS`` bitset private flags
|
||||
==================================== ====== ==========================
|
||||
|
||||
``ETHTOOL_A_PRIVFLAGS_FLAGS`` can either set the whole set of private flags or
|
||||
modify only values of some of them.
|
||||
|
||||
|
||||
RINGS_GET
|
||||
=========
|
||||
|
||||
Gets ring sizes like ``ETHTOOL_GRINGPARAM`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
==================================== ====== ==========================
|
||||
``ETHTOOL_A_RINGS_HEADER`` nested request header
|
||||
==================================== ====== ==========================
|
||||
|
||||
Kernel response contents:
|
||||
|
||||
==================================== ====== ==========================
|
||||
``ETHTOOL_A_RINGS_HEADER`` nested reply header
|
||||
``ETHTOOL_A_RINGS_RX_MAX`` u32 max size of RX ring
|
||||
``ETHTOOL_A_RINGS_RX_MINI_MAX`` u32 max size of RX mini ring
|
||||
``ETHTOOL_A_RINGS_RX_JUMBO_MAX`` u32 max size of RX jumbo ring
|
||||
``ETHTOOL_A_RINGS_TX_MAX`` u32 max size of TX ring
|
||||
``ETHTOOL_A_RINGS_RX`` u32 size of RX ring
|
||||
``ETHTOOL_A_RINGS_RX_MINI`` u32 size of RX mini ring
|
||||
``ETHTOOL_A_RINGS_RX_JUMBO`` u32 size of RX jumbo ring
|
||||
``ETHTOOL_A_RINGS_TX`` u32 size of TX ring
|
||||
==================================== ====== ==========================
|
||||
|
||||
|
||||
RINGS_SET
|
||||
=========
|
||||
|
||||
Sets ring sizes like ``ETHTOOL_SRINGPARAM`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
==================================== ====== ==========================
|
||||
``ETHTOOL_A_RINGS_HEADER`` nested reply header
|
||||
``ETHTOOL_A_RINGS_RX`` u32 size of RX ring
|
||||
``ETHTOOL_A_RINGS_RX_MINI`` u32 size of RX mini ring
|
||||
``ETHTOOL_A_RINGS_RX_JUMBO`` u32 size of RX jumbo ring
|
||||
``ETHTOOL_A_RINGS_TX`` u32 size of TX ring
|
||||
==================================== ====== ==========================
|
||||
|
||||
Kernel checks that requested ring sizes do not exceed limits reported by
|
||||
driver. Driver may impose additional constraints and may not suspport all
|
||||
attributes.
|
||||
|
||||
|
||||
CHANNELS_GET
|
||||
============
|
||||
|
||||
Gets channel counts like ``ETHTOOL_GCHANNELS`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
==================================== ====== ==========================
|
||||
``ETHTOOL_A_CHANNELS_HEADER`` nested request header
|
||||
==================================== ====== ==========================
|
||||
|
||||
Kernel response contents:
|
||||
|
||||
===================================== ====== ==========================
|
||||
``ETHTOOL_A_CHANNELS_HEADER`` nested reply header
|
||||
``ETHTOOL_A_CHANNELS_RX_MAX`` u32 max receive channels
|
||||
``ETHTOOL_A_CHANNELS_TX_MAX`` u32 max transmit channels
|
||||
``ETHTOOL_A_CHANNELS_OTHER_MAX`` u32 max other channels
|
||||
``ETHTOOL_A_CHANNELS_COMBINED_MAX`` u32 max combined channels
|
||||
``ETHTOOL_A_CHANNELS_RX_COUNT`` u32 receive channel count
|
||||
``ETHTOOL_A_CHANNELS_TX_COUNT`` u32 transmit channel count
|
||||
``ETHTOOL_A_CHANNELS_OTHER_COUNT`` u32 other channel count
|
||||
``ETHTOOL_A_CHANNELS_COMBINED_COUNT`` u32 combined channel count
|
||||
===================================== ====== ==========================
|
||||
|
||||
|
||||
CHANNELS_SET
|
||||
============
|
||||
|
||||
Sets channel counts like ``ETHTOOL_SCHANNELS`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
===================================== ====== ==========================
|
||||
``ETHTOOL_A_CHANNELS_HEADER`` nested request header
|
||||
``ETHTOOL_A_CHANNELS_RX_COUNT`` u32 receive channel count
|
||||
``ETHTOOL_A_CHANNELS_TX_COUNT`` u32 transmit channel count
|
||||
``ETHTOOL_A_CHANNELS_OTHER_COUNT`` u32 other channel count
|
||||
``ETHTOOL_A_CHANNELS_COMBINED_COUNT`` u32 combined channel count
|
||||
===================================== ====== ==========================
|
||||
|
||||
Kernel checks that requested channel counts do not exceed limits reported by
|
||||
driver. Driver may impose additional constraints and may not suspport all
|
||||
attributes.
|
||||
|
||||
|
||||
COALESCE_GET
|
||||
============
|
||||
|
||||
Gets coalescing parameters like ``ETHTOOL_GCOALESCE`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
==================================== ====== ==========================
|
||||
``ETHTOOL_A_COALESCE_HEADER`` nested request header
|
||||
==================================== ====== ==========================
|
||||
|
||||
Kernel response contents:
|
||||
|
||||
=========================================== ====== =======================
|
||||
``ETHTOOL_A_COALESCE_HEADER`` nested reply header
|
||||
``ETHTOOL_A_COALESCE_RX_USECS`` u32 delay (us), normal Rx
|
||||
``ETHTOOL_A_COALESCE_RX_MAX_FRAMES`` u32 max packets, normal Rx
|
||||
``ETHTOOL_A_COALESCE_RX_USECS_IRQ`` u32 delay (us), Rx in IRQ
|
||||
``ETHTOOL_A_COALESCE_RX_MAX_FRAMES_IRQ`` u32 max packets, Rx in IRQ
|
||||
``ETHTOOL_A_COALESCE_TX_USECS`` u32 delay (us), normal Tx
|
||||
``ETHTOOL_A_COALESCE_TX_MAX_FRAMES`` u32 max packets, normal Tx
|
||||
``ETHTOOL_A_COALESCE_TX_USECS_IRQ`` u32 delay (us), Tx in IRQ
|
||||
``ETHTOOL_A_COALESCE_TX_MAX_FRAMES_IRQ`` u32 IRQ packets, Tx in IRQ
|
||||
``ETHTOOL_A_COALESCE_STATS_BLOCK_USECS`` u32 delay of stats update
|
||||
``ETHTOOL_A_COALESCE_USE_ADAPTIVE_RX`` bool adaptive Rx coalesce
|
||||
``ETHTOOL_A_COALESCE_USE_ADAPTIVE_TX`` bool adaptive Tx coalesce
|
||||
``ETHTOOL_A_COALESCE_PKT_RATE_LOW`` u32 threshold for low rate
|
||||
``ETHTOOL_A_COALESCE_RX_USECS_LOW`` u32 delay (us), low Rx
|
||||
``ETHTOOL_A_COALESCE_RX_MAX_FRAMES_LOW`` u32 max packets, low Rx
|
||||
``ETHTOOL_A_COALESCE_TX_USECS_LOW`` u32 delay (us), low Tx
|
||||
``ETHTOOL_A_COALESCE_TX_MAX_FRAMES_LOW`` u32 max packets, low Tx
|
||||
``ETHTOOL_A_COALESCE_PKT_RATE_HIGH`` u32 threshold for high rate
|
||||
``ETHTOOL_A_COALESCE_RX_USECS_HIGH`` u32 delay (us), high Rx
|
||||
``ETHTOOL_A_COALESCE_RX_MAX_FRAMES_HIGH`` u32 max packets, high Rx
|
||||
``ETHTOOL_A_COALESCE_TX_USECS_HIGH`` u32 delay (us), high Tx
|
||||
``ETHTOOL_A_COALESCE_TX_MAX_FRAMES_HIGH`` u32 max packets, high Tx
|
||||
``ETHTOOL_A_COALESCE_RATE_SAMPLE_INTERVAL`` u32 rate sampling interval
|
||||
=========================================== ====== =======================
|
||||
|
||||
Attributes are only included in reply if their value is not zero or the
|
||||
corresponding bit in ``ethtool_ops::supported_coalesce_params`` is set (i.e.
|
||||
they are declared as supported by driver).
|
||||
|
||||
|
||||
COALESCE_SET
|
||||
============
|
||||
|
||||
Sets coalescing parameters like ``ETHTOOL_SCOALESCE`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
=========================================== ====== =======================
|
||||
``ETHTOOL_A_COALESCE_HEADER`` nested request header
|
||||
``ETHTOOL_A_COALESCE_RX_USECS`` u32 delay (us), normal Rx
|
||||
``ETHTOOL_A_COALESCE_RX_MAX_FRAMES`` u32 max packets, normal Rx
|
||||
``ETHTOOL_A_COALESCE_RX_USECS_IRQ`` u32 delay (us), Rx in IRQ
|
||||
``ETHTOOL_A_COALESCE_RX_MAX_FRAMES_IRQ`` u32 max packets, Rx in IRQ
|
||||
``ETHTOOL_A_COALESCE_TX_USECS`` u32 delay (us), normal Tx
|
||||
``ETHTOOL_A_COALESCE_TX_MAX_FRAMES`` u32 max packets, normal Tx
|
||||
``ETHTOOL_A_COALESCE_TX_USECS_IRQ`` u32 delay (us), Tx in IRQ
|
||||
``ETHTOOL_A_COALESCE_TX_MAX_FRAMES_IRQ`` u32 IRQ packets, Tx in IRQ
|
||||
``ETHTOOL_A_COALESCE_STATS_BLOCK_USECS`` u32 delay of stats update
|
||||
``ETHTOOL_A_COALESCE_USE_ADAPTIVE_RX`` bool adaptive Rx coalesce
|
||||
``ETHTOOL_A_COALESCE_USE_ADAPTIVE_TX`` bool adaptive Tx coalesce
|
||||
``ETHTOOL_A_COALESCE_PKT_RATE_LOW`` u32 threshold for low rate
|
||||
``ETHTOOL_A_COALESCE_RX_USECS_LOW`` u32 delay (us), low Rx
|
||||
``ETHTOOL_A_COALESCE_RX_MAX_FRAMES_LOW`` u32 max packets, low Rx
|
||||
``ETHTOOL_A_COALESCE_TX_USECS_LOW`` u32 delay (us), low Tx
|
||||
``ETHTOOL_A_COALESCE_TX_MAX_FRAMES_LOW`` u32 max packets, low Tx
|
||||
``ETHTOOL_A_COALESCE_PKT_RATE_HIGH`` u32 threshold for high rate
|
||||
``ETHTOOL_A_COALESCE_RX_USECS_HIGH`` u32 delay (us), high Rx
|
||||
``ETHTOOL_A_COALESCE_RX_MAX_FRAMES_HIGH`` u32 max packets, high Rx
|
||||
``ETHTOOL_A_COALESCE_TX_USECS_HIGH`` u32 delay (us), high Tx
|
||||
``ETHTOOL_A_COALESCE_TX_MAX_FRAMES_HIGH`` u32 max packets, high Tx
|
||||
``ETHTOOL_A_COALESCE_RATE_SAMPLE_INTERVAL`` u32 rate sampling interval
|
||||
=========================================== ====== =======================
|
||||
|
||||
Request is rejected if it attributes declared as unsupported by driver (i.e.
|
||||
such that the corresponding bit in ``ethtool_ops::supported_coalesce_params``
|
||||
is not set), regardless of their values. Driver may impose additional
|
||||
constraints on coalescing parameters and their values.
|
||||
|
||||
|
||||
PAUSE_GET
|
||||
============
|
||||
|
||||
Gets channel counts like ``ETHTOOL_GPAUSE`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
===================================== ====== ==========================
|
||||
``ETHTOOL_A_PAUSE_HEADER`` nested request header
|
||||
===================================== ====== ==========================
|
||||
|
||||
Kernel response contents:
|
||||
|
||||
===================================== ====== ==========================
|
||||
``ETHTOOL_A_PAUSE_HEADER`` nested request header
|
||||
``ETHTOOL_A_PAUSE_AUTONEG`` bool pause autonegotiation
|
||||
``ETHTOOL_A_PAUSE_RX`` bool receive pause frames
|
||||
``ETHTOOL_A_PAUSE_TX`` bool transmit pause frames
|
||||
===================================== ====== ==========================
|
||||
|
||||
|
||||
PAUSE_SET
|
||||
============
|
||||
|
||||
Sets pause parameters like ``ETHTOOL_GPAUSEPARAM`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
===================================== ====== ==========================
|
||||
``ETHTOOL_A_PAUSE_HEADER`` nested request header
|
||||
``ETHTOOL_A_PAUSE_AUTONEG`` bool pause autonegotiation
|
||||
``ETHTOOL_A_PAUSE_RX`` bool receive pause frames
|
||||
``ETHTOOL_A_PAUSE_TX`` bool transmit pause frames
|
||||
===================================== ====== ==========================
|
||||
|
||||
|
||||
EEE_GET
|
||||
=======
|
||||
|
||||
Gets channel counts like ``ETHTOOL_GEEE`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
===================================== ====== ==========================
|
||||
``ETHTOOL_A_EEE_HEADER`` nested request header
|
||||
===================================== ====== ==========================
|
||||
|
||||
Kernel response contents:
|
||||
|
||||
===================================== ====== ==========================
|
||||
``ETHTOOL_A_EEE_HEADER`` nested request header
|
||||
``ETHTOOL_A_EEE_MODES_OURS`` bool supported/advertised modes
|
||||
``ETHTOOL_A_EEE_MODES_PEER`` bool peer advertised link modes
|
||||
``ETHTOOL_A_EEE_ACTIVE`` bool EEE is actively used
|
||||
``ETHTOOL_A_EEE_ENABLED`` bool EEE is enabled
|
||||
``ETHTOOL_A_EEE_TX_LPI_ENABLED`` bool Tx lpi enabled
|
||||
``ETHTOOL_A_EEE_TX_LPI_TIMER`` u32 Tx lpi timeout (in us)
|
||||
===================================== ====== ==========================
|
||||
|
||||
In ``ETHTOOL_A_EEE_MODES_OURS``, mask consists of link modes for which EEE is
|
||||
enabled, value of link modes for which EEE is advertised. Link modes for which
|
||||
peer advertises EEE are listed in ``ETHTOOL_A_EEE_MODES_PEER`` (no mask). The
|
||||
netlink interface allows reporting EEE status for all link modes but only
|
||||
first 32 are provided by the ``ethtool_ops`` callback.
|
||||
|
||||
|
||||
EEE_SET
|
||||
=======
|
||||
|
||||
Sets pause parameters like ``ETHTOOL_GEEEPARAM`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
===================================== ====== ==========================
|
||||
``ETHTOOL_A_EEE_HEADER`` nested request header
|
||||
``ETHTOOL_A_EEE_MODES_OURS`` bool advertised modes
|
||||
``ETHTOOL_A_EEE_ENABLED`` bool EEE is enabled
|
||||
``ETHTOOL_A_EEE_TX_LPI_ENABLED`` bool Tx lpi enabled
|
||||
``ETHTOOL_A_EEE_TX_LPI_TIMER`` u32 Tx lpi timeout (in us)
|
||||
===================================== ====== ==========================
|
||||
|
||||
``ETHTOOL_A_EEE_MODES_OURS`` is used to either list link modes to advertise
|
||||
EEE for (if there is no mask) or specify changes to the list (if there is
|
||||
a mask). The netlink interface allows reporting EEE status for all link modes
|
||||
but only first 32 can be set at the moment as that is what the ``ethtool_ops``
|
||||
callback supports.
|
||||
|
||||
|
||||
TSINFO_GET
|
||||
==========
|
||||
|
||||
Gets timestamping information like ``ETHTOOL_GET_TS_INFO`` ioctl request.
|
||||
|
||||
Request contents:
|
||||
|
||||
===================================== ====== ==========================
|
||||
``ETHTOOL_A_TSINFO_HEADER`` nested request header
|
||||
===================================== ====== ==========================
|
||||
|
||||
Kernel response contents:
|
||||
|
||||
===================================== ====== ==========================
|
||||
``ETHTOOL_A_TSINFO_HEADER`` nested request header
|
||||
``ETHTOOL_A_TSINFO_TIMESTAMPING`` bitset SO_TIMESTAMPING flags
|
||||
``ETHTOOL_A_TSINFO_TX_TYPES`` bitset supported Tx types
|
||||
``ETHTOOL_A_TSINFO_RX_FILTERS`` bitset supported Rx filters
|
||||
``ETHTOOL_A_TSINFO_PHC_INDEX`` u32 PTP hw clock index
|
||||
===================================== ====== ==========================
|
||||
|
||||
``ETHTOOL_A_TSINFO_PHC_INDEX`` is absent if there is no associated PHC (there
|
||||
is no special value for this case). The bitset attributes are omitted if they
|
||||
would be empty (no bit set).
|
||||
|
||||
|
||||
Request translation
|
||||
===================
|
||||
|
||||
@ -545,37 +980,37 @@ have their netlink replacement yet.
|
||||
``ETHTOOL_GLINK`` ``ETHTOOL_MSG_LINKSTATE_GET``
|
||||
``ETHTOOL_GEEPROM`` n/a
|
||||
``ETHTOOL_SEEPROM`` n/a
|
||||
``ETHTOOL_GCOALESCE`` n/a
|
||||
``ETHTOOL_SCOALESCE`` n/a
|
||||
``ETHTOOL_GRINGPARAM`` n/a
|
||||
``ETHTOOL_SRINGPARAM`` n/a
|
||||
``ETHTOOL_GPAUSEPARAM`` n/a
|
||||
``ETHTOOL_SPAUSEPARAM`` n/a
|
||||
``ETHTOOL_GRXCSUM`` n/a
|
||||
``ETHTOOL_SRXCSUM`` n/a
|
||||
``ETHTOOL_GTXCSUM`` n/a
|
||||
``ETHTOOL_STXCSUM`` n/a
|
||||
``ETHTOOL_GSG`` n/a
|
||||
``ETHTOOL_SSG`` n/a
|
||||
``ETHTOOL_GCOALESCE`` ``ETHTOOL_MSG_COALESCE_GET``
|
||||
``ETHTOOL_SCOALESCE`` ``ETHTOOL_MSG_COALESCE_SET``
|
||||
``ETHTOOL_GRINGPARAM`` ``ETHTOOL_MSG_RINGS_GET``
|
||||
``ETHTOOL_SRINGPARAM`` ``ETHTOOL_MSG_RINGS_SET``
|
||||
``ETHTOOL_GPAUSEPARAM`` ``ETHTOOL_MSG_PAUSE_GET``
|
||||
``ETHTOOL_SPAUSEPARAM`` ``ETHTOOL_MSG_PAUSE_SET``
|
||||
``ETHTOOL_GRXCSUM`` ``ETHTOOL_MSG_FEATURES_GET``
|
||||
``ETHTOOL_SRXCSUM`` ``ETHTOOL_MSG_FEATURES_SET``
|
||||
``ETHTOOL_GTXCSUM`` ``ETHTOOL_MSG_FEATURES_GET``
|
||||
``ETHTOOL_STXCSUM`` ``ETHTOOL_MSG_FEATURES_SET``
|
||||
``ETHTOOL_GSG`` ``ETHTOOL_MSG_FEATURES_GET``
|
||||
``ETHTOOL_SSG`` ``ETHTOOL_MSG_FEATURES_SET``
|
||||
``ETHTOOL_TEST`` n/a
|
||||
``ETHTOOL_GSTRINGS`` ``ETHTOOL_MSG_STRSET_GET``
|
||||
``ETHTOOL_PHYS_ID`` n/a
|
||||
``ETHTOOL_GSTATS`` n/a
|
||||
``ETHTOOL_GTSO`` n/a
|
||||
``ETHTOOL_STSO`` n/a
|
||||
``ETHTOOL_GTSO`` ``ETHTOOL_MSG_FEATURES_GET``
|
||||
``ETHTOOL_STSO`` ``ETHTOOL_MSG_FEATURES_SET``
|
||||
``ETHTOOL_GPERMADDR`` rtnetlink ``RTM_GETLINK``
|
||||
``ETHTOOL_GUFO`` n/a
|
||||
``ETHTOOL_SUFO`` n/a
|
||||
``ETHTOOL_GGSO`` n/a
|
||||
``ETHTOOL_SGSO`` n/a
|
||||
``ETHTOOL_GFLAGS`` n/a
|
||||
``ETHTOOL_SFLAGS`` n/a
|
||||
``ETHTOOL_GPFLAGS`` n/a
|
||||
``ETHTOOL_SPFLAGS`` n/a
|
||||
``ETHTOOL_GUFO`` ``ETHTOOL_MSG_FEATURES_GET``
|
||||
``ETHTOOL_SUFO`` ``ETHTOOL_MSG_FEATURES_SET``
|
||||
``ETHTOOL_GGSO`` ``ETHTOOL_MSG_FEATURES_GET``
|
||||
``ETHTOOL_SGSO`` ``ETHTOOL_MSG_FEATURES_SET``
|
||||
``ETHTOOL_GFLAGS`` ``ETHTOOL_MSG_FEATURES_GET``
|
||||
``ETHTOOL_SFLAGS`` ``ETHTOOL_MSG_FEATURES_SET``
|
||||
``ETHTOOL_GPFLAGS`` ``ETHTOOL_MSG_PRIVFLAGS_GET``
|
||||
``ETHTOOL_SPFLAGS`` ``ETHTOOL_MSG_PRIVFLAGS_SET``
|
||||
``ETHTOOL_GRXFH`` n/a
|
||||
``ETHTOOL_SRXFH`` n/a
|
||||
``ETHTOOL_GGRO`` n/a
|
||||
``ETHTOOL_SGRO`` n/a
|
||||
``ETHTOOL_GGRO`` ``ETHTOOL_MSG_FEATURES_GET``
|
||||
``ETHTOOL_SGRO`` ``ETHTOOL_MSG_FEATURES_SET``
|
||||
``ETHTOOL_GRXRINGS`` n/a
|
||||
``ETHTOOL_GRXCLSRLCNT`` n/a
|
||||
``ETHTOOL_GRXCLSRULE`` n/a
|
||||
@ -589,18 +1024,18 @@ have their netlink replacement yet.
|
||||
``ETHTOOL_GSSET_INFO`` ``ETHTOOL_MSG_STRSET_GET``
|
||||
``ETHTOOL_GRXFHINDIR`` n/a
|
||||
``ETHTOOL_SRXFHINDIR`` n/a
|
||||
``ETHTOOL_GFEATURES`` n/a
|
||||
``ETHTOOL_SFEATURES`` n/a
|
||||
``ETHTOOL_GCHANNELS`` n/a
|
||||
``ETHTOOL_SCHANNELS`` n/a
|
||||
``ETHTOOL_GFEATURES`` ``ETHTOOL_MSG_FEATURES_GET``
|
||||
``ETHTOOL_SFEATURES`` ``ETHTOOL_MSG_FEATURES_SET``
|
||||
``ETHTOOL_GCHANNELS`` ``ETHTOOL_MSG_CHANNELS_GET``
|
||||
``ETHTOOL_SCHANNELS`` ``ETHTOOL_MSG_CHANNELS_SET``
|
||||
``ETHTOOL_SET_DUMP`` n/a
|
||||
``ETHTOOL_GET_DUMP_FLAG`` n/a
|
||||
``ETHTOOL_GET_DUMP_DATA`` n/a
|
||||
``ETHTOOL_GET_TS_INFO`` n/a
|
||||
``ETHTOOL_GET_TS_INFO`` ``ETHTOOL_MSG_TSINFO_GET``
|
||||
``ETHTOOL_GMODULEINFO`` n/a
|
||||
``ETHTOOL_GMODULEEEPROM`` n/a
|
||||
``ETHTOOL_GEEE`` n/a
|
||||
``ETHTOOL_SEEE`` n/a
|
||||
``ETHTOOL_GEEE`` ``ETHTOOL_MSG_EEE_GET``
|
||||
``ETHTOOL_SEEE`` ``ETHTOOL_MSG_EEE_SET``
|
||||
``ETHTOOL_GRSSH`` n/a
|
||||
``ETHTOOL_SRSSH`` n/a
|
||||
``ETHTOOL_GTUNABLE`` n/a
|
||||
|
@ -606,7 +606,7 @@ before a conversion to the new layout is being done behind the scenes!
|
||||
|
||||
Currently, the classic BPF format is being used for JITing on most
|
||||
32-bit architectures, whereas x86-64, aarch64, s390x, powerpc64,
|
||||
sparc64, arm32, riscv (RV64G) perform JIT compilation from eBPF
|
||||
sparc64, arm32, riscv64, riscv32 perform JIT compilation from eBPF
|
||||
instruction set.
|
||||
|
||||
Some core changes of the new internal format:
|
||||
|
@ -8,6 +8,7 @@ Contents:
|
||||
|
||||
netdev-FAQ
|
||||
af_xdp
|
||||
bareudp
|
||||
batman-adv
|
||||
can
|
||||
can_ucan_protocol
|
||||
@ -33,6 +34,7 @@ Contents:
|
||||
tls
|
||||
tls-offload
|
||||
nfc
|
||||
6lowpan
|
||||
|
||||
.. only:: subproject and html
|
||||
|
||||
|
@ -958,6 +958,15 @@ ip_nonlocal_bind - BOOLEAN
|
||||
which can be quite useful - but may break some applications.
|
||||
Default: 0
|
||||
|
||||
ip_autobind_reuse - BOOLEAN
|
||||
By default, bind() does not select the ports automatically even if
|
||||
the new socket and all sockets bound to the port have SO_REUSEADDR.
|
||||
ip_autobind_reuse allows bind() to reuse the port and this is useful
|
||||
when you use bind()+connect(), but may break some applications.
|
||||
The preferred solution is to use IP_BIND_ADDRESS_NO_PORT and this
|
||||
option should only be set by experts.
|
||||
Default: 0
|
||||
|
||||
ip_dynaddr - BOOLEAN
|
||||
If set non-zero, enables support for dynamic addresses.
|
||||
If set to a non-zero value larger than 1, a kernel log
|
||||
|
159
Documentation/networking/page_pool.rst
Normal file
159
Documentation/networking/page_pool.rst
Normal file
@ -0,0 +1,159 @@
|
||||
.. SPDX-License-Identifier: GPL-2.0
|
||||
|
||||
=============
|
||||
Page Pool API
|
||||
=============
|
||||
|
||||
The page_pool allocator is optimized for the XDP mode that uses one frame
|
||||
per-page, but it can fallback on the regular page allocator APIs.
|
||||
|
||||
Basic use involves replacing alloc_pages() calls with the
|
||||
page_pool_alloc_pages() call. Drivers should use page_pool_dev_alloc_pages()
|
||||
replacing dev_alloc_pages().
|
||||
|
||||
API keeps track of inflight pages, in order to let API user know
|
||||
when it is safe to free a page_pool object. Thus, API users
|
||||
must run page_pool_release_page() when a page is leaving the page_pool or
|
||||
call page_pool_put_page() where appropriate in order to maintain correct
|
||||
accounting.
|
||||
|
||||
API user must call page_pool_put_page() once on a page, as it
|
||||
will either recycle the page, or in case of refcnt > 1, it will
|
||||
release the DMA mapping and inflight state accounting.
|
||||
|
||||
Architecture overview
|
||||
=====================
|
||||
|
||||
.. code-block:: none
|
||||
|
||||
+------------------+
|
||||
| Driver |
|
||||
+------------------+
|
||||
^
|
||||
|
|
||||
|
|
||||
|
|
||||
v
|
||||
+--------------------------------------------+
|
||||
| request memory |
|
||||
+--------------------------------------------+
|
||||
^ ^
|
||||
| |
|
||||
| Pool empty | Pool has entries
|
||||
| |
|
||||
v v
|
||||
+-----------------------+ +------------------------+
|
||||
| alloc (and map) pages | | get page from cache |
|
||||
+-----------------------+ +------------------------+
|
||||
^ ^
|
||||
| |
|
||||
| cache available | No entries, refill
|
||||
| | from ptr-ring
|
||||
| |
|
||||
v v
|
||||
+-----------------+ +------------------+
|
||||
| Fast cache | | ptr-ring cache |
|
||||
+-----------------+ +------------------+
|
||||
|
||||
API interface
|
||||
=============
|
||||
The number of pools created **must** match the number of hardware queues
|
||||
unless hardware restrictions make that impossible. This would otherwise beat the
|
||||
purpose of page pool, which is allocate pages fast from cache without locking.
|
||||
This lockless guarantee naturally comes from running under a NAPI softirq.
|
||||
The protection doesn't strictly have to be NAPI, any guarantee that allocating
|
||||
a page will cause no race conditions is enough.
|
||||
|
||||
* page_pool_create(): Create a pool.
|
||||
* flags: PP_FLAG_DMA_MAP, PP_FLAG_DMA_SYNC_DEV
|
||||
* order: 2^order pages on allocation
|
||||
* pool_size: size of the ptr_ring
|
||||
* nid: preferred NUMA node for allocation
|
||||
* dev: struct device. Used on DMA operations
|
||||
* dma_dir: DMA direction
|
||||
* max_len: max DMA sync memory size
|
||||
* offset: DMA address offset
|
||||
|
||||
* page_pool_put_page(): The outcome of this depends on the page refcnt. If the
|
||||
driver bumps the refcnt > 1 this will unmap the page. If the page refcnt is 1
|
||||
the allocator owns the page and will try to recycle it in one of the pool
|
||||
caches. If PP_FLAG_DMA_SYNC_DEV is set, the page will be synced for_device
|
||||
using dma_sync_single_range_for_device().
|
||||
|
||||
* page_pool_put_full_page(): Similar to page_pool_put_page(), but will DMA sync
|
||||
for the entire memory area configured in area pool->max_len.
|
||||
|
||||
* page_pool_recycle_direct(): Similar to page_pool_put_full_page() but caller
|
||||
must guarantee safe context (e.g NAPI), since it will recycle the page
|
||||
directly into the pool fast cache.
|
||||
|
||||
* page_pool_release_page(): Unmap the page (if mapped) and account for it on
|
||||
inflight counters.
|
||||
|
||||
* page_pool_dev_alloc_pages(): Get a page from the page allocator or page_pool
|
||||
caches.
|
||||
|
||||
* page_pool_get_dma_addr(): Retrieve the stored DMA address.
|
||||
|
||||
* page_pool_get_dma_dir(): Retrieve the stored DMA direction.
|
||||
|
||||
Coding examples
|
||||
===============
|
||||
|
||||
Registration
|
||||
------------
|
||||
|
||||
.. code-block:: c
|
||||
|
||||
/* Page pool registration */
|
||||
struct page_pool_params pp_params = { 0 };
|
||||
struct xdp_rxq_info xdp_rxq;
|
||||
int err;
|
||||
|
||||
pp_params.order = 0;
|
||||
/* internal DMA mapping in page_pool */
|
||||
pp_params.flags = PP_FLAG_DMA_MAP;
|
||||
pp_params.pool_size = DESC_NUM;
|
||||
pp_params.nid = NUMA_NO_NODE;
|
||||
pp_params.dev = priv->dev;
|
||||
pp_params.dma_dir = xdp_prog ? DMA_BIDIRECTIONAL : DMA_FROM_DEVICE;
|
||||
page_pool = page_pool_create(&pp_params);
|
||||
|
||||
err = xdp_rxq_info_reg(&xdp_rxq, ndev, 0);
|
||||
if (err)
|
||||
goto err_out;
|
||||
|
||||
err = xdp_rxq_info_reg_mem_model(&xdp_rxq, MEM_TYPE_PAGE_POOL, page_pool);
|
||||
if (err)
|
||||
goto err_out;
|
||||
|
||||
NAPI poller
|
||||
-----------
|
||||
|
||||
|
||||
.. code-block:: c
|
||||
|
||||
/* NAPI Rx poller */
|
||||
enum dma_data_direction dma_dir;
|
||||
|
||||
dma_dir = page_pool_get_dma_dir(dring->page_pool);
|
||||
while (done < budget) {
|
||||
if (some error)
|
||||
page_pool_recycle_direct(page_pool, page);
|
||||
if (packet_is_xdp) {
|
||||
if XDP_DROP:
|
||||
page_pool_recycle_direct(page_pool, page);
|
||||
} else (packet_is_skb) {
|
||||
page_pool_release_page(page_pool, page);
|
||||
new_page = page_pool_dev_alloc_pages(page_pool);
|
||||
}
|
||||
}
|
||||
|
||||
Driver unload
|
||||
-------------
|
||||
|
||||
.. code-block:: c
|
||||
|
||||
/* Driver unload */
|
||||
page_pool_put_full_page(page_pool, page, false);
|
||||
xdp_rxq_info_unreg(&xdp_rxq);
|
@ -74,10 +74,13 @@ phylib to the sfp/phylink support. Please send patches to improve
|
||||
this documentation.
|
||||
|
||||
1. Optionally split the network driver's phylib update function into
|
||||
three parts dealing with link-down, link-up and reconfiguring the
|
||||
MAC settings. This can be done as a separate preparation commit.
|
||||
two parts dealing with link-down and link-up. This can be done as
|
||||
a separate preparation commit.
|
||||
|
||||
An example of this preparation can be found in git commit fc548b991fb0.
|
||||
An older example of this preparation can be found in git commit
|
||||
fc548b991fb0, although this was splitting into three parts; the
|
||||
link-up part now includes configuring the MAC for the link settings.
|
||||
Please see :c:func:`mac_link_up` for more information on this.
|
||||
|
||||
2. Replace::
|
||||
|
||||
@ -135,27 +138,27 @@ this documentation.
|
||||
|
||||
.. code-block:: c
|
||||
|
||||
static int foo_ethtool_set_link_ksettings(struct net_device *dev,
|
||||
const struct ethtool_link_ksettings *cmd)
|
||||
{
|
||||
struct foo_priv *priv = netdev_priv(dev);
|
||||
static int foo_ethtool_set_link_ksettings(struct net_device *dev,
|
||||
const struct ethtool_link_ksettings *cmd)
|
||||
{
|
||||
struct foo_priv *priv = netdev_priv(dev);
|
||||
|
||||
return phylink_ethtool_ksettings_set(priv->phylink, cmd);
|
||||
}
|
||||
|
||||
return phylink_ethtool_ksettings_set(priv->phylink, cmd);
|
||||
}
|
||||
static int foo_ethtool_get_link_ksettings(struct net_device *dev,
|
||||
struct ethtool_link_ksettings *cmd)
|
||||
{
|
||||
struct foo_priv *priv = netdev_priv(dev);
|
||||
|
||||
return phylink_ethtool_ksettings_get(priv->phylink, cmd);
|
||||
}
|
||||
|
||||
static int foo_ethtool_get_link_ksettings(struct net_device *dev,
|
||||
struct ethtool_link_ksettings *cmd)
|
||||
{
|
||||
struct foo_priv *priv = netdev_priv(dev);
|
||||
|
||||
return phylink_ethtool_ksettings_get(priv->phylink, cmd);
|
||||
}
|
||||
|
||||
7. Replace the call to:
|
||||
7. Replace the call to::
|
||||
|
||||
phy_dev = of_phy_connect(dev, node, link_func, flags, phy_interface);
|
||||
|
||||
and associated code with a call to:
|
||||
and associated code with a call to::
|
||||
|
||||
err = phylink_of_phy_connect(priv->phylink, node, flags);
|
||||
|
||||
@ -207,6 +210,14 @@ this documentation.
|
||||
using. This is particularly important for in-band negotiation
|
||||
methods such as 1000base-X and SGMII.
|
||||
|
||||
The :c:func:`mac_link_up` method is used to inform the MAC that the
|
||||
link has come up. The call includes the negotiation mode and interface
|
||||
for reference only. The finalised link parameters are also supplied
|
||||
(speed, duplex and flow control/pause enablement settings) which
|
||||
should be used to configure the MAC when the MAC and PCS are not
|
||||
tightly integrated, or when the settings are not coming from in-band
|
||||
negotiation.
|
||||
|
||||
The :c:func:`mac_config` method is used to update the MAC with the
|
||||
requested state, and must avoid unnecessarily taking the link down
|
||||
when making changes to the MAC configuration. This means the
|
||||
|
50
MAINTAINERS
50
MAINTAINERS
@ -176,7 +176,7 @@ L: linux-wpan@vger.kernel.org
|
||||
S: Maintained
|
||||
F: net/6lowpan/
|
||||
F: include/net/6lowpan.h
|
||||
F: Documentation/networking/6lowpan.txt
|
||||
F: Documentation/networking/6lowpan.rst
|
||||
|
||||
6PACK NETWORK DRIVER FOR AX.25
|
||||
M: Andreas Koensgen <ajk@comnets.uni-bremen.de>
|
||||
@ -3173,6 +3173,8 @@ R: Martin KaFai Lau <kafai@fb.com>
|
||||
R: Song Liu <songliubraving@fb.com>
|
||||
R: Yonghong Song <yhs@fb.com>
|
||||
R: Andrii Nakryiko <andriin@fb.com>
|
||||
R: John Fastabend <john.fastabend@gmail.com>
|
||||
R: KP Singh <kpsingh@chromium.org>
|
||||
L: netdev@vger.kernel.org
|
||||
L: bpf@vger.kernel.org
|
||||
T: git git://git.kernel.org/pub/scm/linux/kernel/git/bpf/bpf.git
|
||||
@ -3239,11 +3241,22 @@ L: bpf@vger.kernel.org
|
||||
S: Maintained
|
||||
F: arch/powerpc/net/
|
||||
|
||||
BPF JIT for RISC-V (RV64G)
|
||||
M: Björn Töpel <bjorn.topel@gmail.com>
|
||||
BPF JIT for RISC-V (32-bit)
|
||||
M: Luke Nelson <luke.r.nels@gmail.com>
|
||||
M: Xi Wang <xi.wang@gmail.com>
|
||||
L: netdev@vger.kernel.org
|
||||
L: bpf@vger.kernel.org
|
||||
S: Maintained
|
||||
F: arch/riscv/net/
|
||||
X: arch/riscv/net/bpf_jit_comp64.c
|
||||
|
||||
BPF JIT for RISC-V (64-bit)
|
||||
M: Björn Töpel <bjorn.topel@gmail.com>
|
||||
L: netdev@vger.kernel.org
|
||||
L: bpf@vger.kernel.org
|
||||
S: Maintained
|
||||
F: arch/riscv/net/
|
||||
X: arch/riscv/net/bpf_jit_comp32.c
|
||||
|
||||
BPF JIT for S390
|
||||
M: Ilya Leoshkevich <iii@linux.ibm.com>
|
||||
@ -9385,6 +9398,8 @@ F: include/net/l3mdev.h
|
||||
L7 BPF FRAMEWORK
|
||||
M: John Fastabend <john.fastabend@gmail.com>
|
||||
M: Daniel Borkmann <daniel@iogearbox.net>
|
||||
M: Jakub Sitnicki <jakub@cloudflare.com>
|
||||
M: Lorenz Bauer <lmb@cloudflare.com>
|
||||
L: netdev@vger.kernel.org
|
||||
L: bpf@vger.kernel.org
|
||||
S: Maintained
|
||||
@ -9392,6 +9407,7 @@ F: include/linux/skmsg.h
|
||||
F: net/core/skmsg.c
|
||||
F: net/core/sock_map.c
|
||||
F: net/ipv4/tcp_bpf.c
|
||||
F: net/ipv4/udp_bpf.c
|
||||
|
||||
LANTIQ / INTEL Ethernet drivers
|
||||
M: Hauke Mehrtens <hauke@hauke-m.de>
|
||||
@ -10122,6 +10138,13 @@ M: Nicolas Pitre <nico@fluxnic.net>
|
||||
S: Odd Fixes
|
||||
F: drivers/mmc/host/mvsdio.*
|
||||
|
||||
MARVELL USB MDIO CONTROLLER DRIVER
|
||||
M: Tobias Waldekranz <tobias@waldekranz.com>
|
||||
L: netdev@vger.kernel.org
|
||||
S: Maintained
|
||||
F: drivers/net/phy/mdio-mvusb.c
|
||||
F: Documentation/devicetree/bindings/net/marvell,mvusb.yaml
|
||||
|
||||
MARVELL XENON MMC/SD/SDIO HOST CONTROLLER DRIVER
|
||||
M: Hu Ziji <huziji@marvell.com>
|
||||
L: linux-mmc@vger.kernel.org
|
||||
@ -11738,6 +11761,7 @@ W: https://github.com/multipath-tcp/mptcp_net-next/wiki
|
||||
B: https://github.com/multipath-tcp/mptcp_net-next/issues
|
||||
S: Maintained
|
||||
F: include/net/mptcp.h
|
||||
F: include/uapi/linux/mptcp.h
|
||||
F: net/mptcp/
|
||||
F: tools/testing/selftests/net/mptcp/
|
||||
|
||||
@ -13706,6 +13730,12 @@ L: alsa-devel@alsa-project.org (moderated for non-subscribers)
|
||||
S: Supported
|
||||
F: sound/soc/qcom/
|
||||
|
||||
QCOM IPA DRIVER
|
||||
M: Alex Elder <elder@kernel.org>
|
||||
L: netdev@vger.kernel.org
|
||||
S: Supported
|
||||
F: drivers/net/ipa/
|
||||
|
||||
QEMU MACHINE EMULATOR AND VIRTUALIZER SUPPORT
|
||||
M: Gabriel Somlo <somlo@cmu.edu>
|
||||
M: "Michael S. Tsirkin" <mst@redhat.com>
|
||||
@ -16175,6 +16205,13 @@ L: netdev@vger.kernel.org
|
||||
S: Supported
|
||||
F: drivers/net/ethernet/synopsys/
|
||||
|
||||
SYNOPSYS DESIGNWARE ETHERNET XPCS DRIVER
|
||||
M: Jose Abreu <Jose.Abreu@synopsys.com>
|
||||
L: netdev@vger.kernel.org
|
||||
S: Supported
|
||||
F: drivers/net/phy/mdio-xpcs.c
|
||||
F: include/linux/mdio-xpcs.h
|
||||
|
||||
SYNOPSYS DESIGNWARE I2C DRIVER
|
||||
M: Jarkko Nikula <jarkko.nikula@linux.intel.com>
|
||||
R: Andy Shevchenko <andriy.shevchenko@linux.intel.com>
|
||||
@ -17917,6 +17954,13 @@ S: Supported
|
||||
F: arch/x86/kernel/cpu/vmware.c
|
||||
F: arch/x86/include/asm/vmware.h
|
||||
|
||||
VMWARE VIRTUAL PTP CLOCK DRIVER
|
||||
M: Vivek Thampi <vithampi@vmware.com>
|
||||
M: "VMware, Inc." <pv-drivers@vmware.com>
|
||||
L: netdev@vger.kernel.org
|
||||
S: Supported
|
||||
F: drivers/ptp/ptp_vmw.c
|
||||
|
||||
VMWARE PVRDMA DRIVER
|
||||
M: Adit Ranadive <aditr@vmware.com>
|
||||
M: VMware PV-Drivers <pv-drivers@vmware.com>
|
||||
|
@ -180,7 +180,7 @@
|
||||
&fec {
|
||||
pinctrl-names = "default";
|
||||
pinctrl-0 = <&pinctrl_enet>;
|
||||
phy-mode = "rgmii";
|
||||
phy-mode = "rgmii-id";
|
||||
phy-handle = <ðphy>;
|
||||
phy-reset-duration = <10>;
|
||||
phy-reset-gpios = <&gpio1 25 GPIO_ACTIVE_LOW>;
|
||||
|
@ -187,4 +187,53 @@
|
||||
#size-cells = <0>;
|
||||
};
|
||||
};
|
||||
|
||||
mcu_cpsw: ethernet@46000000 {
|
||||
compatible = "ti,am654-cpsw-nuss";
|
||||
#address-cells = <2>;
|
||||
#size-cells = <2>;
|
||||
reg = <0x0 0x46000000 0x0 0x200000>;
|
||||
reg-names = "cpsw_nuss";
|
||||
ranges = <0x0 0x0 0x0 0x46000000 0x0 0x200000>;
|
||||
dma-coherent;
|
||||
clocks = <&k3_clks 5 10>;
|
||||
clock-names = "fck";
|
||||
power-domains = <&k3_pds 5 TI_SCI_PD_EXCLUSIVE>;
|
||||
|
||||
dmas = <&mcu_udmap 0xf000>,
|
||||
<&mcu_udmap 0xf001>,
|
||||
<&mcu_udmap 0xf002>,
|
||||
<&mcu_udmap 0xf003>,
|
||||
<&mcu_udmap 0xf004>,
|
||||
<&mcu_udmap 0xf005>,
|
||||
<&mcu_udmap 0xf006>,
|
||||
<&mcu_udmap 0xf007>,
|
||||
<&mcu_udmap 0x7000>;
|
||||
dma-names = "tx0", "tx1", "tx2", "tx3",
|
||||
"tx4", "tx5", "tx6", "tx7",
|
||||
"rx";
|
||||
|
||||
ethernet-ports {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
cpsw_port1: port@1 {
|
||||
reg = <1>;
|
||||
ti,mac-only;
|
||||
label = "port1";
|
||||
ti,syscon-efuse = <&mcu_conf 0x200>;
|
||||
phys = <&phy_gmii_sel 1>;
|
||||
};
|
||||
};
|
||||
|
||||
davinci_mdio: mdio@f00 {
|
||||
compatible = "ti,cpsw-mdio","ti,davinci_mdio";
|
||||
reg = <0x0 0xf00 0x0 0x100>;
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
clocks = <&k3_clks 5 10>;
|
||||
clock-names = "fck";
|
||||
bus_freq = <1000000>;
|
||||
};
|
||||
};
|
||||
};
|
||||
|
@ -30,6 +30,7 @@
|
||||
i2c3 = &main_i2c1;
|
||||
i2c4 = &main_i2c2;
|
||||
i2c5 = &main_i2c3;
|
||||
ethernet0 = &cpsw_port1;
|
||||
};
|
||||
|
||||
chosen { };
|
||||
|
@ -7,6 +7,7 @@
|
||||
|
||||
#include "k3-am654.dtsi"
|
||||
#include <dt-bindings/input/input.h>
|
||||
#include <dt-bindings/net/ti-dp83867.h>
|
||||
|
||||
/ {
|
||||
compatible = "ti,am654-evm", "ti,am654";
|
||||
@ -95,7 +96,30 @@
|
||||
wkup_pca554_default: wkup_pca554_default {
|
||||
pinctrl-single,pins = <
|
||||
AM65X_WKUP_IOPAD(0x0034, PIN_INPUT, 7) /* (T1) MCU_OSPI1_CLK.WKUP_GPIO0_25 */
|
||||
>;
|
||||
};
|
||||
|
||||
mcu_cpsw_pins_default: mcu_cpsw_pins_default {
|
||||
pinctrl-single,pins = <
|
||||
AM65X_WKUP_IOPAD(0x0058, PIN_OUTPUT, 0) /* (N4) MCU_RGMII1_TX_CTL */
|
||||
AM65X_WKUP_IOPAD(0x005c, PIN_INPUT, 0) /* (N5) MCU_RGMII1_RX_CTL */
|
||||
AM65X_WKUP_IOPAD(0x0060, PIN_OUTPUT, 0) /* (M2) MCU_RGMII1_TD3 */
|
||||
AM65X_WKUP_IOPAD(0x0064, PIN_OUTPUT, 0) /* (M3) MCU_RGMII1_TD2 */
|
||||
AM65X_WKUP_IOPAD(0x0068, PIN_OUTPUT, 0) /* (M4) MCU_RGMII1_TD1 */
|
||||
AM65X_WKUP_IOPAD(0x006c, PIN_OUTPUT, 0) /* (M5) MCU_RGMII1_TD0 */
|
||||
AM65X_WKUP_IOPAD(0x0078, PIN_INPUT, 0) /* (L2) MCU_RGMII1_RD3 */
|
||||
AM65X_WKUP_IOPAD(0x007c, PIN_INPUT, 0) /* (L5) MCU_RGMII1_RD2 */
|
||||
AM65X_WKUP_IOPAD(0x0080, PIN_INPUT, 0) /* (M6) MCU_RGMII1_RD1 */
|
||||
AM65X_WKUP_IOPAD(0x0084, PIN_INPUT, 0) /* (L6) MCU_RGMII1_RD0 */
|
||||
AM65X_WKUP_IOPAD(0x0070, PIN_INPUT, 0) /* (N1) MCU_RGMII1_TXC */
|
||||
AM65X_WKUP_IOPAD(0x0074, PIN_INPUT, 0) /* (M1) MCU_RGMII1_RXC */
|
||||
>;
|
||||
};
|
||||
|
||||
mcu_mdio_pins_default: mcu_mdio1_pins_default {
|
||||
pinctrl-single,pins = <
|
||||
AM65X_WKUP_IOPAD(0x008c, PIN_OUTPUT, 0) /* (L1) MCU_MDIO0_MDC */
|
||||
AM65X_WKUP_IOPAD(0x0088, PIN_INPUT, 0) /* (L4) MCU_MDIO0_MDIO */
|
||||
>;
|
||||
};
|
||||
};
|
||||
@ -419,3 +443,21 @@
|
||||
data-lanes = <1 2>;
|
||||
};
|
||||
};
|
||||
|
||||
&mcu_cpsw {
|
||||
pinctrl-names = "default";
|
||||
pinctrl-0 = <&mcu_cpsw_pins_default &mcu_mdio_pins_default>;
|
||||
};
|
||||
|
||||
&davinci_mdio {
|
||||
phy0: ethernet-phy@0 {
|
||||
reg = <0>;
|
||||
ti,rx-internal-delay = <DP83867_RGMIIDCTL_2_00_NS>;
|
||||
ti,fifo-depth = <DP83867_PHYCR_FIFO_DEPTH_4_B_NIB>;
|
||||
};
|
||||
};
|
||||
|
||||
&cpsw_port1 {
|
||||
phy-mode = "rgmii-rxid";
|
||||
phy-handle = <&phy0>;
|
||||
};
|
||||
|
@ -8,6 +8,7 @@
|
||||
#include "k3-j721e-som-p0.dtsi"
|
||||
#include <dt-bindings/gpio/gpio.h>
|
||||
#include <dt-bindings/input/input.h>
|
||||
#include <dt-bindings/net/ti-dp83867.h>
|
||||
|
||||
/ {
|
||||
chosen {
|
||||
@ -128,6 +129,30 @@
|
||||
J721E_WKUP_IOPAD(0x38, PIN_INPUT, 0) /* (A23) MCU_OSPI1_LBCLKO */
|
||||
>;
|
||||
};
|
||||
|
||||
mcu_cpsw_pins_default: mcu_cpsw_pins_default {
|
||||
pinctrl-single,pins = <
|
||||
J721E_WKUP_IOPAD(0x0058, PIN_OUTPUT, 0) /* MCU_RGMII1_TX_CTL */
|
||||
J721E_WKUP_IOPAD(0x005c, PIN_INPUT, 0) /* MCU_RGMII1_RX_CTL */
|
||||
J721E_WKUP_IOPAD(0x0060, PIN_OUTPUT, 0) /* MCU_RGMII1_TD3 */
|
||||
J721E_WKUP_IOPAD(0x0064, PIN_OUTPUT, 0) /* MCU_RGMII1_TD2 */
|
||||
J721E_WKUP_IOPAD(0x0068, PIN_OUTPUT, 0) /* MCU_RGMII1_TD1 */
|
||||
J721E_WKUP_IOPAD(0x006c, PIN_OUTPUT, 0) /* MCU_RGMII1_TD0 */
|
||||
J721E_WKUP_IOPAD(0x0078, PIN_INPUT, 0) /* MCU_RGMII1_RD3 */
|
||||
J721E_WKUP_IOPAD(0x007c, PIN_INPUT, 0) /* MCU_RGMII1_RD2 */
|
||||
J721E_WKUP_IOPAD(0x0080, PIN_INPUT, 0) /* MCU_RGMII1_RD1 */
|
||||
J721E_WKUP_IOPAD(0x0084, PIN_INPUT, 0) /* MCU_RGMII1_RD0 */
|
||||
J721E_WKUP_IOPAD(0x0070, PIN_INPUT, 0) /* MCU_RGMII1_TXC */
|
||||
J721E_WKUP_IOPAD(0x0074, PIN_INPUT, 0) /* MCU_RGMII1_RXC */
|
||||
>;
|
||||
};
|
||||
|
||||
mcu_mdio_pins_default: mcu_mdio1_pins_default {
|
||||
pinctrl-single,pins = <
|
||||
J721E_WKUP_IOPAD(0x008c, PIN_OUTPUT, 0) /* MCU_MDIO0_MDC */
|
||||
J721E_WKUP_IOPAD(0x0088, PIN_INPUT, 0) /* MCU_MDIO0_MDIO */
|
||||
>;
|
||||
};
|
||||
};
|
||||
|
||||
&wkup_uart0 {
|
||||
@ -429,3 +454,21 @@
|
||||
#gpio-cells = <2>;
|
||||
};
|
||||
};
|
||||
|
||||
&mcu_cpsw {
|
||||
pinctrl-names = "default";
|
||||
pinctrl-0 = <&mcu_cpsw_pins_default &mcu_mdio_pins_default>;
|
||||
};
|
||||
|
||||
&davinci_mdio {
|
||||
phy0: ethernet-phy@0 {
|
||||
reg = <0>;
|
||||
ti,rx-internal-delay = <DP83867_RGMIIDCTL_2_00_NS>;
|
||||
ti,fifo-depth = <DP83867_PHYCR_FIFO_DEPTH_4_B_NIB>;
|
||||
};
|
||||
};
|
||||
|
||||
&cpsw_port1 {
|
||||
phy-mode = "rgmii-rxid";
|
||||
phy-handle = <&phy0>;
|
||||
};
|
||||
|
@ -270,4 +270,53 @@
|
||||
ti,sci-rm-range-rflow = <0x00>; /* GP RFLOW */
|
||||
};
|
||||
};
|
||||
|
||||
mcu_cpsw: ethernet@46000000 {
|
||||
compatible = "ti,j721e-cpsw-nuss";
|
||||
#address-cells = <2>;
|
||||
#size-cells = <2>;
|
||||
reg = <0x0 0x46000000 0x0 0x200000>;
|
||||
reg-names = "cpsw_nuss";
|
||||
ranges = <0x0 0x0 0x0 0x46000000 0x0 0x200000>;
|
||||
dma-coherent;
|
||||
clocks = <&k3_clks 18 22>;
|
||||
clock-names = "fck";
|
||||
power-domains = <&k3_pds 18 TI_SCI_PD_EXCLUSIVE>;
|
||||
|
||||
dmas = <&mcu_udmap 0xf000>,
|
||||
<&mcu_udmap 0xf001>,
|
||||
<&mcu_udmap 0xf002>,
|
||||
<&mcu_udmap 0xf003>,
|
||||
<&mcu_udmap 0xf004>,
|
||||
<&mcu_udmap 0xf005>,
|
||||
<&mcu_udmap 0xf006>,
|
||||
<&mcu_udmap 0xf007>,
|
||||
<&mcu_udmap 0x7000>;
|
||||
dma-names = "tx0", "tx1", "tx2", "tx3",
|
||||
"tx4", "tx5", "tx6", "tx7",
|
||||
"rx";
|
||||
|
||||
ethernet-ports {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
cpsw_port1: port@1 {
|
||||
reg = <1>;
|
||||
ti,mac-only;
|
||||
label = "port1";
|
||||
ti,syscon-efuse = <&mcu_conf 0x200>;
|
||||
phys = <&phy_gmii_sel 1>;
|
||||
};
|
||||
};
|
||||
|
||||
davinci_mdio: mdio@f00 {
|
||||
compatible = "ti,cpsw-mdio","ti,davinci_mdio";
|
||||
reg = <0x0 0xf00 0x0 0x100>;
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
clocks = <&k3_clks 18 22>;
|
||||
clock-names = "fck";
|
||||
bus_freq = <1000000>;
|
||||
};
|
||||
};
|
||||
};
|
||||
|
@ -30,6 +30,7 @@
|
||||
serial9 = &main_uart7;
|
||||
serial10 = &main_uart8;
|
||||
serial11 = &main_uart9;
|
||||
ethernet0 = &cpsw_port1;
|
||||
};
|
||||
|
||||
chosen { };
|
||||
|
@ -284,6 +284,7 @@ CONFIG_SMSC911X=y
|
||||
CONFIG_SNI_AVE=y
|
||||
CONFIG_SNI_NETSEC=y
|
||||
CONFIG_STMMAC_ETH=m
|
||||
CONFIG_TI_K3_AM65_CPSW_NUSS=y
|
||||
CONFIG_MDIO_BUS_MUX_MMIOREG=y
|
||||
CONFIG_MARVELL_PHY=m
|
||||
CONFIG_MARVELL_10G_PHY=m
|
||||
@ -699,6 +700,8 @@ CONFIG_QCOM_HIDMA_MGMT=y
|
||||
CONFIG_QCOM_HIDMA=y
|
||||
CONFIG_RCAR_DMAC=y
|
||||
CONFIG_RENESAS_USB_DMAC=m
|
||||
CONFIG_TI_K3_UDMA=y
|
||||
CONFIG_TI_K3_UDMA_GLUE_LAYER=y
|
||||
CONFIG_VFIO=y
|
||||
CONFIG_VFIO_PCI=y
|
||||
CONFIG_VIRTIO_PCI=y
|
||||
|
@ -303,12 +303,6 @@ SECTIONS
|
||||
*(.branch_lt)
|
||||
}
|
||||
|
||||
#ifdef CONFIG_DEBUG_INFO_BTF
|
||||
.BTF : AT(ADDR(.BTF) - LOAD_OFFSET) {
|
||||
*(.BTF)
|
||||
}
|
||||
#endif
|
||||
|
||||
.opd : AT(ADDR(.opd) - LOAD_OFFSET) {
|
||||
__start_opd = .;
|
||||
KEEP(*(.opd))
|
||||
|
@ -55,7 +55,7 @@ config RISCV
|
||||
select ARCH_HAS_PTE_SPECIAL
|
||||
select ARCH_HAS_MMIOWB
|
||||
select ARCH_HAS_DEBUG_VIRTUAL
|
||||
select HAVE_EBPF_JIT if 64BIT
|
||||
select HAVE_EBPF_JIT
|
||||
select EDAC_SUPPORT
|
||||
select ARCH_HAS_GIGANTIC_PAGE
|
||||
select ARCH_WANT_HUGE_PMD_SHARE if 64BIT
|
||||
|
@ -1,2 +1,9 @@
|
||||
# SPDX-License-Identifier: GPL-2.0-only
|
||||
obj-$(CONFIG_BPF_JIT) += bpf_jit_comp.o
|
||||
|
||||
obj-$(CONFIG_BPF_JIT) += bpf_jit_core.o
|
||||
|
||||
ifeq ($(CONFIG_ARCH_RV64I),y)
|
||||
obj-$(CONFIG_BPF_JIT) += bpf_jit_comp64.o
|
||||
else
|
||||
obj-$(CONFIG_BPF_JIT) += bpf_jit_comp32.o
|
||||
endif
|
||||
|
514
arch/riscv/net/bpf_jit.h
Normal file
514
arch/riscv/net/bpf_jit.h
Normal file
@ -0,0 +1,514 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* Common functionality for RV32 and RV64 BPF JIT compilers
|
||||
*
|
||||
* Copyright (c) 2019 Björn Töpel <bjorn.topel@gmail.com>
|
||||
*
|
||||
*/
|
||||
|
||||
#ifndef _BPF_JIT_H
|
||||
#define _BPF_JIT_H
|
||||
|
||||
#include <linux/bpf.h>
|
||||
#include <linux/filter.h>
|
||||
#include <asm/cacheflush.h>
|
||||
|
||||
enum {
|
||||
RV_REG_ZERO = 0, /* The constant value 0 */
|
||||
RV_REG_RA = 1, /* Return address */
|
||||
RV_REG_SP = 2, /* Stack pointer */
|
||||
RV_REG_GP = 3, /* Global pointer */
|
||||
RV_REG_TP = 4, /* Thread pointer */
|
||||
RV_REG_T0 = 5, /* Temporaries */
|
||||
RV_REG_T1 = 6,
|
||||
RV_REG_T2 = 7,
|
||||
RV_REG_FP = 8, /* Saved register/frame pointer */
|
||||
RV_REG_S1 = 9, /* Saved register */
|
||||
RV_REG_A0 = 10, /* Function argument/return values */
|
||||
RV_REG_A1 = 11, /* Function arguments */
|
||||
RV_REG_A2 = 12,
|
||||
RV_REG_A3 = 13,
|
||||
RV_REG_A4 = 14,
|
||||
RV_REG_A5 = 15,
|
||||
RV_REG_A6 = 16,
|
||||
RV_REG_A7 = 17,
|
||||
RV_REG_S2 = 18, /* Saved registers */
|
||||
RV_REG_S3 = 19,
|
||||
RV_REG_S4 = 20,
|
||||
RV_REG_S5 = 21,
|
||||
RV_REG_S6 = 22,
|
||||
RV_REG_S7 = 23,
|
||||
RV_REG_S8 = 24,
|
||||
RV_REG_S9 = 25,
|
||||
RV_REG_S10 = 26,
|
||||
RV_REG_S11 = 27,
|
||||
RV_REG_T3 = 28, /* Temporaries */
|
||||
RV_REG_T4 = 29,
|
||||
RV_REG_T5 = 30,
|
||||
RV_REG_T6 = 31,
|
||||
};
|
||||
|
||||
struct rv_jit_context {
|
||||
struct bpf_prog *prog;
|
||||
u32 *insns; /* RV insns */
|
||||
int ninsns;
|
||||
int epilogue_offset;
|
||||
int *offset; /* BPF to RV */
|
||||
unsigned long flags;
|
||||
int stack_size;
|
||||
};
|
||||
|
||||
struct rv_jit_data {
|
||||
struct bpf_binary_header *header;
|
||||
u8 *image;
|
||||
struct rv_jit_context ctx;
|
||||
};
|
||||
|
||||
static inline void bpf_fill_ill_insns(void *area, unsigned int size)
|
||||
{
|
||||
memset(area, 0, size);
|
||||
}
|
||||
|
||||
static inline void bpf_flush_icache(void *start, void *end)
|
||||
{
|
||||
flush_icache_range((unsigned long)start, (unsigned long)end);
|
||||
}
|
||||
|
||||
static inline void emit(const u32 insn, struct rv_jit_context *ctx)
|
||||
{
|
||||
if (ctx->insns)
|
||||
ctx->insns[ctx->ninsns] = insn;
|
||||
|
||||
ctx->ninsns++;
|
||||
}
|
||||
|
||||
static inline int epilogue_offset(struct rv_jit_context *ctx)
|
||||
{
|
||||
int to = ctx->epilogue_offset, from = ctx->ninsns;
|
||||
|
||||
return (to - from) << 2;
|
||||
}
|
||||
|
||||
/* Return -1 or inverted cond. */
|
||||
static inline int invert_bpf_cond(u8 cond)
|
||||
{
|
||||
switch (cond) {
|
||||
case BPF_JEQ:
|
||||
return BPF_JNE;
|
||||
case BPF_JGT:
|
||||
return BPF_JLE;
|
||||
case BPF_JLT:
|
||||
return BPF_JGE;
|
||||
case BPF_JGE:
|
||||
return BPF_JLT;
|
||||
case BPF_JLE:
|
||||
return BPF_JGT;
|
||||
case BPF_JNE:
|
||||
return BPF_JEQ;
|
||||
case BPF_JSGT:
|
||||
return BPF_JSLE;
|
||||
case BPF_JSLT:
|
||||
return BPF_JSGE;
|
||||
case BPF_JSGE:
|
||||
return BPF_JSLT;
|
||||
case BPF_JSLE:
|
||||
return BPF_JSGT;
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
static inline bool is_12b_int(long val)
|
||||
{
|
||||
return -(1L << 11) <= val && val < (1L << 11);
|
||||
}
|
||||
|
||||
static inline int is_12b_check(int off, int insn)
|
||||
{
|
||||
if (!is_12b_int(off)) {
|
||||
pr_err("bpf-jit: insn=%d 12b < offset=%d not supported yet!\n",
|
||||
insn, (int)off);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline bool is_13b_int(long val)
|
||||
{
|
||||
return -(1L << 12) <= val && val < (1L << 12);
|
||||
}
|
||||
|
||||
static inline bool is_21b_int(long val)
|
||||
{
|
||||
return -(1L << 20) <= val && val < (1L << 20);
|
||||
}
|
||||
|
||||
static inline int rv_offset(int insn, int off, struct rv_jit_context *ctx)
|
||||
{
|
||||
int from, to;
|
||||
|
||||
off++; /* BPF branch is from PC+1, RV is from PC */
|
||||
from = (insn > 0) ? ctx->offset[insn - 1] : 0;
|
||||
to = (insn + off > 0) ? ctx->offset[insn + off - 1] : 0;
|
||||
return (to - from) << 2;
|
||||
}
|
||||
|
||||
/* Instruction formats. */
|
||||
|
||||
static inline u32 rv_r_insn(u8 funct7, u8 rs2, u8 rs1, u8 funct3, u8 rd,
|
||||
u8 opcode)
|
||||
{
|
||||
return (funct7 << 25) | (rs2 << 20) | (rs1 << 15) | (funct3 << 12) |
|
||||
(rd << 7) | opcode;
|
||||
}
|
||||
|
||||
static inline u32 rv_i_insn(u16 imm11_0, u8 rs1, u8 funct3, u8 rd, u8 opcode)
|
||||
{
|
||||
return (imm11_0 << 20) | (rs1 << 15) | (funct3 << 12) | (rd << 7) |
|
||||
opcode;
|
||||
}
|
||||
|
||||
static inline u32 rv_s_insn(u16 imm11_0, u8 rs2, u8 rs1, u8 funct3, u8 opcode)
|
||||
{
|
||||
u8 imm11_5 = imm11_0 >> 5, imm4_0 = imm11_0 & 0x1f;
|
||||
|
||||
return (imm11_5 << 25) | (rs2 << 20) | (rs1 << 15) | (funct3 << 12) |
|
||||
(imm4_0 << 7) | opcode;
|
||||
}
|
||||
|
||||
static inline u32 rv_b_insn(u16 imm12_1, u8 rs2, u8 rs1, u8 funct3, u8 opcode)
|
||||
{
|
||||
u8 imm12 = ((imm12_1 & 0x800) >> 5) | ((imm12_1 & 0x3f0) >> 4);
|
||||
u8 imm4_1 = ((imm12_1 & 0xf) << 1) | ((imm12_1 & 0x400) >> 10);
|
||||
|
||||
return (imm12 << 25) | (rs2 << 20) | (rs1 << 15) | (funct3 << 12) |
|
||||
(imm4_1 << 7) | opcode;
|
||||
}
|
||||
|
||||
static inline u32 rv_u_insn(u32 imm31_12, u8 rd, u8 opcode)
|
||||
{
|
||||
return (imm31_12 << 12) | (rd << 7) | opcode;
|
||||
}
|
||||
|
||||
static inline u32 rv_j_insn(u32 imm20_1, u8 rd, u8 opcode)
|
||||
{
|
||||
u32 imm;
|
||||
|
||||
imm = (imm20_1 & 0x80000) | ((imm20_1 & 0x3ff) << 9) |
|
||||
((imm20_1 & 0x400) >> 2) | ((imm20_1 & 0x7f800) >> 11);
|
||||
|
||||
return (imm << 12) | (rd << 7) | opcode;
|
||||
}
|
||||
|
||||
static inline u32 rv_amo_insn(u8 funct5, u8 aq, u8 rl, u8 rs2, u8 rs1,
|
||||
u8 funct3, u8 rd, u8 opcode)
|
||||
{
|
||||
u8 funct7 = (funct5 << 2) | (aq << 1) | rl;
|
||||
|
||||
return rv_r_insn(funct7, rs2, rs1, funct3, rd, opcode);
|
||||
}
|
||||
|
||||
/* Instructions shared by both RV32 and RV64. */
|
||||
|
||||
static inline u32 rv_addi(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 0, rd, 0x13);
|
||||
}
|
||||
|
||||
static inline u32 rv_andi(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 7, rd, 0x13);
|
||||
}
|
||||
|
||||
static inline u32 rv_ori(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 6, rd, 0x13);
|
||||
}
|
||||
|
||||
static inline u32 rv_xori(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 4, rd, 0x13);
|
||||
}
|
||||
|
||||
static inline u32 rv_slli(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 1, rd, 0x13);
|
||||
}
|
||||
|
||||
static inline u32 rv_srli(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 5, rd, 0x13);
|
||||
}
|
||||
|
||||
static inline u32 rv_srai(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(0x400 | imm11_0, rs1, 5, rd, 0x13);
|
||||
}
|
||||
|
||||
static inline u32 rv_lui(u8 rd, u32 imm31_12)
|
||||
{
|
||||
return rv_u_insn(imm31_12, rd, 0x37);
|
||||
}
|
||||
|
||||
static inline u32 rv_auipc(u8 rd, u32 imm31_12)
|
||||
{
|
||||
return rv_u_insn(imm31_12, rd, 0x17);
|
||||
}
|
||||
|
||||
static inline u32 rv_add(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 0, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_sub(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0x20, rs2, rs1, 0, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_sltu(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 3, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_and(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 7, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_or(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 6, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_xor(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 4, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_sll(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 1, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_srl(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 5, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_sra(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0x20, rs2, rs1, 5, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_mul(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 0, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_mulhu(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 3, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_divu(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 5, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_remu(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 7, rd, 0x33);
|
||||
}
|
||||
|
||||
static inline u32 rv_jal(u8 rd, u32 imm20_1)
|
||||
{
|
||||
return rv_j_insn(imm20_1, rd, 0x6f);
|
||||
}
|
||||
|
||||
static inline u32 rv_jalr(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 0, rd, 0x67);
|
||||
}
|
||||
|
||||
static inline u32 rv_beq(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_b_insn(imm12_1, rs2, rs1, 0, 0x63);
|
||||
}
|
||||
|
||||
static inline u32 rv_bne(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_b_insn(imm12_1, rs2, rs1, 1, 0x63);
|
||||
}
|
||||
|
||||
static inline u32 rv_bltu(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_b_insn(imm12_1, rs2, rs1, 6, 0x63);
|
||||
}
|
||||
|
||||
static inline u32 rv_bgtu(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_bltu(rs2, rs1, imm12_1);
|
||||
}
|
||||
|
||||
static inline u32 rv_bgeu(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_b_insn(imm12_1, rs2, rs1, 7, 0x63);
|
||||
}
|
||||
|
||||
static inline u32 rv_bleu(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_bgeu(rs2, rs1, imm12_1);
|
||||
}
|
||||
|
||||
static inline u32 rv_blt(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_b_insn(imm12_1, rs2, rs1, 4, 0x63);
|
||||
}
|
||||
|
||||
static inline u32 rv_bgt(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_blt(rs2, rs1, imm12_1);
|
||||
}
|
||||
|
||||
static inline u32 rv_bge(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_b_insn(imm12_1, rs2, rs1, 5, 0x63);
|
||||
}
|
||||
|
||||
static inline u32 rv_ble(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_bge(rs2, rs1, imm12_1);
|
||||
}
|
||||
|
||||
static inline u32 rv_lw(u8 rd, u16 imm11_0, u8 rs1)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 2, rd, 0x03);
|
||||
}
|
||||
|
||||
static inline u32 rv_lbu(u8 rd, u16 imm11_0, u8 rs1)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 4, rd, 0x03);
|
||||
}
|
||||
|
||||
static inline u32 rv_lhu(u8 rd, u16 imm11_0, u8 rs1)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 5, rd, 0x03);
|
||||
}
|
||||
|
||||
static inline u32 rv_sb(u8 rs1, u16 imm11_0, u8 rs2)
|
||||
{
|
||||
return rv_s_insn(imm11_0, rs2, rs1, 0, 0x23);
|
||||
}
|
||||
|
||||
static inline u32 rv_sh(u8 rs1, u16 imm11_0, u8 rs2)
|
||||
{
|
||||
return rv_s_insn(imm11_0, rs2, rs1, 1, 0x23);
|
||||
}
|
||||
|
||||
static inline u32 rv_sw(u8 rs1, u16 imm11_0, u8 rs2)
|
||||
{
|
||||
return rv_s_insn(imm11_0, rs2, rs1, 2, 0x23);
|
||||
}
|
||||
|
||||
static inline u32 rv_amoadd_w(u8 rd, u8 rs2, u8 rs1, u8 aq, u8 rl)
|
||||
{
|
||||
return rv_amo_insn(0, aq, rl, rs2, rs1, 2, rd, 0x2f);
|
||||
}
|
||||
|
||||
/*
|
||||
* RV64-only instructions.
|
||||
*
|
||||
* These instructions are not available on RV32. Wrap them below a #if to
|
||||
* ensure that the RV32 JIT doesn't emit any of these instructions.
|
||||
*/
|
||||
|
||||
#if __riscv_xlen == 64
|
||||
|
||||
static inline u32 rv_addiw(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 0, rd, 0x1b);
|
||||
}
|
||||
|
||||
static inline u32 rv_slliw(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 1, rd, 0x1b);
|
||||
}
|
||||
|
||||
static inline u32 rv_srliw(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 5, rd, 0x1b);
|
||||
}
|
||||
|
||||
static inline u32 rv_sraiw(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(0x400 | imm11_0, rs1, 5, rd, 0x1b);
|
||||
}
|
||||
|
||||
static inline u32 rv_addw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 0, rd, 0x3b);
|
||||
}
|
||||
|
||||
static inline u32 rv_subw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0x20, rs2, rs1, 0, rd, 0x3b);
|
||||
}
|
||||
|
||||
static inline u32 rv_sllw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 1, rd, 0x3b);
|
||||
}
|
||||
|
||||
static inline u32 rv_srlw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 5, rd, 0x3b);
|
||||
}
|
||||
|
||||
static inline u32 rv_sraw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0x20, rs2, rs1, 5, rd, 0x3b);
|
||||
}
|
||||
|
||||
static inline u32 rv_mulw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 0, rd, 0x3b);
|
||||
}
|
||||
|
||||
static inline u32 rv_divuw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 5, rd, 0x3b);
|
||||
}
|
||||
|
||||
static inline u32 rv_remuw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 7, rd, 0x3b);
|
||||
}
|
||||
|
||||
static inline u32 rv_ld(u8 rd, u16 imm11_0, u8 rs1)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 3, rd, 0x03);
|
||||
}
|
||||
|
||||
static inline u32 rv_lwu(u8 rd, u16 imm11_0, u8 rs1)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 6, rd, 0x03);
|
||||
}
|
||||
|
||||
static inline u32 rv_sd(u8 rs1, u16 imm11_0, u8 rs2)
|
||||
{
|
||||
return rv_s_insn(imm11_0, rs2, rs1, 3, 0x23);
|
||||
}
|
||||
|
||||
static inline u32 rv_amoadd_d(u8 rd, u8 rs2, u8 rs1, u8 aq, u8 rl)
|
||||
{
|
||||
return rv_amo_insn(0, aq, rl, rs2, rs1, 3, rd, 0x2f);
|
||||
}
|
||||
|
||||
#endif /* __riscv_xlen == 64 */
|
||||
|
||||
void bpf_jit_build_prologue(struct rv_jit_context *ctx);
|
||||
void bpf_jit_build_epilogue(struct rv_jit_context *ctx);
|
||||
|
||||
int bpf_jit_emit_insn(const struct bpf_insn *insn, struct rv_jit_context *ctx,
|
||||
bool extra_pass);
|
||||
|
||||
#endif /* _BPF_JIT_H */
|
1310
arch/riscv/net/bpf_jit_comp32.c
Normal file
1310
arch/riscv/net/bpf_jit_comp32.c
Normal file
File diff suppressed because it is too large
Load Diff
@ -7,42 +7,7 @@
|
||||
|
||||
#include <linux/bpf.h>
|
||||
#include <linux/filter.h>
|
||||
#include <asm/cacheflush.h>
|
||||
|
||||
enum {
|
||||
RV_REG_ZERO = 0, /* The constant value 0 */
|
||||
RV_REG_RA = 1, /* Return address */
|
||||
RV_REG_SP = 2, /* Stack pointer */
|
||||
RV_REG_GP = 3, /* Global pointer */
|
||||
RV_REG_TP = 4, /* Thread pointer */
|
||||
RV_REG_T0 = 5, /* Temporaries */
|
||||
RV_REG_T1 = 6,
|
||||
RV_REG_T2 = 7,
|
||||
RV_REG_FP = 8,
|
||||
RV_REG_S1 = 9, /* Saved registers */
|
||||
RV_REG_A0 = 10, /* Function argument/return values */
|
||||
RV_REG_A1 = 11, /* Function arguments */
|
||||
RV_REG_A2 = 12,
|
||||
RV_REG_A3 = 13,
|
||||
RV_REG_A4 = 14,
|
||||
RV_REG_A5 = 15,
|
||||
RV_REG_A6 = 16,
|
||||
RV_REG_A7 = 17,
|
||||
RV_REG_S2 = 18, /* Saved registers */
|
||||
RV_REG_S3 = 19,
|
||||
RV_REG_S4 = 20,
|
||||
RV_REG_S5 = 21,
|
||||
RV_REG_S6 = 22,
|
||||
RV_REG_S7 = 23,
|
||||
RV_REG_S8 = 24,
|
||||
RV_REG_S9 = 25,
|
||||
RV_REG_S10 = 26,
|
||||
RV_REG_S11 = 27,
|
||||
RV_REG_T3 = 28, /* Temporaries */
|
||||
RV_REG_T4 = 29,
|
||||
RV_REG_T5 = 30,
|
||||
RV_REG_T6 = 31,
|
||||
};
|
||||
#include "bpf_jit.h"
|
||||
|
||||
#define RV_REG_TCC RV_REG_A6
|
||||
#define RV_REG_TCC_SAVED RV_REG_S6 /* Store A6 in S6 if program do calls */
|
||||
@ -73,22 +38,6 @@ enum {
|
||||
RV_CTX_F_SEEN_S6 = RV_REG_S6,
|
||||
};
|
||||
|
||||
struct rv_jit_context {
|
||||
struct bpf_prog *prog;
|
||||
u32 *insns; /* RV insns */
|
||||
int ninsns;
|
||||
int epilogue_offset;
|
||||
int *offset; /* BPF to RV */
|
||||
unsigned long flags;
|
||||
int stack_size;
|
||||
};
|
||||
|
||||
struct rv_jit_data {
|
||||
struct bpf_binary_header *header;
|
||||
u8 *image;
|
||||
struct rv_jit_context ctx;
|
||||
};
|
||||
|
||||
static u8 bpf_to_rv_reg(int bpf_reg, struct rv_jit_context *ctx)
|
||||
{
|
||||
u8 reg = regmap[bpf_reg];
|
||||
@ -156,346 +105,11 @@ static u8 rv_tail_call_reg(struct rv_jit_context *ctx)
|
||||
return RV_REG_A6;
|
||||
}
|
||||
|
||||
static void emit(const u32 insn, struct rv_jit_context *ctx)
|
||||
{
|
||||
if (ctx->insns)
|
||||
ctx->insns[ctx->ninsns] = insn;
|
||||
|
||||
ctx->ninsns++;
|
||||
}
|
||||
|
||||
static u32 rv_r_insn(u8 funct7, u8 rs2, u8 rs1, u8 funct3, u8 rd, u8 opcode)
|
||||
{
|
||||
return (funct7 << 25) | (rs2 << 20) | (rs1 << 15) | (funct3 << 12) |
|
||||
(rd << 7) | opcode;
|
||||
}
|
||||
|
||||
static u32 rv_i_insn(u16 imm11_0, u8 rs1, u8 funct3, u8 rd, u8 opcode)
|
||||
{
|
||||
return (imm11_0 << 20) | (rs1 << 15) | (funct3 << 12) | (rd << 7) |
|
||||
opcode;
|
||||
}
|
||||
|
||||
static u32 rv_s_insn(u16 imm11_0, u8 rs2, u8 rs1, u8 funct3, u8 opcode)
|
||||
{
|
||||
u8 imm11_5 = imm11_0 >> 5, imm4_0 = imm11_0 & 0x1f;
|
||||
|
||||
return (imm11_5 << 25) | (rs2 << 20) | (rs1 << 15) | (funct3 << 12) |
|
||||
(imm4_0 << 7) | opcode;
|
||||
}
|
||||
|
||||
static u32 rv_sb_insn(u16 imm12_1, u8 rs2, u8 rs1, u8 funct3, u8 opcode)
|
||||
{
|
||||
u8 imm12 = ((imm12_1 & 0x800) >> 5) | ((imm12_1 & 0x3f0) >> 4);
|
||||
u8 imm4_1 = ((imm12_1 & 0xf) << 1) | ((imm12_1 & 0x400) >> 10);
|
||||
|
||||
return (imm12 << 25) | (rs2 << 20) | (rs1 << 15) | (funct3 << 12) |
|
||||
(imm4_1 << 7) | opcode;
|
||||
}
|
||||
|
||||
static u32 rv_u_insn(u32 imm31_12, u8 rd, u8 opcode)
|
||||
{
|
||||
return (imm31_12 << 12) | (rd << 7) | opcode;
|
||||
}
|
||||
|
||||
static u32 rv_uj_insn(u32 imm20_1, u8 rd, u8 opcode)
|
||||
{
|
||||
u32 imm;
|
||||
|
||||
imm = (imm20_1 & 0x80000) | ((imm20_1 & 0x3ff) << 9) |
|
||||
((imm20_1 & 0x400) >> 2) | ((imm20_1 & 0x7f800) >> 11);
|
||||
|
||||
return (imm << 12) | (rd << 7) | opcode;
|
||||
}
|
||||
|
||||
static u32 rv_amo_insn(u8 funct5, u8 aq, u8 rl, u8 rs2, u8 rs1,
|
||||
u8 funct3, u8 rd, u8 opcode)
|
||||
{
|
||||
u8 funct7 = (funct5 << 2) | (aq << 1) | rl;
|
||||
|
||||
return rv_r_insn(funct7, rs2, rs1, funct3, rd, opcode);
|
||||
}
|
||||
|
||||
static u32 rv_addiw(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 0, rd, 0x1b);
|
||||
}
|
||||
|
||||
static u32 rv_addi(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 0, rd, 0x13);
|
||||
}
|
||||
|
||||
static u32 rv_addw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 0, rd, 0x3b);
|
||||
}
|
||||
|
||||
static u32 rv_add(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 0, rd, 0x33);
|
||||
}
|
||||
|
||||
static u32 rv_subw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0x20, rs2, rs1, 0, rd, 0x3b);
|
||||
}
|
||||
|
||||
static u32 rv_sub(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0x20, rs2, rs1, 0, rd, 0x33);
|
||||
}
|
||||
|
||||
static u32 rv_and(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 7, rd, 0x33);
|
||||
}
|
||||
|
||||
static u32 rv_or(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 6, rd, 0x33);
|
||||
}
|
||||
|
||||
static u32 rv_xor(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 4, rd, 0x33);
|
||||
}
|
||||
|
||||
static u32 rv_mulw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 0, rd, 0x3b);
|
||||
}
|
||||
|
||||
static u32 rv_mul(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 0, rd, 0x33);
|
||||
}
|
||||
|
||||
static u32 rv_divuw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 5, rd, 0x3b);
|
||||
}
|
||||
|
||||
static u32 rv_divu(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 5, rd, 0x33);
|
||||
}
|
||||
|
||||
static u32 rv_remuw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 7, rd, 0x3b);
|
||||
}
|
||||
|
||||
static u32 rv_remu(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(1, rs2, rs1, 7, rd, 0x33);
|
||||
}
|
||||
|
||||
static u32 rv_sllw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 1, rd, 0x3b);
|
||||
}
|
||||
|
||||
static u32 rv_sll(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 1, rd, 0x33);
|
||||
}
|
||||
|
||||
static u32 rv_srlw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 5, rd, 0x3b);
|
||||
}
|
||||
|
||||
static u32 rv_srl(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0, rs2, rs1, 5, rd, 0x33);
|
||||
}
|
||||
|
||||
static u32 rv_sraw(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0x20, rs2, rs1, 5, rd, 0x3b);
|
||||
}
|
||||
|
||||
static u32 rv_sra(u8 rd, u8 rs1, u8 rs2)
|
||||
{
|
||||
return rv_r_insn(0x20, rs2, rs1, 5, rd, 0x33);
|
||||
}
|
||||
|
||||
static u32 rv_lui(u8 rd, u32 imm31_12)
|
||||
{
|
||||
return rv_u_insn(imm31_12, rd, 0x37);
|
||||
}
|
||||
|
||||
static u32 rv_slli(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 1, rd, 0x13);
|
||||
}
|
||||
|
||||
static u32 rv_andi(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 7, rd, 0x13);
|
||||
}
|
||||
|
||||
static u32 rv_ori(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 6, rd, 0x13);
|
||||
}
|
||||
|
||||
static u32 rv_xori(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 4, rd, 0x13);
|
||||
}
|
||||
|
||||
static u32 rv_slliw(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 1, rd, 0x1b);
|
||||
}
|
||||
|
||||
static u32 rv_srliw(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 5, rd, 0x1b);
|
||||
}
|
||||
|
||||
static u32 rv_srli(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 5, rd, 0x13);
|
||||
}
|
||||
|
||||
static u32 rv_sraiw(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(0x400 | imm11_0, rs1, 5, rd, 0x1b);
|
||||
}
|
||||
|
||||
static u32 rv_srai(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(0x400 | imm11_0, rs1, 5, rd, 0x13);
|
||||
}
|
||||
|
||||
static u32 rv_jal(u8 rd, u32 imm20_1)
|
||||
{
|
||||
return rv_uj_insn(imm20_1, rd, 0x6f);
|
||||
}
|
||||
|
||||
static u32 rv_jalr(u8 rd, u8 rs1, u16 imm11_0)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 0, rd, 0x67);
|
||||
}
|
||||
|
||||
static u32 rv_beq(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_sb_insn(imm12_1, rs2, rs1, 0, 0x63);
|
||||
}
|
||||
|
||||
static u32 rv_bltu(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_sb_insn(imm12_1, rs2, rs1, 6, 0x63);
|
||||
}
|
||||
|
||||
static u32 rv_bgeu(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_sb_insn(imm12_1, rs2, rs1, 7, 0x63);
|
||||
}
|
||||
|
||||
static u32 rv_bne(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_sb_insn(imm12_1, rs2, rs1, 1, 0x63);
|
||||
}
|
||||
|
||||
static u32 rv_blt(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_sb_insn(imm12_1, rs2, rs1, 4, 0x63);
|
||||
}
|
||||
|
||||
static u32 rv_bge(u8 rs1, u8 rs2, u16 imm12_1)
|
||||
{
|
||||
return rv_sb_insn(imm12_1, rs2, rs1, 5, 0x63);
|
||||
}
|
||||
|
||||
static u32 rv_sb(u8 rs1, u16 imm11_0, u8 rs2)
|
||||
{
|
||||
return rv_s_insn(imm11_0, rs2, rs1, 0, 0x23);
|
||||
}
|
||||
|
||||
static u32 rv_sh(u8 rs1, u16 imm11_0, u8 rs2)
|
||||
{
|
||||
return rv_s_insn(imm11_0, rs2, rs1, 1, 0x23);
|
||||
}
|
||||
|
||||
static u32 rv_sw(u8 rs1, u16 imm11_0, u8 rs2)
|
||||
{
|
||||
return rv_s_insn(imm11_0, rs2, rs1, 2, 0x23);
|
||||
}
|
||||
|
||||
static u32 rv_sd(u8 rs1, u16 imm11_0, u8 rs2)
|
||||
{
|
||||
return rv_s_insn(imm11_0, rs2, rs1, 3, 0x23);
|
||||
}
|
||||
|
||||
static u32 rv_lbu(u8 rd, u16 imm11_0, u8 rs1)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 4, rd, 0x03);
|
||||
}
|
||||
|
||||
static u32 rv_lhu(u8 rd, u16 imm11_0, u8 rs1)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 5, rd, 0x03);
|
||||
}
|
||||
|
||||
static u32 rv_lwu(u8 rd, u16 imm11_0, u8 rs1)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 6, rd, 0x03);
|
||||
}
|
||||
|
||||
static u32 rv_ld(u8 rd, u16 imm11_0, u8 rs1)
|
||||
{
|
||||
return rv_i_insn(imm11_0, rs1, 3, rd, 0x03);
|
||||
}
|
||||
|
||||
static u32 rv_amoadd_w(u8 rd, u8 rs2, u8 rs1, u8 aq, u8 rl)
|
||||
{
|
||||
return rv_amo_insn(0, aq, rl, rs2, rs1, 2, rd, 0x2f);
|
||||
}
|
||||
|
||||
static u32 rv_amoadd_d(u8 rd, u8 rs2, u8 rs1, u8 aq, u8 rl)
|
||||
{
|
||||
return rv_amo_insn(0, aq, rl, rs2, rs1, 3, rd, 0x2f);
|
||||
}
|
||||
|
||||
static u32 rv_auipc(u8 rd, u32 imm31_12)
|
||||
{
|
||||
return rv_u_insn(imm31_12, rd, 0x17);
|
||||
}
|
||||
|
||||
static bool is_12b_int(s64 val)
|
||||
{
|
||||
return -(1 << 11) <= val && val < (1 << 11);
|
||||
}
|
||||
|
||||
static bool is_13b_int(s64 val)
|
||||
{
|
||||
return -(1 << 12) <= val && val < (1 << 12);
|
||||
}
|
||||
|
||||
static bool is_21b_int(s64 val)
|
||||
{
|
||||
return -(1L << 20) <= val && val < (1L << 20);
|
||||
}
|
||||
|
||||
static bool is_32b_int(s64 val)
|
||||
{
|
||||
return -(1L << 31) <= val && val < (1L << 31);
|
||||
}
|
||||
|
||||
static int is_12b_check(int off, int insn)
|
||||
{
|
||||
if (!is_12b_int(off)) {
|
||||
pr_err("bpf-jit: insn=%d 12b < offset=%d not supported yet!\n",
|
||||
insn, (int)off);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void emit_imm(u8 rd, s64 val, struct rv_jit_context *ctx)
|
||||
{
|
||||
/* Note that the immediate from the add is sign-extended,
|
||||
@ -535,23 +149,6 @@ static void emit_imm(u8 rd, s64 val, struct rv_jit_context *ctx)
|
||||
emit(rv_addi(rd, rd, lower), ctx);
|
||||
}
|
||||
|
||||
static int rv_offset(int insn, int off, struct rv_jit_context *ctx)
|
||||
{
|
||||
int from, to;
|
||||
|
||||
off++; /* BPF branch is from PC+1, RV is from PC */
|
||||
from = (insn > 0) ? ctx->offset[insn - 1] : 0;
|
||||
to = (insn + off > 0) ? ctx->offset[insn + off - 1] : 0;
|
||||
return (to - from) << 2;
|
||||
}
|
||||
|
||||
static int epilogue_offset(struct rv_jit_context *ctx)
|
||||
{
|
||||
int to = ctx->epilogue_offset, from = ctx->ninsns;
|
||||
|
||||
return (to - from) << 2;
|
||||
}
|
||||
|
||||
static void __build_epilogue(bool is_tail_call, struct rv_jit_context *ctx)
|
||||
{
|
||||
int stack_adjust = ctx->stack_size, store_offset = stack_adjust - 8;
|
||||
@ -596,34 +193,6 @@ static void __build_epilogue(bool is_tail_call, struct rv_jit_context *ctx)
|
||||
ctx);
|
||||
}
|
||||
|
||||
/* return -1 or inverted cond */
|
||||
static int invert_bpf_cond(u8 cond)
|
||||
{
|
||||
switch (cond) {
|
||||
case BPF_JEQ:
|
||||
return BPF_JNE;
|
||||
case BPF_JGT:
|
||||
return BPF_JLE;
|
||||
case BPF_JLT:
|
||||
return BPF_JGE;
|
||||
case BPF_JGE:
|
||||
return BPF_JLT;
|
||||
case BPF_JLE:
|
||||
return BPF_JGT;
|
||||
case BPF_JNE:
|
||||
return BPF_JEQ;
|
||||
case BPF_JSGT:
|
||||
return BPF_JSLE;
|
||||
case BPF_JSLT:
|
||||
return BPF_JSGE;
|
||||
case BPF_JSGE:
|
||||
return BPF_JSLT;
|
||||
case BPF_JSLE:
|
||||
return BPF_JSGT;
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
static void emit_bcc(u8 cond, u8 rd, u8 rs, int rvoff,
|
||||
struct rv_jit_context *ctx)
|
||||
{
|
||||
@ -855,8 +424,8 @@ static int emit_call(bool fixed, u64 addr, struct rv_jit_context *ctx)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int emit_insn(const struct bpf_insn *insn, struct rv_jit_context *ctx,
|
||||
bool extra_pass)
|
||||
int bpf_jit_emit_insn(const struct bpf_insn *insn, struct rv_jit_context *ctx,
|
||||
bool extra_pass)
|
||||
{
|
||||
bool is64 = BPF_CLASS(insn->code) == BPF_ALU64 ||
|
||||
BPF_CLASS(insn->code) == BPF_JMP;
|
||||
@ -1434,7 +1003,7 @@ out_be:
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void build_prologue(struct rv_jit_context *ctx)
|
||||
void bpf_jit_build_prologue(struct rv_jit_context *ctx)
|
||||
{
|
||||
int stack_adjust = 0, store_offset, bpf_stack_adjust;
|
||||
|
||||
@ -1515,175 +1084,11 @@ static void build_prologue(struct rv_jit_context *ctx)
|
||||
ctx->stack_size = stack_adjust;
|
||||
}
|
||||
|
||||
static void build_epilogue(struct rv_jit_context *ctx)
|
||||
void bpf_jit_build_epilogue(struct rv_jit_context *ctx)
|
||||
{
|
||||
__build_epilogue(false, ctx);
|
||||
}
|
||||
|
||||
static int build_body(struct rv_jit_context *ctx, bool extra_pass, int *offset)
|
||||
{
|
||||
const struct bpf_prog *prog = ctx->prog;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < prog->len; i++) {
|
||||
const struct bpf_insn *insn = &prog->insnsi[i];
|
||||
int ret;
|
||||
|
||||
ret = emit_insn(insn, ctx, extra_pass);
|
||||
if (ret > 0) {
|
||||
i++;
|
||||
if (offset)
|
||||
offset[i] = ctx->ninsns;
|
||||
continue;
|
||||
}
|
||||
if (offset)
|
||||
offset[i] = ctx->ninsns;
|
||||
if (ret)
|
||||
return ret;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void bpf_fill_ill_insns(void *area, unsigned int size)
|
||||
{
|
||||
memset(area, 0, size);
|
||||
}
|
||||
|
||||
static void bpf_flush_icache(void *start, void *end)
|
||||
{
|
||||
flush_icache_range((unsigned long)start, (unsigned long)end);
|
||||
}
|
||||
|
||||
bool bpf_jit_needs_zext(void)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
struct bpf_prog *bpf_int_jit_compile(struct bpf_prog *prog)
|
||||
{
|
||||
bool tmp_blinded = false, extra_pass = false;
|
||||
struct bpf_prog *tmp, *orig_prog = prog;
|
||||
int pass = 0, prev_ninsns = 0, i;
|
||||
struct rv_jit_data *jit_data;
|
||||
unsigned int image_size = 0;
|
||||
struct rv_jit_context *ctx;
|
||||
|
||||
if (!prog->jit_requested)
|
||||
return orig_prog;
|
||||
|
||||
tmp = bpf_jit_blind_constants(prog);
|
||||
if (IS_ERR(tmp))
|
||||
return orig_prog;
|
||||
if (tmp != prog) {
|
||||
tmp_blinded = true;
|
||||
prog = tmp;
|
||||
}
|
||||
|
||||
jit_data = prog->aux->jit_data;
|
||||
if (!jit_data) {
|
||||
jit_data = kzalloc(sizeof(*jit_data), GFP_KERNEL);
|
||||
if (!jit_data) {
|
||||
prog = orig_prog;
|
||||
goto out;
|
||||
}
|
||||
prog->aux->jit_data = jit_data;
|
||||
}
|
||||
|
||||
ctx = &jit_data->ctx;
|
||||
|
||||
if (ctx->offset) {
|
||||
extra_pass = true;
|
||||
image_size = sizeof(u32) * ctx->ninsns;
|
||||
goto skip_init_ctx;
|
||||
}
|
||||
|
||||
ctx->prog = prog;
|
||||
ctx->offset = kcalloc(prog->len, sizeof(int), GFP_KERNEL);
|
||||
if (!ctx->offset) {
|
||||
prog = orig_prog;
|
||||
goto out_offset;
|
||||
}
|
||||
for (i = 0; i < prog->len; i++) {
|
||||
prev_ninsns += 32;
|
||||
ctx->offset[i] = prev_ninsns;
|
||||
}
|
||||
|
||||
for (i = 0; i < 16; i++) {
|
||||
pass++;
|
||||
ctx->ninsns = 0;
|
||||
if (build_body(ctx, extra_pass, ctx->offset)) {
|
||||
prog = orig_prog;
|
||||
goto out_offset;
|
||||
}
|
||||
build_prologue(ctx);
|
||||
ctx->epilogue_offset = ctx->ninsns;
|
||||
build_epilogue(ctx);
|
||||
|
||||
if (ctx->ninsns == prev_ninsns) {
|
||||
if (jit_data->header)
|
||||
break;
|
||||
|
||||
image_size = sizeof(u32) * ctx->ninsns;
|
||||
jit_data->header =
|
||||
bpf_jit_binary_alloc(image_size,
|
||||
&jit_data->image,
|
||||
sizeof(u32),
|
||||
bpf_fill_ill_insns);
|
||||
if (!jit_data->header) {
|
||||
prog = orig_prog;
|
||||
goto out_offset;
|
||||
}
|
||||
|
||||
ctx->insns = (u32 *)jit_data->image;
|
||||
/* Now, when the image is allocated, the image
|
||||
* can potentially shrink more (auipc/jalr ->
|
||||
* jal).
|
||||
*/
|
||||
}
|
||||
prev_ninsns = ctx->ninsns;
|
||||
}
|
||||
|
||||
if (i == 16) {
|
||||
pr_err("bpf-jit: image did not converge in <%d passes!\n", i);
|
||||
bpf_jit_binary_free(jit_data->header);
|
||||
prog = orig_prog;
|
||||
goto out_offset;
|
||||
}
|
||||
|
||||
skip_init_ctx:
|
||||
pass++;
|
||||
ctx->ninsns = 0;
|
||||
|
||||
build_prologue(ctx);
|
||||
if (build_body(ctx, extra_pass, NULL)) {
|
||||
bpf_jit_binary_free(jit_data->header);
|
||||
prog = orig_prog;
|
||||
goto out_offset;
|
||||
}
|
||||
build_epilogue(ctx);
|
||||
|
||||
if (bpf_jit_enable > 1)
|
||||
bpf_jit_dump(prog->len, image_size, pass, ctx->insns);
|
||||
|
||||
prog->bpf_func = (void *)ctx->insns;
|
||||
prog->jited = 1;
|
||||
prog->jited_len = image_size;
|
||||
|
||||
bpf_flush_icache(jit_data->header, ctx->insns + ctx->ninsns);
|
||||
|
||||
if (!prog->is_func || extra_pass) {
|
||||
out_offset:
|
||||
kfree(ctx->offset);
|
||||
kfree(jit_data);
|
||||
prog->aux->jit_data = NULL;
|
||||
}
|
||||
out:
|
||||
if (tmp_blinded)
|
||||
bpf_jit_prog_release_other(prog, prog == orig_prog ?
|
||||
tmp : orig_prog);
|
||||
return prog;
|
||||
}
|
||||
|
||||
void *bpf_jit_alloc_exec(unsigned long size)
|
||||
{
|
||||
return __vmalloc_node_range(size, PAGE_SIZE, BPF_JIT_REGION_START,
|
166
arch/riscv/net/bpf_jit_core.c
Normal file
166
arch/riscv/net/bpf_jit_core.c
Normal file
@ -0,0 +1,166 @@
|
||||
// SPDX-License-Identifier: GPL-2.0
|
||||
/*
|
||||
* Common functionality for RV32 and RV64 BPF JIT compilers
|
||||
*
|
||||
* Copyright (c) 2019 Björn Töpel <bjorn.topel@gmail.com>
|
||||
*
|
||||
*/
|
||||
|
||||
#include <linux/bpf.h>
|
||||
#include <linux/filter.h>
|
||||
#include "bpf_jit.h"
|
||||
|
||||
/* Number of iterations to try until offsets converge. */
|
||||
#define NR_JIT_ITERATIONS 16
|
||||
|
||||
static int build_body(struct rv_jit_context *ctx, bool extra_pass, int *offset)
|
||||
{
|
||||
const struct bpf_prog *prog = ctx->prog;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < prog->len; i++) {
|
||||
const struct bpf_insn *insn = &prog->insnsi[i];
|
||||
int ret;
|
||||
|
||||
ret = bpf_jit_emit_insn(insn, ctx, extra_pass);
|
||||
/* BPF_LD | BPF_IMM | BPF_DW: skip the next instruction. */
|
||||
if (ret > 0)
|
||||
i++;
|
||||
if (offset)
|
||||
offset[i] = ctx->ninsns;
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
bool bpf_jit_needs_zext(void)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
struct bpf_prog *bpf_int_jit_compile(struct bpf_prog *prog)
|
||||
{
|
||||
bool tmp_blinded = false, extra_pass = false;
|
||||
struct bpf_prog *tmp, *orig_prog = prog;
|
||||
int pass = 0, prev_ninsns = 0, i;
|
||||
struct rv_jit_data *jit_data;
|
||||
struct rv_jit_context *ctx;
|
||||
unsigned int image_size = 0;
|
||||
|
||||
if (!prog->jit_requested)
|
||||
return orig_prog;
|
||||
|
||||
tmp = bpf_jit_blind_constants(prog);
|
||||
if (IS_ERR(tmp))
|
||||
return orig_prog;
|
||||
if (tmp != prog) {
|
||||
tmp_blinded = true;
|
||||
prog = tmp;
|
||||
}
|
||||
|
||||
jit_data = prog->aux->jit_data;
|
||||
if (!jit_data) {
|
||||
jit_data = kzalloc(sizeof(*jit_data), GFP_KERNEL);
|
||||
if (!jit_data) {
|
||||
prog = orig_prog;
|
||||
goto out;
|
||||
}
|
||||
prog->aux->jit_data = jit_data;
|
||||
}
|
||||
|
||||
ctx = &jit_data->ctx;
|
||||
|
||||
if (ctx->offset) {
|
||||
extra_pass = true;
|
||||
image_size = sizeof(u32) * ctx->ninsns;
|
||||
goto skip_init_ctx;
|
||||
}
|
||||
|
||||
ctx->prog = prog;
|
||||
ctx->offset = kcalloc(prog->len, sizeof(int), GFP_KERNEL);
|
||||
if (!ctx->offset) {
|
||||
prog = orig_prog;
|
||||
goto out_offset;
|
||||
}
|
||||
for (i = 0; i < prog->len; i++) {
|
||||
prev_ninsns += 32;
|
||||
ctx->offset[i] = prev_ninsns;
|
||||
}
|
||||
|
||||
for (i = 0; i < NR_JIT_ITERATIONS; i++) {
|
||||
pass++;
|
||||
ctx->ninsns = 0;
|
||||
if (build_body(ctx, extra_pass, ctx->offset)) {
|
||||
prog = orig_prog;
|
||||
goto out_offset;
|
||||
}
|
||||
bpf_jit_build_prologue(ctx);
|
||||
ctx->epilogue_offset = ctx->ninsns;
|
||||
bpf_jit_build_epilogue(ctx);
|
||||
|
||||
if (ctx->ninsns == prev_ninsns) {
|
||||
if (jit_data->header)
|
||||
break;
|
||||
|
||||
image_size = sizeof(u32) * ctx->ninsns;
|
||||
jit_data->header =
|
||||
bpf_jit_binary_alloc(image_size,
|
||||
&jit_data->image,
|
||||
sizeof(u32),
|
||||
bpf_fill_ill_insns);
|
||||
if (!jit_data->header) {
|
||||
prog = orig_prog;
|
||||
goto out_offset;
|
||||
}
|
||||
|
||||
ctx->insns = (u32 *)jit_data->image;
|
||||
/*
|
||||
* Now, when the image is allocated, the image can
|
||||
* potentially shrink more (auipc/jalr -> jal).
|
||||
*/
|
||||
}
|
||||
prev_ninsns = ctx->ninsns;
|
||||
}
|
||||
|
||||
if (i == NR_JIT_ITERATIONS) {
|
||||
pr_err("bpf-jit: image did not converge in <%d passes!\n", i);
|
||||
bpf_jit_binary_free(jit_data->header);
|
||||
prog = orig_prog;
|
||||
goto out_offset;
|
||||
}
|
||||
|
||||
skip_init_ctx:
|
||||
pass++;
|
||||
ctx->ninsns = 0;
|
||||
|
||||
bpf_jit_build_prologue(ctx);
|
||||
if (build_body(ctx, extra_pass, NULL)) {
|
||||
bpf_jit_binary_free(jit_data->header);
|
||||
prog = orig_prog;
|
||||
goto out_offset;
|
||||
}
|
||||
bpf_jit_build_epilogue(ctx);
|
||||
|
||||
if (bpf_jit_enable > 1)
|
||||
bpf_jit_dump(prog->len, image_size, pass, ctx->insns);
|
||||
|
||||
prog->bpf_func = (void *)ctx->insns;
|
||||
prog->jited = 1;
|
||||
prog->jited_len = image_size;
|
||||
|
||||
bpf_flush_icache(jit_data->header, ctx->insns + ctx->ninsns);
|
||||
|
||||
if (!prog->is_func || extra_pass) {
|
||||
out_offset:
|
||||
kfree(ctx->offset);
|
||||
kfree(jit_data);
|
||||
prog->aux->jit_data = NULL;
|
||||
}
|
||||
out:
|
||||
|
||||
if (tmp_blinded)
|
||||
bpf_jit_prog_release_other(prog, prog == orig_prog ?
|
||||
tmp : orig_prog);
|
||||
return prog;
|
||||
}
|
@ -338,7 +338,7 @@ typedef void qdio_handler_t(struct ccw_device *, unsigned int, int,
|
||||
* @no_output_qs: number of output queues
|
||||
* @input_handler: handler to be called for input queues
|
||||
* @output_handler: handler to be called for output queues
|
||||
* @queue_start_poll_array: polling handlers (one per input queue or NULL)
|
||||
* @irq_poll: Data IRQ polling handler (NULL when not supported)
|
||||
* @scan_threshold: # of in-use buffers that triggers scan on output queue
|
||||
* @int_parm: interruption parameter
|
||||
* @input_sbal_addr_array: address of no_input_qs * 128 pointers
|
||||
@ -359,8 +359,7 @@ struct qdio_initialize {
|
||||
unsigned int no_output_qs;
|
||||
qdio_handler_t *input_handler;
|
||||
qdio_handler_t *output_handler;
|
||||
void (**queue_start_poll_array) (struct ccw_device *, int,
|
||||
unsigned long);
|
||||
void (*irq_poll)(struct ccw_device *cdev, unsigned long data);
|
||||
unsigned int scan_threshold;
|
||||
unsigned long int_parm;
|
||||
struct qdio_buffer **input_sbal_addr_array;
|
||||
@ -415,8 +414,8 @@ extern int qdio_activate(struct ccw_device *);
|
||||
extern void qdio_release_aob(struct qaob *);
|
||||
extern int do_QDIO(struct ccw_device *, unsigned int, int, unsigned int,
|
||||
unsigned int);
|
||||
extern int qdio_start_irq(struct ccw_device *, int);
|
||||
extern int qdio_stop_irq(struct ccw_device *, int);
|
||||
extern int qdio_start_irq(struct ccw_device *cdev);
|
||||
extern int qdio_stop_irq(struct ccw_device *cdev);
|
||||
extern int qdio_get_next_buffers(struct ccw_device *, int, int *, int *);
|
||||
extern int qdio_inspect_queue(struct ccw_device *cdev, unsigned int nr,
|
||||
bool is_input, unsigned int *bufnr,
|
||||
|
@ -1508,6 +1508,7 @@ static int vector_set_coalesce(struct net_device *netdev,
|
||||
}
|
||||
|
||||
static const struct ethtool_ops vector_net_ethtool_ops = {
|
||||
.supported_coalesce_params = ETHTOOL_COALESCE_TX_USECS,
|
||||
.get_drvinfo = vector_net_get_drvinfo,
|
||||
.get_link = ethtool_op_get_link,
|
||||
.get_ts_info = ethtool_op_get_ts_info,
|
||||
|
@ -238,7 +238,11 @@ page_table_range_init(unsigned long start, unsigned long end, pgd_t *pgd_base)
|
||||
}
|
||||
}
|
||||
|
||||
static inline int is_kernel_text(unsigned long addr)
|
||||
/*
|
||||
* The <linux/kallsyms.h> already defines is_kernel_text,
|
||||
* using '__' prefix not to get in conflict.
|
||||
*/
|
||||
static inline int __is_kernel_text(unsigned long addr)
|
||||
{
|
||||
if (addr >= (unsigned long)_text && addr <= (unsigned long)__init_end)
|
||||
return 1;
|
||||
@ -328,8 +332,8 @@ repeat:
|
||||
addr2 = (pfn + PTRS_PER_PTE-1) * PAGE_SIZE +
|
||||
PAGE_OFFSET + PAGE_SIZE-1;
|
||||
|
||||
if (is_kernel_text(addr) ||
|
||||
is_kernel_text(addr2))
|
||||
if (__is_kernel_text(addr) ||
|
||||
__is_kernel_text(addr2))
|
||||
prot = PAGE_KERNEL_LARGE_EXEC;
|
||||
|
||||
pages_2m++;
|
||||
@ -354,7 +358,7 @@ repeat:
|
||||
*/
|
||||
pgprot_t init_prot = __pgprot(PTE_IDENT_ATTR);
|
||||
|
||||
if (is_kernel_text(addr))
|
||||
if (__is_kernel_text(addr))
|
||||
prot = PAGE_KERNEL_EXEC;
|
||||
|
||||
pages_4k++;
|
||||
@ -843,7 +847,7 @@ static void mark_nxdata_nx(void)
|
||||
*/
|
||||
unsigned long start = PFN_ALIGN(_etext);
|
||||
/*
|
||||
* This comes from is_kernel_text upper limit. Also HPAGE where used:
|
||||
* This comes from __is_kernel_text upper limit. Also HPAGE where used:
|
||||
*/
|
||||
unsigned long size = (((unsigned long)__init_end + HPAGE_SIZE) & HPAGE_MASK) - start;
|
||||
|
||||
|
@ -1361,41 +1361,144 @@ static void restore_regs(const struct btf_func_model *m, u8 **prog, int nr_args,
|
||||
-(stack_size - i * 8));
|
||||
}
|
||||
|
||||
static int invoke_bpf(const struct btf_func_model *m, u8 **pprog,
|
||||
struct bpf_prog **progs, int prog_cnt, int stack_size)
|
||||
static int invoke_bpf_prog(const struct btf_func_model *m, u8 **pprog,
|
||||
struct bpf_prog *p, int stack_size, bool mod_ret)
|
||||
{
|
||||
u8 *prog = *pprog;
|
||||
int cnt = 0, i;
|
||||
int cnt = 0;
|
||||
|
||||
for (i = 0; i < prog_cnt; i++) {
|
||||
if (emit_call(&prog, __bpf_prog_enter, prog))
|
||||
return -EINVAL;
|
||||
/* remember prog start time returned by __bpf_prog_enter */
|
||||
emit_mov_reg(&prog, true, BPF_REG_6, BPF_REG_0);
|
||||
if (emit_call(&prog, __bpf_prog_enter, prog))
|
||||
return -EINVAL;
|
||||
/* remember prog start time returned by __bpf_prog_enter */
|
||||
emit_mov_reg(&prog, true, BPF_REG_6, BPF_REG_0);
|
||||
|
||||
/* arg1: lea rdi, [rbp - stack_size] */
|
||||
EMIT4(0x48, 0x8D, 0x7D, -stack_size);
|
||||
/* arg2: progs[i]->insnsi for interpreter */
|
||||
if (!progs[i]->jited)
|
||||
emit_mov_imm64(&prog, BPF_REG_2,
|
||||
(long) progs[i]->insnsi >> 32,
|
||||
(u32) (long) progs[i]->insnsi);
|
||||
/* call JITed bpf program or interpreter */
|
||||
if (emit_call(&prog, progs[i]->bpf_func, prog))
|
||||
return -EINVAL;
|
||||
/* arg1: lea rdi, [rbp - stack_size] */
|
||||
EMIT4(0x48, 0x8D, 0x7D, -stack_size);
|
||||
/* arg2: progs[i]->insnsi for interpreter */
|
||||
if (!p->jited)
|
||||
emit_mov_imm64(&prog, BPF_REG_2,
|
||||
(long) p->insnsi >> 32,
|
||||
(u32) (long) p->insnsi);
|
||||
/* call JITed bpf program or interpreter */
|
||||
if (emit_call(&prog, p->bpf_func, prog))
|
||||
return -EINVAL;
|
||||
|
||||
/* arg1: mov rdi, progs[i] */
|
||||
emit_mov_imm64(&prog, BPF_REG_1, (long) progs[i] >> 32,
|
||||
(u32) (long) progs[i]);
|
||||
/* arg2: mov rsi, rbx <- start time in nsec */
|
||||
emit_mov_reg(&prog, true, BPF_REG_2, BPF_REG_6);
|
||||
if (emit_call(&prog, __bpf_prog_exit, prog))
|
||||
/* BPF_TRAMP_MODIFY_RETURN trampolines can modify the return
|
||||
* of the previous call which is then passed on the stack to
|
||||
* the next BPF program.
|
||||
*/
|
||||
if (mod_ret)
|
||||
emit_stx(&prog, BPF_DW, BPF_REG_FP, BPF_REG_0, -8);
|
||||
|
||||
/* arg1: mov rdi, progs[i] */
|
||||
emit_mov_imm64(&prog, BPF_REG_1, (long) p >> 32,
|
||||
(u32) (long) p);
|
||||
/* arg2: mov rsi, rbx <- start time in nsec */
|
||||
emit_mov_reg(&prog, true, BPF_REG_2, BPF_REG_6);
|
||||
if (emit_call(&prog, __bpf_prog_exit, prog))
|
||||
return -EINVAL;
|
||||
|
||||
*pprog = prog;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void emit_nops(u8 **pprog, unsigned int len)
|
||||
{
|
||||
unsigned int i, noplen;
|
||||
u8 *prog = *pprog;
|
||||
int cnt = 0;
|
||||
|
||||
while (len > 0) {
|
||||
noplen = len;
|
||||
|
||||
if (noplen > ASM_NOP_MAX)
|
||||
noplen = ASM_NOP_MAX;
|
||||
|
||||
for (i = 0; i < noplen; i++)
|
||||
EMIT1(ideal_nops[noplen][i]);
|
||||
len -= noplen;
|
||||
}
|
||||
|
||||
*pprog = prog;
|
||||
}
|
||||
|
||||
static void emit_align(u8 **pprog, u32 align)
|
||||
{
|
||||
u8 *target, *prog = *pprog;
|
||||
|
||||
target = PTR_ALIGN(prog, align);
|
||||
if (target != prog)
|
||||
emit_nops(&prog, target - prog);
|
||||
|
||||
*pprog = prog;
|
||||
}
|
||||
|
||||
static int emit_cond_near_jump(u8 **pprog, void *func, void *ip, u8 jmp_cond)
|
||||
{
|
||||
u8 *prog = *pprog;
|
||||
int cnt = 0;
|
||||
s64 offset;
|
||||
|
||||
offset = func - (ip + 2 + 4);
|
||||
if (!is_simm32(offset)) {
|
||||
pr_err("Target %p is out of range\n", func);
|
||||
return -EINVAL;
|
||||
}
|
||||
EMIT2_off32(0x0F, jmp_cond + 0x10, offset);
|
||||
*pprog = prog;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int invoke_bpf(const struct btf_func_model *m, u8 **pprog,
|
||||
struct bpf_tramp_progs *tp, int stack_size)
|
||||
{
|
||||
int i;
|
||||
u8 *prog = *pprog;
|
||||
|
||||
for (i = 0; i < tp->nr_progs; i++) {
|
||||
if (invoke_bpf_prog(m, &prog, tp->progs[i], stack_size, false))
|
||||
return -EINVAL;
|
||||
}
|
||||
*pprog = prog;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int invoke_bpf_mod_ret(const struct btf_func_model *m, u8 **pprog,
|
||||
struct bpf_tramp_progs *tp, int stack_size,
|
||||
u8 **branches)
|
||||
{
|
||||
u8 *prog = *pprog;
|
||||
int i, cnt = 0;
|
||||
|
||||
/* The first fmod_ret program will receive a garbage return value.
|
||||
* Set this to 0 to avoid confusing the program.
|
||||
*/
|
||||
emit_mov_imm32(&prog, false, BPF_REG_0, 0);
|
||||
emit_stx(&prog, BPF_DW, BPF_REG_FP, BPF_REG_0, -8);
|
||||
for (i = 0; i < tp->nr_progs; i++) {
|
||||
if (invoke_bpf_prog(m, &prog, tp->progs[i], stack_size, true))
|
||||
return -EINVAL;
|
||||
|
||||
/* mod_ret prog stored return value into [rbp - 8]. Emit:
|
||||
* if (*(u64 *)(rbp - 8) != 0)
|
||||
* goto do_fexit;
|
||||
*/
|
||||
/* cmp QWORD PTR [rbp - 0x8], 0x0 */
|
||||
EMIT4(0x48, 0x83, 0x7d, 0xf8); EMIT1(0x00);
|
||||
|
||||
/* Save the location of the branch and Generate 6 nops
|
||||
* (4 bytes for an offset and 2 bytes for the jump) These nops
|
||||
* are replaced with a conditional jump once do_fexit (i.e. the
|
||||
* start of the fexit invocation) is finalized.
|
||||
*/
|
||||
branches[i] = prog;
|
||||
emit_nops(&prog, 4 + 2);
|
||||
}
|
||||
|
||||
*pprog = prog;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Example:
|
||||
* __be16 eth_type_trans(struct sk_buff *skb, struct net_device *dev);
|
||||
* its 'struct btf_func_model' will be nr_args=2
|
||||
@ -1458,12 +1561,15 @@ static int invoke_bpf(const struct btf_func_model *m, u8 **pprog,
|
||||
*/
|
||||
int arch_prepare_bpf_trampoline(void *image, void *image_end,
|
||||
const struct btf_func_model *m, u32 flags,
|
||||
struct bpf_prog **fentry_progs, int fentry_cnt,
|
||||
struct bpf_prog **fexit_progs, int fexit_cnt,
|
||||
struct bpf_tramp_progs *tprogs,
|
||||
void *orig_call)
|
||||
{
|
||||
int cnt = 0, nr_args = m->nr_args;
|
||||
int ret, i, cnt = 0, nr_args = m->nr_args;
|
||||
int stack_size = nr_args * 8;
|
||||
struct bpf_tramp_progs *fentry = &tprogs[BPF_TRAMP_FENTRY];
|
||||
struct bpf_tramp_progs *fexit = &tprogs[BPF_TRAMP_FEXIT];
|
||||
struct bpf_tramp_progs *fmod_ret = &tprogs[BPF_TRAMP_MODIFY_RETURN];
|
||||
u8 **branches = NULL;
|
||||
u8 *prog;
|
||||
|
||||
/* x86-64 supports up to 6 arguments. 7+ can be added in the future */
|
||||
@ -1492,28 +1598,64 @@ int arch_prepare_bpf_trampoline(void *image, void *image_end,
|
||||
|
||||
save_regs(m, &prog, nr_args, stack_size);
|
||||
|
||||
if (fentry_cnt)
|
||||
if (invoke_bpf(m, &prog, fentry_progs, fentry_cnt, stack_size))
|
||||
if (fentry->nr_progs)
|
||||
if (invoke_bpf(m, &prog, fentry, stack_size))
|
||||
return -EINVAL;
|
||||
|
||||
if (fmod_ret->nr_progs) {
|
||||
branches = kcalloc(fmod_ret->nr_progs, sizeof(u8 *),
|
||||
GFP_KERNEL);
|
||||
if (!branches)
|
||||
return -ENOMEM;
|
||||
|
||||
if (invoke_bpf_mod_ret(m, &prog, fmod_ret, stack_size,
|
||||
branches)) {
|
||||
ret = -EINVAL;
|
||||
goto cleanup;
|
||||
}
|
||||
}
|
||||
|
||||
if (flags & BPF_TRAMP_F_CALL_ORIG) {
|
||||
if (fentry_cnt)
|
||||
if (fentry->nr_progs || fmod_ret->nr_progs)
|
||||
restore_regs(m, &prog, nr_args, stack_size);
|
||||
|
||||
/* call original function */
|
||||
if (emit_call(&prog, orig_call, prog))
|
||||
return -EINVAL;
|
||||
if (emit_call(&prog, orig_call, prog)) {
|
||||
ret = -EINVAL;
|
||||
goto cleanup;
|
||||
}
|
||||
/* remember return value in a stack for bpf prog to access */
|
||||
emit_stx(&prog, BPF_DW, BPF_REG_FP, BPF_REG_0, -8);
|
||||
}
|
||||
|
||||
if (fexit_cnt)
|
||||
if (invoke_bpf(m, &prog, fexit_progs, fexit_cnt, stack_size))
|
||||
return -EINVAL;
|
||||
if (fmod_ret->nr_progs) {
|
||||
/* From Intel 64 and IA-32 Architectures Optimization
|
||||
* Reference Manual, 3.4.1.4 Code Alignment, Assembly/Compiler
|
||||
* Coding Rule 11: All branch targets should be 16-byte
|
||||
* aligned.
|
||||
*/
|
||||
emit_align(&prog, 16);
|
||||
/* Update the branches saved in invoke_bpf_mod_ret with the
|
||||
* aligned address of do_fexit.
|
||||
*/
|
||||
for (i = 0; i < fmod_ret->nr_progs; i++)
|
||||
emit_cond_near_jump(&branches[i], prog, branches[i],
|
||||
X86_JNE);
|
||||
}
|
||||
|
||||
if (fexit->nr_progs)
|
||||
if (invoke_bpf(m, &prog, fexit, stack_size)) {
|
||||
ret = -EINVAL;
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
if (flags & BPF_TRAMP_F_RESTORE_REGS)
|
||||
restore_regs(m, &prog, nr_args, stack_size);
|
||||
|
||||
/* This needs to be done regardless. If there were fmod_ret programs,
|
||||
* the return value is only updated on the stack and still needs to be
|
||||
* restored to R0.
|
||||
*/
|
||||
if (flags & BPF_TRAMP_F_CALL_ORIG)
|
||||
/* restore original return value back into RAX */
|
||||
emit_ldx(&prog, BPF_DW, BPF_REG_0, BPF_REG_FP, -8);
|
||||
@ -1525,45 +1667,15 @@ int arch_prepare_bpf_trampoline(void *image, void *image_end,
|
||||
EMIT4(0x48, 0x83, 0xC4, 8); /* add rsp, 8 */
|
||||
EMIT1(0xC3); /* ret */
|
||||
/* Make sure the trampoline generation logic doesn't overflow */
|
||||
if (WARN_ON_ONCE(prog > (u8 *)image_end - BPF_INSN_SAFETY))
|
||||
return -EFAULT;
|
||||
return prog - (u8 *)image;
|
||||
}
|
||||
|
||||
static int emit_cond_near_jump(u8 **pprog, void *func, void *ip, u8 jmp_cond)
|
||||
{
|
||||
u8 *prog = *pprog;
|
||||
int cnt = 0;
|
||||
s64 offset;
|
||||
|
||||
offset = func - (ip + 2 + 4);
|
||||
if (!is_simm32(offset)) {
|
||||
pr_err("Target %p is out of range\n", func);
|
||||
return -EINVAL;
|
||||
if (WARN_ON_ONCE(prog > (u8 *)image_end - BPF_INSN_SAFETY)) {
|
||||
ret = -EFAULT;
|
||||
goto cleanup;
|
||||
}
|
||||
EMIT2_off32(0x0F, jmp_cond + 0x10, offset);
|
||||
*pprog = prog;
|
||||
return 0;
|
||||
}
|
||||
ret = prog - (u8 *)image;
|
||||
|
||||
static void emit_nops(u8 **pprog, unsigned int len)
|
||||
{
|
||||
unsigned int i, noplen;
|
||||
u8 *prog = *pprog;
|
||||
int cnt = 0;
|
||||
|
||||
while (len > 0) {
|
||||
noplen = len;
|
||||
|
||||
if (noplen > ASM_NOP_MAX)
|
||||
noplen = ASM_NOP_MAX;
|
||||
|
||||
for (i = 0; i < noplen; i++)
|
||||
EMIT1(ideal_nops[noplen][i]);
|
||||
len -= noplen;
|
||||
}
|
||||
|
||||
*pprog = prog;
|
||||
cleanup:
|
||||
kfree(branches);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int emit_fallback_jump(u8 **pprog)
|
||||
@ -1588,7 +1700,7 @@ static int emit_fallback_jump(u8 **pprog)
|
||||
|
||||
static int emit_bpf_dispatcher(u8 **pprog, int a, int b, s64 *progs)
|
||||
{
|
||||
u8 *jg_reloc, *jg_target, *prog = *pprog;
|
||||
u8 *jg_reloc, *prog = *pprog;
|
||||
int pivot, err, jg_bytes = 1, cnt = 0;
|
||||
s64 jg_offset;
|
||||
|
||||
@ -1643,9 +1755,7 @@ static int emit_bpf_dispatcher(u8 **pprog, int a, int b, s64 *progs)
|
||||
* Coding Rule 11: All branch targets should be 16-byte
|
||||
* aligned.
|
||||
*/
|
||||
jg_target = PTR_ALIGN(prog, 16);
|
||||
if (jg_target != prog)
|
||||
emit_nops(&prog, jg_target - prog);
|
||||
emit_align(&prog, 16);
|
||||
jg_offset = prog - jg_reloc;
|
||||
emit_code(jg_reloc - jg_bytes, jg_offset, jg_bytes);
|
||||
|
||||
|
@ -3504,6 +3504,126 @@ out:
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(device_move);
|
||||
|
||||
static int device_attrs_change_owner(struct device *dev, kuid_t kuid,
|
||||
kgid_t kgid)
|
||||
{
|
||||
struct kobject *kobj = &dev->kobj;
|
||||
struct class *class = dev->class;
|
||||
const struct device_type *type = dev->type;
|
||||
int error;
|
||||
|
||||
if (class) {
|
||||
/*
|
||||
* Change the device groups of the device class for @dev to
|
||||
* @kuid/@kgid.
|
||||
*/
|
||||
error = sysfs_groups_change_owner(kobj, class->dev_groups, kuid,
|
||||
kgid);
|
||||
if (error)
|
||||
return error;
|
||||
}
|
||||
|
||||
if (type) {
|
||||
/*
|
||||
* Change the device groups of the device type for @dev to
|
||||
* @kuid/@kgid.
|
||||
*/
|
||||
error = sysfs_groups_change_owner(kobj, type->groups, kuid,
|
||||
kgid);
|
||||
if (error)
|
||||
return error;
|
||||
}
|
||||
|
||||
/* Change the device groups of @dev to @kuid/@kgid. */
|
||||
error = sysfs_groups_change_owner(kobj, dev->groups, kuid, kgid);
|
||||
if (error)
|
||||
return error;
|
||||
|
||||
if (device_supports_offline(dev) && !dev->offline_disabled) {
|
||||
/* Change online device attributes of @dev to @kuid/@kgid. */
|
||||
error = sysfs_file_change_owner(kobj, dev_attr_online.attr.name,
|
||||
kuid, kgid);
|
||||
if (error)
|
||||
return error;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* device_change_owner - change the owner of an existing device.
|
||||
* @dev: device.
|
||||
* @kuid: new owner's kuid
|
||||
* @kgid: new owner's kgid
|
||||
*
|
||||
* This changes the owner of @dev and its corresponding sysfs entries to
|
||||
* @kuid/@kgid. This function closely mirrors how @dev was added via driver
|
||||
* core.
|
||||
*
|
||||
* Returns 0 on success or error code on failure.
|
||||
*/
|
||||
int device_change_owner(struct device *dev, kuid_t kuid, kgid_t kgid)
|
||||
{
|
||||
int error;
|
||||
struct kobject *kobj = &dev->kobj;
|
||||
|
||||
dev = get_device(dev);
|
||||
if (!dev)
|
||||
return -EINVAL;
|
||||
|
||||
/*
|
||||
* Change the kobject and the default attributes and groups of the
|
||||
* ktype associated with it to @kuid/@kgid.
|
||||
*/
|
||||
error = sysfs_change_owner(kobj, kuid, kgid);
|
||||
if (error)
|
||||
goto out;
|
||||
|
||||
/*
|
||||
* Change the uevent file for @dev to the new owner. The uevent file
|
||||
* was created in a separate step when @dev got added and we mirror
|
||||
* that step here.
|
||||
*/
|
||||
error = sysfs_file_change_owner(kobj, dev_attr_uevent.attr.name, kuid,
|
||||
kgid);
|
||||
if (error)
|
||||
goto out;
|
||||
|
||||
/*
|
||||
* Change the device groups, the device groups associated with the
|
||||
* device class, and the groups associated with the device type of @dev
|
||||
* to @kuid/@kgid.
|
||||
*/
|
||||
error = device_attrs_change_owner(dev, kuid, kgid);
|
||||
if (error)
|
||||
goto out;
|
||||
|
||||
error = dpm_sysfs_change_owner(dev, kuid, kgid);
|
||||
if (error)
|
||||
goto out;
|
||||
|
||||
#ifdef CONFIG_BLOCK
|
||||
if (sysfs_deprecated && dev->class == &block_class)
|
||||
goto out;
|
||||
#endif
|
||||
|
||||
/*
|
||||
* Change the owner of the symlink located in the class directory of
|
||||
* the device class associated with @dev which points to the actual
|
||||
* directory entry for @dev to @kuid/@kgid. This ensures that the
|
||||
* symlink shows the same permissions as its target.
|
||||
*/
|
||||
error = sysfs_link_change_owner(&dev->class->p->subsys.kobj, &dev->kobj,
|
||||
dev_name(dev), kuid, kgid);
|
||||
if (error)
|
||||
goto out;
|
||||
|
||||
out:
|
||||
put_device(dev);
|
||||
return error;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(device_change_owner);
|
||||
|
||||
/**
|
||||
* device_shutdown - call ->shutdown() on each device to shutdown.
|
||||
*/
|
||||
|
@ -74,6 +74,7 @@ extern int pm_qos_sysfs_add_flags(struct device *dev);
|
||||
extern void pm_qos_sysfs_remove_flags(struct device *dev);
|
||||
extern int pm_qos_sysfs_add_latency_tolerance(struct device *dev);
|
||||
extern void pm_qos_sysfs_remove_latency_tolerance(struct device *dev);
|
||||
extern int dpm_sysfs_change_owner(struct device *dev, kuid_t kuid, kgid_t kgid);
|
||||
|
||||
#else /* CONFIG_PM */
|
||||
|
||||
@ -88,6 +89,8 @@ static inline void pm_runtime_remove(struct device *dev) {}
|
||||
|
||||
static inline int dpm_sysfs_add(struct device *dev) { return 0; }
|
||||
static inline void dpm_sysfs_remove(struct device *dev) {}
|
||||
static inline int dpm_sysfs_change_owner(struct device *dev, kuid_t kuid,
|
||||
kgid_t kgid) { return 0; }
|
||||
|
||||
#endif
|
||||
|
||||
|
@ -480,6 +480,14 @@ static ssize_t wakeup_last_time_ms_show(struct device *dev,
|
||||
return enabled ? sprintf(buf, "%lld\n", msec) : sprintf(buf, "\n");
|
||||
}
|
||||
|
||||
static inline int dpm_sysfs_wakeup_change_owner(struct device *dev, kuid_t kuid,
|
||||
kgid_t kgid)
|
||||
{
|
||||
if (dev->power.wakeup && dev->power.wakeup->dev)
|
||||
return device_change_owner(dev->power.wakeup->dev, kuid, kgid);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static DEVICE_ATTR_RO(wakeup_last_time_ms);
|
||||
|
||||
#ifdef CONFIG_PM_AUTOSLEEP
|
||||
@ -501,7 +509,13 @@ static ssize_t wakeup_prevent_sleep_time_ms_show(struct device *dev,
|
||||
|
||||
static DEVICE_ATTR_RO(wakeup_prevent_sleep_time_ms);
|
||||
#endif /* CONFIG_PM_AUTOSLEEP */
|
||||
#endif /* CONFIG_PM_SLEEP */
|
||||
#else /* CONFIG_PM_SLEEP */
|
||||
static inline int dpm_sysfs_wakeup_change_owner(struct device *dev, kuid_t kuid,
|
||||
kgid_t kgid)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_PM_ADVANCED_DEBUG
|
||||
static ssize_t runtime_usage_show(struct device *dev,
|
||||
@ -684,6 +698,45 @@ int dpm_sysfs_add(struct device *dev)
|
||||
return rc;
|
||||
}
|
||||
|
||||
int dpm_sysfs_change_owner(struct device *dev, kuid_t kuid, kgid_t kgid)
|
||||
{
|
||||
int rc;
|
||||
|
||||
if (device_pm_not_required(dev))
|
||||
return 0;
|
||||
|
||||
rc = sysfs_group_change_owner(&dev->kobj, &pm_attr_group, kuid, kgid);
|
||||
if (rc)
|
||||
return rc;
|
||||
|
||||
if (pm_runtime_callbacks_present(dev)) {
|
||||
rc = sysfs_group_change_owner(
|
||||
&dev->kobj, &pm_runtime_attr_group, kuid, kgid);
|
||||
if (rc)
|
||||
return rc;
|
||||
}
|
||||
|
||||
if (device_can_wakeup(dev)) {
|
||||
rc = sysfs_group_change_owner(&dev->kobj, &pm_wakeup_attr_group,
|
||||
kuid, kgid);
|
||||
if (rc)
|
||||
return rc;
|
||||
|
||||
rc = dpm_sysfs_wakeup_change_owner(dev, kuid, kgid);
|
||||
if (rc)
|
||||
return rc;
|
||||
}
|
||||
|
||||
if (dev->power.set_latency_tolerance) {
|
||||
rc = sysfs_group_change_owner(
|
||||
&dev->kobj, &pm_qos_latency_tolerance_attr_group, kuid,
|
||||
kgid);
|
||||
if (rc)
|
||||
return rc;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int wakeup_sysfs_add(struct device *dev)
|
||||
{
|
||||
return sysfs_merge_group(&dev->kobj, &pm_wakeup_attr_group);
|
||||
|
@ -211,12 +211,12 @@ config BT_HCIUART_RTL
|
||||
depends on BT_HCIUART
|
||||
depends on BT_HCIUART_SERDEV
|
||||
depends on GPIOLIB
|
||||
depends on ACPI
|
||||
depends on (ACPI || SERIAL_DEV_CTRL_TTYPORT)
|
||||
select BT_HCIUART_3WIRE
|
||||
select BT_RTL
|
||||
help
|
||||
The Realtek protocol support enables Bluetooth HCI over 3-Wire
|
||||
serial port internface for Realtek Bluetooth controllers.
|
||||
serial port interface for Realtek Bluetooth controllers.
|
||||
|
||||
Say Y here to compile support for Realtek protocol.
|
||||
|
||||
|
@ -133,8 +133,8 @@ static int bfusb_send_bulk(struct bfusb_data *data, struct sk_buff *skb)
|
||||
|
||||
err = usb_submit_urb(urb, GFP_ATOMIC);
|
||||
if (err) {
|
||||
BT_ERR("%s bulk tx submit failed urb %p err %d",
|
||||
data->hdev->name, urb, err);
|
||||
bt_dev_err(data->hdev, "bulk tx submit failed urb %p err %d",
|
||||
urb, err);
|
||||
skb_unlink(skb, &data->pending_q);
|
||||
usb_free_urb(urb);
|
||||
} else
|
||||
@ -232,8 +232,8 @@ static int bfusb_rx_submit(struct bfusb_data *data, struct urb *urb)
|
||||
|
||||
err = usb_submit_urb(urb, GFP_ATOMIC);
|
||||
if (err) {
|
||||
BT_ERR("%s bulk rx submit failed urb %p err %d",
|
||||
data->hdev->name, urb, err);
|
||||
bt_dev_err(data->hdev, "bulk rx submit failed urb %p err %d",
|
||||
urb, err);
|
||||
skb_unlink(skb, &data->pending_q);
|
||||
kfree_skb(skb);
|
||||
usb_free_urb(urb);
|
||||
@ -247,7 +247,7 @@ static inline int bfusb_recv_block(struct bfusb_data *data, int hdr, unsigned ch
|
||||
BT_DBG("bfusb %p hdr 0x%02x data %p len %d", data, hdr, buf, len);
|
||||
|
||||
if (hdr & 0x10) {
|
||||
BT_ERR("%s error in block", data->hdev->name);
|
||||
bt_dev_err(data->hdev, "error in block");
|
||||
kfree_skb(data->reassembly);
|
||||
data->reassembly = NULL;
|
||||
return -EIO;
|
||||
@ -259,13 +259,13 @@ static inline int bfusb_recv_block(struct bfusb_data *data, int hdr, unsigned ch
|
||||
int pkt_len = 0;
|
||||
|
||||
if (data->reassembly) {
|
||||
BT_ERR("%s unexpected start block", data->hdev->name);
|
||||
bt_dev_err(data->hdev, "unexpected start block");
|
||||
kfree_skb(data->reassembly);
|
||||
data->reassembly = NULL;
|
||||
}
|
||||
|
||||
if (len < 1) {
|
||||
BT_ERR("%s no packet type found", data->hdev->name);
|
||||
bt_dev_err(data->hdev, "no packet type found");
|
||||
return -EPROTO;
|
||||
}
|
||||
|
||||
@ -277,7 +277,7 @@ static inline int bfusb_recv_block(struct bfusb_data *data, int hdr, unsigned ch
|
||||
struct hci_event_hdr *hdr = (struct hci_event_hdr *) buf;
|
||||
pkt_len = HCI_EVENT_HDR_SIZE + hdr->plen;
|
||||
} else {
|
||||
BT_ERR("%s event block is too short", data->hdev->name);
|
||||
bt_dev_err(data->hdev, "event block is too short");
|
||||
return -EILSEQ;
|
||||
}
|
||||
break;
|
||||
@ -287,7 +287,7 @@ static inline int bfusb_recv_block(struct bfusb_data *data, int hdr, unsigned ch
|
||||
struct hci_acl_hdr *hdr = (struct hci_acl_hdr *) buf;
|
||||
pkt_len = HCI_ACL_HDR_SIZE + __le16_to_cpu(hdr->dlen);
|
||||
} else {
|
||||
BT_ERR("%s data block is too short", data->hdev->name);
|
||||
bt_dev_err(data->hdev, "data block is too short");
|
||||
return -EILSEQ;
|
||||
}
|
||||
break;
|
||||
@ -297,7 +297,7 @@ static inline int bfusb_recv_block(struct bfusb_data *data, int hdr, unsigned ch
|
||||
struct hci_sco_hdr *hdr = (struct hci_sco_hdr *) buf;
|
||||
pkt_len = HCI_SCO_HDR_SIZE + hdr->dlen;
|
||||
} else {
|
||||
BT_ERR("%s audio block is too short", data->hdev->name);
|
||||
bt_dev_err(data->hdev, "audio block is too short");
|
||||
return -EILSEQ;
|
||||
}
|
||||
break;
|
||||
@ -305,7 +305,7 @@ static inline int bfusb_recv_block(struct bfusb_data *data, int hdr, unsigned ch
|
||||
|
||||
skb = bt_skb_alloc(pkt_len, GFP_ATOMIC);
|
||||
if (!skb) {
|
||||
BT_ERR("%s no memory for the packet", data->hdev->name);
|
||||
bt_dev_err(data->hdev, "no memory for the packet");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
@ -314,7 +314,7 @@ static inline int bfusb_recv_block(struct bfusb_data *data, int hdr, unsigned ch
|
||||
data->reassembly = skb;
|
||||
} else {
|
||||
if (!data->reassembly) {
|
||||
BT_ERR("%s unexpected continuation block", data->hdev->name);
|
||||
bt_dev_err(data->hdev, "unexpected continuation block");
|
||||
return -EIO;
|
||||
}
|
||||
}
|
||||
@ -366,8 +366,7 @@ static void bfusb_rx_complete(struct urb *urb)
|
||||
}
|
||||
|
||||
if (count < len) {
|
||||
BT_ERR("%s block extends over URB buffer ranges",
|
||||
data->hdev->name);
|
||||
bt_dev_err(data->hdev, "block extends over URB buffer ranges");
|
||||
}
|
||||
|
||||
if ((hdr & 0xe1) == 0xc1)
|
||||
@ -391,8 +390,8 @@ resubmit:
|
||||
|
||||
err = usb_submit_urb(urb, GFP_ATOMIC);
|
||||
if (err) {
|
||||
BT_ERR("%s bulk resubmit failed urb %p err %d",
|
||||
data->hdev->name, urb, err);
|
||||
bt_dev_err(data->hdev, "bulk resubmit failed urb %p err %d",
|
||||
urb, err);
|
||||
}
|
||||
|
||||
unlock:
|
||||
@ -477,7 +476,7 @@ static int bfusb_send_frame(struct hci_dev *hdev, struct sk_buff *skb)
|
||||
/* Max HCI frame size seems to be 1511 + 1 */
|
||||
nskb = bt_skb_alloc(count + 32, GFP_KERNEL);
|
||||
if (!nskb) {
|
||||
BT_ERR("Can't allocate memory for new packet");
|
||||
bt_dev_err(hdev, "Can't allocate memory for new packet");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
|
@ -376,13 +376,13 @@ struct ibt_cp_reg_access {
|
||||
__le32 addr;
|
||||
__u8 mode;
|
||||
__u8 len;
|
||||
__u8 data[0];
|
||||
__u8 data[];
|
||||
} __packed;
|
||||
|
||||
struct ibt_rp_reg_access {
|
||||
__u8 status;
|
||||
__le32 addr;
|
||||
__u8 data[0];
|
||||
__u8 data[];
|
||||
} __packed;
|
||||
|
||||
static int regmap_ibt_read(void *context, const void *addr, size_t reg_size,
|
||||
|
@ -139,7 +139,7 @@ int qca_send_pre_shutdown_cmd(struct hci_dev *hdev)
|
||||
EXPORT_SYMBOL_GPL(qca_send_pre_shutdown_cmd);
|
||||
|
||||
static void qca_tlv_check_data(struct qca_fw_config *config,
|
||||
const struct firmware *fw)
|
||||
const struct firmware *fw, enum qca_btsoc_type soc_type)
|
||||
{
|
||||
const u8 *data;
|
||||
u32 type_len;
|
||||
@ -148,6 +148,7 @@ static void qca_tlv_check_data(struct qca_fw_config *config,
|
||||
struct tlv_type_hdr *tlv;
|
||||
struct tlv_type_patch *tlv_patch;
|
||||
struct tlv_type_nvm *tlv_nvm;
|
||||
uint8_t nvm_baud_rate = config->user_baud_rate;
|
||||
|
||||
tlv = (struct tlv_type_hdr *)fw->data;
|
||||
|
||||
@ -216,7 +217,10 @@ static void qca_tlv_check_data(struct qca_fw_config *config,
|
||||
tlv_nvm->data[0] |= 0x80;
|
||||
|
||||
/* UART Baud Rate */
|
||||
tlv_nvm->data[2] = config->user_baud_rate;
|
||||
if (soc_type == QCA_WCN3991)
|
||||
tlv_nvm->data[1] = nvm_baud_rate;
|
||||
else
|
||||
tlv_nvm->data[2] = nvm_baud_rate;
|
||||
|
||||
break;
|
||||
|
||||
@ -354,7 +358,7 @@ static int qca_download_firmware(struct hci_dev *hdev,
|
||||
return ret;
|
||||
}
|
||||
|
||||
qca_tlv_check_data(config, fw);
|
||||
qca_tlv_check_data(config, fw, soc_type);
|
||||
|
||||
segment = fw->data;
|
||||
remain = fw->size;
|
||||
|
@ -79,7 +79,7 @@ struct qca_fw_config {
|
||||
struct edl_event_hdr {
|
||||
__u8 cresp;
|
||||
__u8 rtype;
|
||||
__u8 data[0];
|
||||
__u8 data[];
|
||||
} __packed;
|
||||
|
||||
struct qca_btsoc_version {
|
||||
@ -112,12 +112,12 @@ struct tlv_type_nvm {
|
||||
__le16 tag_len;
|
||||
__le32 reserve1;
|
||||
__le32 reserve2;
|
||||
__u8 data[0];
|
||||
__u8 data[];
|
||||
} __packed;
|
||||
|
||||
struct tlv_type_hdr {
|
||||
__le32 type_len;
|
||||
__u8 data[0];
|
||||
__u8 data[];
|
||||
} __packed;
|
||||
|
||||
enum qca_btsoc_type {
|
||||
|
@ -136,6 +136,18 @@ static const struct id_table ic_id_table[] = {
|
||||
.fw_name = "rtl_bt/rtl8761a_fw.bin",
|
||||
.cfg_name = "rtl_bt/rtl8761a_config" },
|
||||
|
||||
/* 8822C with UART interface */
|
||||
{ .match_flags = IC_MATCH_FL_LMPSUBV | IC_MATCH_FL_HCIREV |
|
||||
IC_MATCH_FL_HCIBUS,
|
||||
.lmp_subver = RTL_ROM_LMP_8822B,
|
||||
.hci_rev = 0x000c,
|
||||
.hci_ver = 0x0a,
|
||||
.hci_bus = HCI_UART,
|
||||
.config_needed = true,
|
||||
.has_rom_version = true,
|
||||
.fw_name = "rtl_bt/rtl8822cs_fw.bin",
|
||||
.cfg_name = "rtl_bt/rtl8822cs_config" },
|
||||
|
||||
/* 8822C with USB interface */
|
||||
{ IC_INFO(RTL_ROM_LMP_8822B, 0xc),
|
||||
.config_needed = false,
|
||||
|
@ -38,13 +38,13 @@ struct rtl_epatch_header {
|
||||
struct rtl_vendor_config_entry {
|
||||
__le16 offset;
|
||||
__u8 len;
|
||||
__u8 data[0];
|
||||
__u8 data[];
|
||||
} __packed;
|
||||
|
||||
struct rtl_vendor_config {
|
||||
__le32 signature;
|
||||
__le16 total_len;
|
||||
struct rtl_vendor_config_entry entry[0];
|
||||
struct rtl_vendor_config_entry entry[];
|
||||
} __packed;
|
||||
|
||||
#if IS_ENABLED(CONFIG_BT_RTL)
|
||||
|
@ -57,6 +57,7 @@ static struct usb_driver btusb_driver;
|
||||
#define BTUSB_IFNUM_2 0x80000
|
||||
#define BTUSB_CW6622 0x100000
|
||||
#define BTUSB_MEDIATEK 0x200000
|
||||
#define BTUSB_WIDEBAND_SPEECH 0x400000
|
||||
|
||||
static const struct usb_device_id btusb_table[] = {
|
||||
/* Generic Bluetooth USB device */
|
||||
@ -333,15 +334,21 @@ static const struct usb_device_id blacklist_table[] = {
|
||||
{ USB_DEVICE(0x1286, 0x204e), .driver_info = BTUSB_MARVELL },
|
||||
|
||||
/* Intel Bluetooth devices */
|
||||
{ USB_DEVICE(0x8087, 0x0025), .driver_info = BTUSB_INTEL_NEW },
|
||||
{ USB_DEVICE(0x8087, 0x0026), .driver_info = BTUSB_INTEL_NEW },
|
||||
{ USB_DEVICE(0x8087, 0x0029), .driver_info = BTUSB_INTEL_NEW },
|
||||
{ USB_DEVICE(0x8087, 0x0025), .driver_info = BTUSB_INTEL_NEW |
|
||||
BTUSB_WIDEBAND_SPEECH },
|
||||
{ USB_DEVICE(0x8087, 0x0026), .driver_info = BTUSB_INTEL_NEW |
|
||||
BTUSB_WIDEBAND_SPEECH },
|
||||
{ USB_DEVICE(0x8087, 0x0029), .driver_info = BTUSB_INTEL_NEW |
|
||||
BTUSB_WIDEBAND_SPEECH },
|
||||
{ USB_DEVICE(0x8087, 0x07da), .driver_info = BTUSB_CSR },
|
||||
{ USB_DEVICE(0x8087, 0x07dc), .driver_info = BTUSB_INTEL },
|
||||
{ USB_DEVICE(0x8087, 0x0a2a), .driver_info = BTUSB_INTEL },
|
||||
{ USB_DEVICE(0x8087, 0x0a2b), .driver_info = BTUSB_INTEL_NEW },
|
||||
{ USB_DEVICE(0x8087, 0x0aa7), .driver_info = BTUSB_INTEL },
|
||||
{ USB_DEVICE(0x8087, 0x0aaa), .driver_info = BTUSB_INTEL_NEW },
|
||||
{ USB_DEVICE(0x8087, 0x0a2b), .driver_info = BTUSB_INTEL_NEW |
|
||||
BTUSB_WIDEBAND_SPEECH },
|
||||
{ USB_DEVICE(0x8087, 0x0aa7), .driver_info = BTUSB_INTEL |
|
||||
BTUSB_WIDEBAND_SPEECH },
|
||||
{ USB_DEVICE(0x8087, 0x0aaa), .driver_info = BTUSB_INTEL_NEW |
|
||||
BTUSB_WIDEBAND_SPEECH },
|
||||
|
||||
/* Other Intel Bluetooth devices */
|
||||
{ USB_VENDOR_AND_INTERFACE_INFO(0x8087, 0xe0, 0x01, 0x01),
|
||||
@ -387,6 +394,7 @@ static const struct usb_device_id blacklist_table[] = {
|
||||
|
||||
/* Additional Realtek 8822CE Bluetooth devices */
|
||||
{ USB_DEVICE(0x04ca, 0x4005), .driver_info = BTUSB_REALTEK },
|
||||
{ USB_DEVICE(0x13d3, 0x3548), .driver_info = BTUSB_REALTEK },
|
||||
|
||||
/* Silicon Wave based devices */
|
||||
{ USB_DEVICE(0x0c10, 0x0000), .driver_info = BTUSB_SWAVE },
|
||||
@ -1930,7 +1938,14 @@ static int btusb_setup_intel(struct hci_dev *hdev)
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
bt_dev_info(hdev, "Intel firmware patch completed and activated");
|
||||
/* Need build number for downloaded fw patches in
|
||||
* every power-on boot
|
||||
*/
|
||||
err = btintel_read_version(hdev, &ver);
|
||||
if (err)
|
||||
return err;
|
||||
bt_dev_info(hdev, "Intel BT fw patch 0x%02x completed & activated",
|
||||
ver.fw_patch_num);
|
||||
|
||||
goto complete;
|
||||
|
||||
@ -3859,6 +3874,9 @@ static int btusb_probe(struct usb_interface *intf,
|
||||
if (id->driver_info & BTUSB_BROKEN_ISOC)
|
||||
data->isoc = NULL;
|
||||
|
||||
if (id->driver_info & BTUSB_WIDEBAND_SPEECH)
|
||||
set_bit(HCI_QUIRK_WIDEBAND_SPEECH_SUPPORTED, &hdev->quirks);
|
||||
|
||||
if (id->driver_info & BTUSB_DIGIANSWER) {
|
||||
data->cmdreq_type = USB_TYPE_VENDOR;
|
||||
set_bit(HCI_QUIRK_RESET_ON_CLOSE, &hdev->quirks);
|
||||
|
@ -27,7 +27,7 @@ struct ag6xx_data {
|
||||
struct pbn_entry {
|
||||
__le32 addr;
|
||||
__le32 plen;
|
||||
__u8 data[0];
|
||||
__u8 data[];
|
||||
} __packed;
|
||||
|
||||
static int ag6xx_open(struct hci_uart *hu)
|
||||
|
@ -71,8 +71,6 @@ static int h4_close(struct hci_uart *hu)
|
||||
{
|
||||
struct h4_struct *h4 = hu->priv;
|
||||
|
||||
hu->priv = NULL;
|
||||
|
||||
BT_DBG("hu %p", hu);
|
||||
|
||||
skb_queue_purge(&h4->txq);
|
||||
@ -85,7 +83,7 @@ static int h4_close(struct hci_uart *hu)
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Enqueue frame for transmittion (padding, crc, etc) */
|
||||
/* Enqueue frame for transmission (padding, crc, etc) */
|
||||
static int h4_enqueue(struct hci_uart *hu, struct sk_buff *skb)
|
||||
{
|
||||
struct h4_struct *h4 = hu->priv;
|
||||
|
@ -11,6 +11,7 @@
|
||||
#include <linux/gpio/consumer.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/mod_devicetable.h>
|
||||
#include <linux/of_device.h>
|
||||
#include <linux/serdev.h>
|
||||
#include <linux/skbuff.h>
|
||||
|
||||
@ -177,7 +178,7 @@ static void h5_peer_reset(struct hci_uart *hu)
|
||||
{
|
||||
struct h5 *h5 = hu->priv;
|
||||
|
||||
BT_ERR("Peer device has reset");
|
||||
bt_dev_err(hu->hdev, "Peer device has reset");
|
||||
|
||||
h5->state = H5_UNINITIALIZED;
|
||||
|
||||
@ -437,21 +438,21 @@ static int h5_rx_3wire_hdr(struct hci_uart *hu, unsigned char c)
|
||||
H5_HDR_LEN(hdr));
|
||||
|
||||
if (((hdr[0] + hdr[1] + hdr[2] + hdr[3]) & 0xff) != 0xff) {
|
||||
BT_ERR("Invalid header checksum");
|
||||
bt_dev_err(hu->hdev, "Invalid header checksum");
|
||||
h5_reset_rx(h5);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (H5_HDR_RELIABLE(hdr) && H5_HDR_SEQ(hdr) != h5->tx_ack) {
|
||||
BT_ERR("Out-of-order packet arrived (%u != %u)",
|
||||
H5_HDR_SEQ(hdr), h5->tx_ack);
|
||||
bt_dev_err(hu->hdev, "Out-of-order packet arrived (%u != %u)",
|
||||
H5_HDR_SEQ(hdr), h5->tx_ack);
|
||||
h5_reset_rx(h5);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (h5->state != H5_ACTIVE &&
|
||||
H5_HDR_PKT_TYPE(hdr) != HCI_3WIRE_LINK_PKT) {
|
||||
BT_ERR("Non-link packet received in non-active state");
|
||||
bt_dev_err(hu->hdev, "Non-link packet received in non-active state");
|
||||
h5_reset_rx(h5);
|
||||
return 0;
|
||||
}
|
||||
@ -474,7 +475,7 @@ static int h5_rx_pkt_start(struct hci_uart *hu, unsigned char c)
|
||||
|
||||
h5->rx_skb = bt_skb_alloc(H5_MAX_LEN, GFP_ATOMIC);
|
||||
if (!h5->rx_skb) {
|
||||
BT_ERR("Can't allocate mem for new packet");
|
||||
bt_dev_err(hu->hdev, "Can't allocate mem for new packet");
|
||||
h5_reset_rx(h5);
|
||||
return -ENOMEM;
|
||||
}
|
||||
@ -550,7 +551,7 @@ static int h5_recv(struct hci_uart *hu, const void *data, int count)
|
||||
|
||||
if (h5->rx_pending > 0) {
|
||||
if (*ptr == SLIP_DELIMITER) {
|
||||
BT_ERR("Too short H5 packet");
|
||||
bt_dev_err(hu->hdev, "Too short H5 packet");
|
||||
h5_reset_rx(h5);
|
||||
continue;
|
||||
}
|
||||
@ -577,13 +578,13 @@ static int h5_enqueue(struct hci_uart *hu, struct sk_buff *skb)
|
||||
struct h5 *h5 = hu->priv;
|
||||
|
||||
if (skb->len > 0xfff) {
|
||||
BT_ERR("Packet too long (%u bytes)", skb->len);
|
||||
bt_dev_err(hu->hdev, "Packet too long (%u bytes)", skb->len);
|
||||
kfree_skb(skb);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (h5->state != H5_ACTIVE) {
|
||||
BT_ERR("Ignoring HCI data in non-active state");
|
||||
bt_dev_err(hu->hdev, "Ignoring HCI data in non-active state");
|
||||
kfree_skb(skb);
|
||||
return 0;
|
||||
}
|
||||
@ -600,7 +601,7 @@ static int h5_enqueue(struct hci_uart *hu, struct sk_buff *skb)
|
||||
break;
|
||||
|
||||
default:
|
||||
BT_ERR("Unknown packet type %u", hci_skb_pkt_type(skb));
|
||||
bt_dev_err(hu->hdev, "Unknown packet type %u", hci_skb_pkt_type(skb));
|
||||
kfree_skb(skb);
|
||||
break;
|
||||
}
|
||||
@ -656,7 +657,7 @@ static struct sk_buff *h5_prepare_pkt(struct hci_uart *hu, u8 pkt_type,
|
||||
int i;
|
||||
|
||||
if (!valid_packet_type(pkt_type)) {
|
||||
BT_ERR("Unknown packet type %u", pkt_type);
|
||||
bt_dev_err(hu->hdev, "Unknown packet type %u", pkt_type);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
@ -733,7 +734,7 @@ static struct sk_buff *h5_dequeue(struct hci_uart *hu)
|
||||
}
|
||||
|
||||
skb_queue_head(&h5->unrel, skb);
|
||||
BT_ERR("Could not dequeue pkt because alloc_skb failed");
|
||||
bt_dev_err(hu->hdev, "Could not dequeue pkt because alloc_skb failed");
|
||||
}
|
||||
|
||||
spin_lock_irqsave_nested(&h5->unack.lock, flags, SINGLE_DEPTH_NESTING);
|
||||
@ -753,7 +754,7 @@ static struct sk_buff *h5_dequeue(struct hci_uart *hu)
|
||||
}
|
||||
|
||||
skb_queue_head(&h5->rel, skb);
|
||||
BT_ERR("Could not dequeue pkt because alloc_skb failed");
|
||||
bt_dev_err(hu->hdev, "Could not dequeue pkt because alloc_skb failed");
|
||||
}
|
||||
|
||||
unlock:
|
||||
@ -785,7 +786,6 @@ static const struct hci_uart_proto h5p = {
|
||||
|
||||
static int h5_serdev_probe(struct serdev_device *serdev)
|
||||
{
|
||||
const struct acpi_device_id *match;
|
||||
struct device *dev = &serdev->dev;
|
||||
struct h5 *h5;
|
||||
|
||||
@ -800,6 +800,8 @@ static int h5_serdev_probe(struct serdev_device *serdev)
|
||||
serdev_device_set_drvdata(serdev, h5);
|
||||
|
||||
if (has_acpi_companion(dev)) {
|
||||
const struct acpi_device_id *match;
|
||||
|
||||
match = acpi_match_device(dev->driver->acpi_match_table, dev);
|
||||
if (!match)
|
||||
return -ENODEV;
|
||||
@ -810,8 +812,17 @@ static int h5_serdev_probe(struct serdev_device *serdev)
|
||||
if (h5->vnd->acpi_gpio_map)
|
||||
devm_acpi_dev_add_driver_gpios(dev,
|
||||
h5->vnd->acpi_gpio_map);
|
||||
} else {
|
||||
const void *data;
|
||||
|
||||
data = of_device_get_match_data(dev);
|
||||
if (!data)
|
||||
return -ENODEV;
|
||||
|
||||
h5->vnd = (const struct h5_vnd *)data;
|
||||
}
|
||||
|
||||
|
||||
h5->enable_gpio = devm_gpiod_get_optional(dev, "enable", GPIOD_OUT_LOW);
|
||||
if (IS_ERR(h5->enable_gpio))
|
||||
return PTR_ERR(h5->enable_gpio);
|
||||
@ -1003,6 +1014,15 @@ static const struct dev_pm_ops h5_serdev_pm_ops = {
|
||||
SET_SYSTEM_SLEEP_PM_OPS(h5_serdev_suspend, h5_serdev_resume)
|
||||
};
|
||||
|
||||
static const struct of_device_id rtl_bluetooth_of_match[] = {
|
||||
#ifdef CONFIG_BT_HCIUART_RTL
|
||||
{ .compatible = "realtek,rtl8822cs-bt",
|
||||
.data = (const void *)&rtl_vnd },
|
||||
#endif
|
||||
{ },
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, rtl_bluetooth_of_match);
|
||||
|
||||
static struct serdev_device_driver h5_serdev_driver = {
|
||||
.probe = h5_serdev_probe,
|
||||
.remove = h5_serdev_remove,
|
||||
@ -1010,6 +1030,7 @@ static struct serdev_device_driver h5_serdev_driver = {
|
||||
.name = "hci_uart_h5",
|
||||
.acpi_match_table = ACPI_PTR(h5_acpi_match),
|
||||
.pm = &h5_serdev_pm_ops,
|
||||
.of_match_table = rtl_bluetooth_of_match,
|
||||
},
|
||||
};
|
||||
|
||||
|
@ -49,7 +49,7 @@
|
||||
struct hci_lpm_pkt {
|
||||
__u8 opcode;
|
||||
__u8 dlen;
|
||||
__u8 data[0];
|
||||
__u8 data[];
|
||||
} __packed;
|
||||
|
||||
struct intel_device {
|
||||
|
@ -29,6 +29,7 @@
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/regulator/consumer.h>
|
||||
#include <linux/serdev.h>
|
||||
#include <linux/mutex.h>
|
||||
#include <asm/unaligned.h>
|
||||
|
||||
#include <net/bluetooth/bluetooth.h>
|
||||
@ -69,7 +70,8 @@ enum qca_flags {
|
||||
QCA_IBS_ENABLED,
|
||||
QCA_DROP_VENDOR_EVENT,
|
||||
QCA_SUSPENDING,
|
||||
QCA_MEMDUMP_COLLECTION
|
||||
QCA_MEMDUMP_COLLECTION,
|
||||
QCA_HW_ERROR_EVENT
|
||||
};
|
||||
|
||||
|
||||
@ -138,18 +140,19 @@ struct qca_data {
|
||||
u32 tx_idle_delay;
|
||||
struct timer_list wake_retrans_timer;
|
||||
u32 wake_retrans;
|
||||
struct timer_list memdump_timer;
|
||||
struct workqueue_struct *workqueue;
|
||||
struct work_struct ws_awake_rx;
|
||||
struct work_struct ws_awake_device;
|
||||
struct work_struct ws_rx_vote_off;
|
||||
struct work_struct ws_tx_vote_off;
|
||||
struct work_struct ctrl_memdump_evt;
|
||||
struct delayed_work ctrl_memdump_timeout;
|
||||
struct qca_memdump_data *qca_memdump;
|
||||
unsigned long flags;
|
||||
struct completion drop_ev_comp;
|
||||
wait_queue_head_t suspend_wait_q;
|
||||
enum qca_memdump_states memdump_state;
|
||||
struct mutex hci_memdump_lock;
|
||||
|
||||
/* For debugging purpose */
|
||||
u64 ibs_sent_wacks;
|
||||
@ -522,23 +525,28 @@ static void hci_ibs_wake_retrans_timeout(struct timer_list *t)
|
||||
hci_uart_tx_wakeup(hu);
|
||||
}
|
||||
|
||||
static void hci_memdump_timeout(struct timer_list *t)
|
||||
{
|
||||
struct qca_data *qca = from_timer(qca, t, tx_idle_timer);
|
||||
struct hci_uart *hu = qca->hu;
|
||||
struct qca_memdump_data *qca_memdump = qca->qca_memdump;
|
||||
char *memdump_buf = qca_memdump->memdump_buf_tail;
|
||||
|
||||
bt_dev_err(hu->hdev, "clearing allocated memory due to memdump timeout");
|
||||
/* Inject hw error event to reset the device and driver. */
|
||||
hci_reset_dev(hu->hdev);
|
||||
vfree(memdump_buf);
|
||||
kfree(qca_memdump);
|
||||
qca->memdump_state = QCA_MEMDUMP_TIMEOUT;
|
||||
del_timer(&qca->memdump_timer);
|
||||
cancel_work_sync(&qca->ctrl_memdump_evt);
|
||||
static void qca_controller_memdump_timeout(struct work_struct *work)
|
||||
{
|
||||
struct qca_data *qca = container_of(work, struct qca_data,
|
||||
ctrl_memdump_timeout.work);
|
||||
struct hci_uart *hu = qca->hu;
|
||||
|
||||
mutex_lock(&qca->hci_memdump_lock);
|
||||
if (test_bit(QCA_MEMDUMP_COLLECTION, &qca->flags)) {
|
||||
qca->memdump_state = QCA_MEMDUMP_TIMEOUT;
|
||||
if (!test_bit(QCA_HW_ERROR_EVENT, &qca->flags)) {
|
||||
/* Inject hw error event to reset the device
|
||||
* and driver.
|
||||
*/
|
||||
hci_reset_dev(hu->hdev);
|
||||
}
|
||||
}
|
||||
|
||||
mutex_unlock(&qca->hci_memdump_lock);
|
||||
}
|
||||
|
||||
|
||||
/* Initialize protocol */
|
||||
static int qca_open(struct hci_uart *hu)
|
||||
{
|
||||
@ -558,6 +566,7 @@ static int qca_open(struct hci_uart *hu)
|
||||
skb_queue_head_init(&qca->tx_wait_q);
|
||||
skb_queue_head_init(&qca->rx_memdump_q);
|
||||
spin_lock_init(&qca->hci_ibs_lock);
|
||||
mutex_init(&qca->hci_memdump_lock);
|
||||
qca->workqueue = alloc_ordered_workqueue("qca_wq", 0);
|
||||
if (!qca->workqueue) {
|
||||
BT_ERR("QCA Workqueue not initialized properly");
|
||||
@ -570,6 +579,8 @@ static int qca_open(struct hci_uart *hu)
|
||||
INIT_WORK(&qca->ws_rx_vote_off, qca_wq_serial_rx_clock_vote_off);
|
||||
INIT_WORK(&qca->ws_tx_vote_off, qca_wq_serial_tx_clock_vote_off);
|
||||
INIT_WORK(&qca->ctrl_memdump_evt, qca_controller_memdump);
|
||||
INIT_DELAYED_WORK(&qca->ctrl_memdump_timeout,
|
||||
qca_controller_memdump_timeout);
|
||||
init_waitqueue_head(&qca->suspend_wait_q);
|
||||
|
||||
qca->hu = hu;
|
||||
@ -596,7 +607,6 @@ static int qca_open(struct hci_uart *hu)
|
||||
|
||||
timer_setup(&qca->tx_idle_timer, hci_ibs_tx_idle_timeout, 0);
|
||||
qca->tx_idle_delay = IBS_HOST_TX_IDLE_TIMEOUT_MS;
|
||||
timer_setup(&qca->memdump_timer, hci_memdump_timeout, 0);
|
||||
|
||||
BT_DBG("HCI_UART_QCA open, tx_idle_delay=%u, wake_retrans=%u",
|
||||
qca->tx_idle_delay, qca->wake_retrans);
|
||||
@ -677,7 +687,6 @@ static int qca_close(struct hci_uart *hu)
|
||||
skb_queue_purge(&qca->rx_memdump_q);
|
||||
del_timer(&qca->tx_idle_timer);
|
||||
del_timer(&qca->wake_retrans_timer);
|
||||
del_timer(&qca->memdump_timer);
|
||||
destroy_workqueue(qca->workqueue);
|
||||
qca->hu = NULL;
|
||||
|
||||
@ -963,11 +972,20 @@ static void qca_controller_memdump(struct work_struct *work)
|
||||
|
||||
while ((skb = skb_dequeue(&qca->rx_memdump_q))) {
|
||||
|
||||
mutex_lock(&qca->hci_memdump_lock);
|
||||
/* Skip processing the received packets if timeout detected. */
|
||||
if (qca->memdump_state == QCA_MEMDUMP_TIMEOUT) {
|
||||
mutex_unlock(&qca->hci_memdump_lock);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!qca_memdump) {
|
||||
qca_memdump = kzalloc(sizeof(struct qca_memdump_data),
|
||||
GFP_ATOMIC);
|
||||
if (!qca_memdump)
|
||||
if (!qca_memdump) {
|
||||
mutex_unlock(&qca->hci_memdump_lock);
|
||||
return;
|
||||
}
|
||||
|
||||
qca->qca_memdump = qca_memdump;
|
||||
}
|
||||
@ -992,13 +1010,15 @@ static void qca_controller_memdump(struct work_struct *work)
|
||||
if (!(dump_size)) {
|
||||
bt_dev_err(hu->hdev, "Rx invalid memdump size");
|
||||
kfree_skb(skb);
|
||||
mutex_unlock(&qca->hci_memdump_lock);
|
||||
return;
|
||||
}
|
||||
|
||||
bt_dev_info(hu->hdev, "QCA collecting dump of size:%u",
|
||||
dump_size);
|
||||
mod_timer(&qca->memdump_timer, (jiffies +
|
||||
msecs_to_jiffies(MEMDUMP_TIMEOUT_MS)));
|
||||
queue_delayed_work(qca->workqueue,
|
||||
&qca->ctrl_memdump_timeout,
|
||||
msecs_to_jiffies(MEMDUMP_TIMEOUT_MS));
|
||||
|
||||
skb_pull(skb, sizeof(dump_size));
|
||||
memdump_buf = vmalloc(dump_size);
|
||||
@ -1016,6 +1036,7 @@ static void qca_controller_memdump(struct work_struct *work)
|
||||
kfree(qca_memdump);
|
||||
kfree_skb(skb);
|
||||
qca->qca_memdump = NULL;
|
||||
mutex_unlock(&qca->hci_memdump_lock);
|
||||
return;
|
||||
}
|
||||
|
||||
@ -1046,16 +1067,20 @@ static void qca_controller_memdump(struct work_struct *work)
|
||||
memdump_buf = qca_memdump->memdump_buf_head;
|
||||
dev_coredumpv(&hu->serdev->dev, memdump_buf,
|
||||
qca_memdump->received_dump, GFP_KERNEL);
|
||||
del_timer(&qca->memdump_timer);
|
||||
cancel_delayed_work(&qca->ctrl_memdump_timeout);
|
||||
kfree(qca->qca_memdump);
|
||||
qca->qca_memdump = NULL;
|
||||
qca->memdump_state = QCA_MEMDUMP_COLLECTED;
|
||||
clear_bit(QCA_MEMDUMP_COLLECTION, &qca->flags);
|
||||
}
|
||||
|
||||
mutex_unlock(&qca->hci_memdump_lock);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
int qca_controller_memdump_event(struct hci_dev *hdev, struct sk_buff *skb)
|
||||
static int qca_controller_memdump_event(struct hci_dev *hdev,
|
||||
struct sk_buff *skb)
|
||||
{
|
||||
struct hci_uart *hu = hci_get_drvdata(hdev);
|
||||
struct qca_data *qca = hu->priv;
|
||||
@ -1406,30 +1431,21 @@ static void qca_wait_for_dump_collection(struct hci_dev *hdev)
|
||||
{
|
||||
struct hci_uart *hu = hci_get_drvdata(hdev);
|
||||
struct qca_data *qca = hu->priv;
|
||||
struct qca_memdump_data *qca_memdump = qca->qca_memdump;
|
||||
char *memdump_buf = NULL;
|
||||
|
||||
wait_on_bit_timeout(&qca->flags, QCA_MEMDUMP_COLLECTION,
|
||||
TASK_UNINTERRUPTIBLE, MEMDUMP_TIMEOUT_MS);
|
||||
|
||||
clear_bit(QCA_MEMDUMP_COLLECTION, &qca->flags);
|
||||
if (qca->memdump_state == QCA_MEMDUMP_IDLE) {
|
||||
bt_dev_err(hu->hdev, "Clearing the buffers due to timeout");
|
||||
if (qca_memdump)
|
||||
memdump_buf = qca_memdump->memdump_buf_tail;
|
||||
vfree(memdump_buf);
|
||||
kfree(qca_memdump);
|
||||
qca->memdump_state = QCA_MEMDUMP_TIMEOUT;
|
||||
del_timer(&qca->memdump_timer);
|
||||
cancel_work_sync(&qca->ctrl_memdump_evt);
|
||||
}
|
||||
}
|
||||
|
||||
static void qca_hw_error(struct hci_dev *hdev, u8 code)
|
||||
{
|
||||
struct hci_uart *hu = hci_get_drvdata(hdev);
|
||||
struct qca_data *qca = hu->priv;
|
||||
struct qca_memdump_data *qca_memdump = qca->qca_memdump;
|
||||
char *memdump_buf = NULL;
|
||||
|
||||
set_bit(QCA_HW_ERROR_EVENT, &qca->flags);
|
||||
bt_dev_info(hdev, "mem_dump_status: %d", qca->memdump_state);
|
||||
|
||||
if (qca->memdump_state == QCA_MEMDUMP_IDLE) {
|
||||
@ -1449,6 +1465,23 @@ static void qca_hw_error(struct hci_dev *hdev, u8 code)
|
||||
bt_dev_info(hdev, "waiting for dump to complete");
|
||||
qca_wait_for_dump_collection(hdev);
|
||||
}
|
||||
|
||||
if (qca->memdump_state != QCA_MEMDUMP_COLLECTED) {
|
||||
bt_dev_err(hu->hdev, "clearing allocated memory due to memdump timeout");
|
||||
mutex_lock(&qca->hci_memdump_lock);
|
||||
if (qca_memdump)
|
||||
memdump_buf = qca_memdump->memdump_buf_head;
|
||||
vfree(memdump_buf);
|
||||
kfree(qca_memdump);
|
||||
qca->qca_memdump = NULL;
|
||||
qca->memdump_state = QCA_MEMDUMP_TIMEOUT;
|
||||
cancel_delayed_work(&qca->ctrl_memdump_timeout);
|
||||
skb_queue_purge(&qca->rx_memdump_q);
|
||||
mutex_unlock(&qca->hci_memdump_lock);
|
||||
cancel_work_sync(&qca->ctrl_memdump_evt);
|
||||
}
|
||||
|
||||
clear_bit(QCA_HW_ERROR_EVENT, &qca->flags);
|
||||
}
|
||||
|
||||
static void qca_cmd_timeout(struct hci_dev *hdev)
|
||||
@ -1529,9 +1562,11 @@ static int qca_power_on(struct hci_dev *hdev)
|
||||
ret = qca_wcn3990_init(hu);
|
||||
} else {
|
||||
qcadev = serdev_device_get_drvdata(hu->serdev);
|
||||
gpiod_set_value_cansleep(qcadev->bt_en, 1);
|
||||
/* Controller needs time to bootup. */
|
||||
msleep(150);
|
||||
if (qcadev->bt_en) {
|
||||
gpiod_set_value_cansleep(qcadev->bt_en, 1);
|
||||
/* Controller needs time to bootup. */
|
||||
msleep(150);
|
||||
}
|
||||
}
|
||||
|
||||
return ret;
|
||||
@ -1717,7 +1752,7 @@ static void qca_power_shutdown(struct hci_uart *hu)
|
||||
host_set_baudrate(hu, 2400);
|
||||
qca_send_power_pulse(hu, false);
|
||||
qca_regulator_disable(qcadev);
|
||||
} else {
|
||||
} else if (qcadev->bt_en) {
|
||||
gpiod_set_value_cansleep(qcadev->bt_en, 0);
|
||||
}
|
||||
}
|
||||
@ -1726,9 +1761,11 @@ static int qca_power_off(struct hci_dev *hdev)
|
||||
{
|
||||
struct hci_uart *hu = hci_get_drvdata(hdev);
|
||||
struct qca_data *qca = hu->priv;
|
||||
enum qca_btsoc_type soc_type = qca_soc_type(hu);
|
||||
|
||||
/* Stop sending shutdown command if soc crashes. */
|
||||
if (qca->memdump_state == QCA_MEMDUMP_IDLE) {
|
||||
if (qca_is_wcn399x(soc_type)
|
||||
&& qca->memdump_state == QCA_MEMDUMP_IDLE) {
|
||||
qca_send_pre_shutdown_cmd(hdev);
|
||||
usleep_range(8000, 10000);
|
||||
}
|
||||
@ -1755,7 +1792,11 @@ static int qca_regulator_enable(struct qca_serdev *qcadev)
|
||||
|
||||
power->vregs_on = true;
|
||||
|
||||
return 0;
|
||||
ret = clk_prepare_enable(qcadev->susclk);
|
||||
if (ret)
|
||||
qca_regulator_disable(qcadev);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void qca_regulator_disable(struct qca_serdev *qcadev)
|
||||
@ -1773,6 +1814,8 @@ static void qca_regulator_disable(struct qca_serdev *qcadev)
|
||||
|
||||
regulator_bulk_disable(power->num_vregs, power->vreg_bulk);
|
||||
power->vregs_on = false;
|
||||
|
||||
clk_disable_unprepare(qcadev->susclk);
|
||||
}
|
||||
|
||||
static int qca_init_regulators(struct qca_power *qca,
|
||||
@ -1811,6 +1854,7 @@ static int qca_serdev_probe(struct serdev_device *serdev)
|
||||
struct hci_dev *hdev;
|
||||
const struct qca_vreg_data *data;
|
||||
int err;
|
||||
bool power_ctrl_enabled = true;
|
||||
|
||||
qcadev = devm_kzalloc(&serdev->dev, sizeof(*qcadev), GFP_KERNEL);
|
||||
if (!qcadev)
|
||||
@ -1839,6 +1883,12 @@ static int qca_serdev_probe(struct serdev_device *serdev)
|
||||
|
||||
qcadev->bt_power->vregs_on = false;
|
||||
|
||||
qcadev->susclk = devm_clk_get_optional(&serdev->dev, NULL);
|
||||
if (IS_ERR(qcadev->susclk)) {
|
||||
dev_err(&serdev->dev, "failed to acquire clk\n");
|
||||
return PTR_ERR(qcadev->susclk);
|
||||
}
|
||||
|
||||
device_property_read_u32(&serdev->dev, "max-speed",
|
||||
&qcadev->oper_speed);
|
||||
if (!qcadev->oper_speed)
|
||||
@ -1851,38 +1901,40 @@ static int qca_serdev_probe(struct serdev_device *serdev)
|
||||
}
|
||||
} else {
|
||||
qcadev->btsoc_type = QCA_ROME;
|
||||
qcadev->bt_en = devm_gpiod_get(&serdev->dev, "enable",
|
||||
qcadev->bt_en = devm_gpiod_get_optional(&serdev->dev, "enable",
|
||||
GPIOD_OUT_LOW);
|
||||
if (IS_ERR(qcadev->bt_en)) {
|
||||
dev_err(&serdev->dev, "failed to acquire enable gpio\n");
|
||||
return PTR_ERR(qcadev->bt_en);
|
||||
if (!qcadev->bt_en) {
|
||||
dev_warn(&serdev->dev, "failed to acquire enable gpio\n");
|
||||
power_ctrl_enabled = false;
|
||||
}
|
||||
|
||||
qcadev->susclk = devm_clk_get(&serdev->dev, NULL);
|
||||
if (IS_ERR(qcadev->susclk)) {
|
||||
dev_err(&serdev->dev, "failed to acquire clk\n");
|
||||
return PTR_ERR(qcadev->susclk);
|
||||
qcadev->susclk = devm_clk_get_optional(&serdev->dev, NULL);
|
||||
if (!qcadev->susclk) {
|
||||
dev_warn(&serdev->dev, "failed to acquire clk\n");
|
||||
} else {
|
||||
err = clk_set_rate(qcadev->susclk, SUSCLK_RATE_32KHZ);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
err = clk_prepare_enable(qcadev->susclk);
|
||||
if (err)
|
||||
return err;
|
||||
}
|
||||
|
||||
err = clk_set_rate(qcadev->susclk, SUSCLK_RATE_32KHZ);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
err = clk_prepare_enable(qcadev->susclk);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
err = hci_uart_register_device(&qcadev->serdev_hu, &qca_proto);
|
||||
if (err) {
|
||||
BT_ERR("Rome serdev registration failed");
|
||||
clk_disable_unprepare(qcadev->susclk);
|
||||
if (qcadev->susclk)
|
||||
clk_disable_unprepare(qcadev->susclk);
|
||||
return err;
|
||||
}
|
||||
}
|
||||
|
||||
hdev = qcadev->serdev_hu.hdev;
|
||||
set_bit(HCI_QUIRK_NON_PERSISTENT_SETUP, &hdev->quirks);
|
||||
hdev->shutdown = qca_power_off;
|
||||
if (power_ctrl_enabled) {
|
||||
hdev = qcadev->serdev_hu.hdev;
|
||||
set_bit(HCI_QUIRK_NON_PERSISTENT_SETUP, &hdev->quirks);
|
||||
hdev->shutdown = qca_power_off;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -1893,7 +1945,7 @@ static void qca_serdev_remove(struct serdev_device *serdev)
|
||||
|
||||
if (qca_is_wcn399x(qcadev->btsoc_type))
|
||||
qca_power_shutdown(&qcadev->serdev_hu);
|
||||
else
|
||||
else if (qcadev->susclk)
|
||||
clk_disable_unprepare(qcadev->susclk);
|
||||
|
||||
hci_uart_unregister_device(&qcadev->serdev_hu);
|
||||
|
@ -42,3 +42,14 @@ config CRYPTO_DEV_CHELSIO_TLS
|
||||
|
||||
To compile this driver as a module, choose M here: the module
|
||||
will be called chtls.
|
||||
|
||||
config CHELSIO_TLS_DEVICE
|
||||
bool "Chelsio Inline KTLS Offload"
|
||||
depends on CHELSIO_T4
|
||||
depends on TLS_DEVICE
|
||||
select CRYPTO_DEV_CHELSIO
|
||||
default y
|
||||
help
|
||||
This flag enables support for kernel tls offload over Chelsio T6
|
||||
crypto accelerator. CONFIG_CHELSIO_TLS_DEVICE flag can be enabled
|
||||
only if CONFIG_TLS and CONFIG_TLS_DEVICE flags are enabled.
|
||||
|
@ -3,5 +3,8 @@ ccflags-y := -I $(srctree)/drivers/net/ethernet/chelsio/cxgb4
|
||||
|
||||
obj-$(CONFIG_CRYPTO_DEV_CHELSIO) += chcr.o
|
||||
chcr-objs := chcr_core.o chcr_algo.o
|
||||
#ifdef CONFIG_CHELSIO_TLS_DEVICE
|
||||
chcr-objs += chcr_ktls.o
|
||||
#endif
|
||||
chcr-$(CONFIG_CHELSIO_IPSEC_INLINE) += chcr_ipsec.o
|
||||
obj-$(CONFIG_CRYPTO_DEV_CHELSIO_TLS) += chtls/
|
||||
|
135
drivers/crypto/chelsio/chcr_common.h
Normal file
135
drivers/crypto/chelsio/chcr_common.h
Normal file
@ -0,0 +1,135 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-only */
|
||||
/* Copyright (C) 2020 Chelsio Communications. All rights reserved. */
|
||||
|
||||
#ifndef __CHCR_COMMON_H__
|
||||
#define __CHCR_COMMON_H__
|
||||
|
||||
#include "cxgb4.h"
|
||||
|
||||
#define CHCR_MAX_SALT 4
|
||||
#define CHCR_KEYCTX_MAC_KEY_SIZE_128 0
|
||||
#define CHCR_KEYCTX_CIPHER_KEY_SIZE_128 0
|
||||
#define CHCR_SCMD_CIPHER_MODE_AES_GCM 2
|
||||
#define CHCR_SCMD_CIPHER_MODE_AES_CTR 3
|
||||
#define CHCR_CPL_TX_SEC_PDU_LEN_64BIT 2
|
||||
#define CHCR_SCMD_SEQ_NO_CTRL_64BIT 3
|
||||
#define CHCR_SCMD_PROTO_VERSION_TLS 0
|
||||
#define CHCR_SCMD_PROTO_VERSION_GENERIC 4
|
||||
#define CHCR_SCMD_AUTH_MODE_GHASH 4
|
||||
#define AES_BLOCK_LEN 16
|
||||
|
||||
enum chcr_state {
|
||||
CHCR_INIT = 0,
|
||||
CHCR_ATTACH,
|
||||
CHCR_DETACH,
|
||||
};
|
||||
|
||||
struct chcr_dev {
|
||||
spinlock_t lock_chcr_dev; /* chcr dev structure lock */
|
||||
enum chcr_state state;
|
||||
atomic_t inflight;
|
||||
int wqretry;
|
||||
struct delayed_work detach_work;
|
||||
struct completion detach_comp;
|
||||
unsigned char tx_channel_id;
|
||||
};
|
||||
|
||||
struct uld_ctx {
|
||||
struct list_head entry;
|
||||
struct cxgb4_lld_info lldi;
|
||||
struct chcr_dev dev;
|
||||
};
|
||||
|
||||
struct ktls_key_ctx {
|
||||
__be32 ctx_hdr;
|
||||
u8 salt[CHCR_MAX_SALT];
|
||||
__be64 iv_to_auth;
|
||||
unsigned char key[TLS_CIPHER_AES_GCM_128_KEY_SIZE +
|
||||
TLS_CIPHER_AES_GCM_256_TAG_SIZE];
|
||||
};
|
||||
|
||||
/* Crypto key context */
|
||||
#define KEY_CONTEXT_CTX_LEN_S 24
|
||||
#define KEY_CONTEXT_CTX_LEN_V(x) ((x) << KEY_CONTEXT_CTX_LEN_S)
|
||||
|
||||
#define KEY_CONTEXT_SALT_PRESENT_S 10
|
||||
#define KEY_CONTEXT_SALT_PRESENT_V(x) ((x) << KEY_CONTEXT_SALT_PRESENT_S)
|
||||
#define KEY_CONTEXT_SALT_PRESENT_F KEY_CONTEXT_SALT_PRESENT_V(1U)
|
||||
|
||||
#define KEY_CONTEXT_VALID_S 0
|
||||
#define KEY_CONTEXT_VALID_V(x) ((x) << KEY_CONTEXT_VALID_S)
|
||||
#define KEY_CONTEXT_VALID_F KEY_CONTEXT_VALID_V(1U)
|
||||
|
||||
#define KEY_CONTEXT_CK_SIZE_S 6
|
||||
#define KEY_CONTEXT_CK_SIZE_V(x) ((x) << KEY_CONTEXT_CK_SIZE_S)
|
||||
|
||||
#define KEY_CONTEXT_MK_SIZE_S 2
|
||||
#define KEY_CONTEXT_MK_SIZE_V(x) ((x) << KEY_CONTEXT_MK_SIZE_S)
|
||||
|
||||
#define KEY_CONTEXT_OPAD_PRESENT_S 11
|
||||
#define KEY_CONTEXT_OPAD_PRESENT_V(x) ((x) << KEY_CONTEXT_OPAD_PRESENT_S)
|
||||
#define KEY_CONTEXT_OPAD_PRESENT_F KEY_CONTEXT_OPAD_PRESENT_V(1U)
|
||||
|
||||
#define FILL_KEY_CTX_HDR(ck_size, mk_size, ctx_len) \
|
||||
htonl(KEY_CONTEXT_MK_SIZE_V(mk_size) | \
|
||||
KEY_CONTEXT_CK_SIZE_V(ck_size) | \
|
||||
KEY_CONTEXT_VALID_F | \
|
||||
KEY_CONTEXT_SALT_PRESENT_F | \
|
||||
KEY_CONTEXT_CTX_LEN_V((ctx_len)))
|
||||
|
||||
struct uld_ctx *assign_chcr_device(void);
|
||||
|
||||
static inline void *chcr_copy_to_txd(const void *src, const struct sge_txq *q,
|
||||
void *pos, int length)
|
||||
{
|
||||
int left = (void *)q->stat - pos;
|
||||
u64 *p;
|
||||
|
||||
if (likely(length <= left)) {
|
||||
memcpy(pos, src, length);
|
||||
pos += length;
|
||||
} else {
|
||||
memcpy(pos, src, left);
|
||||
memcpy(q->desc, src + left, length - left);
|
||||
pos = (void *)q->desc + (length - left);
|
||||
}
|
||||
/* 0-pad to multiple of 16 */
|
||||
p = PTR_ALIGN(pos, 8);
|
||||
if ((uintptr_t)p & 8) {
|
||||
*p = 0;
|
||||
return p + 1;
|
||||
}
|
||||
return p;
|
||||
}
|
||||
|
||||
static inline unsigned int chcr_txq_avail(const struct sge_txq *q)
|
||||
{
|
||||
return q->size - 1 - q->in_use;
|
||||
}
|
||||
|
||||
static inline void chcr_txq_advance(struct sge_txq *q, unsigned int n)
|
||||
{
|
||||
q->in_use += n;
|
||||
q->pidx += n;
|
||||
if (q->pidx >= q->size)
|
||||
q->pidx -= q->size;
|
||||
}
|
||||
|
||||
static inline void chcr_eth_txq_stop(struct sge_eth_txq *q)
|
||||
{
|
||||
netif_tx_stop_queue(q->txq);
|
||||
q->q.stops++;
|
||||
}
|
||||
|
||||
static inline unsigned int chcr_sgl_len(unsigned int n)
|
||||
{
|
||||
n--;
|
||||
return (3 * n) / 2 + (n & 1) + 2;
|
||||
}
|
||||
|
||||
static inline unsigned int chcr_flits_to_desc(unsigned int n)
|
||||
{
|
||||
WARN_ON(n > SGE_MAX_WR_LEN / 8);
|
||||
return DIV_ROUND_UP(n, 8);
|
||||
}
|
||||
#endif /* __CHCR_COMMON_H__ */
|
@ -28,13 +28,21 @@
|
||||
|
||||
static struct chcr_driver_data drv_data;
|
||||
|
||||
typedef int (*chcr_handler_func)(struct chcr_dev *dev, unsigned char *input);
|
||||
static int cpl_fw6_pld_handler(struct chcr_dev *dev, unsigned char *input);
|
||||
typedef int (*chcr_handler_func)(struct adapter *adap, unsigned char *input);
|
||||
static int cpl_fw6_pld_handler(struct adapter *adap, unsigned char *input);
|
||||
static void *chcr_uld_add(const struct cxgb4_lld_info *lld);
|
||||
static int chcr_uld_state_change(void *handle, enum cxgb4_state state);
|
||||
|
||||
#ifdef CONFIG_CHELSIO_IPSEC_INLINE
|
||||
static void update_netdev_features(void);
|
||||
#endif /* CONFIG_CHELSIO_IPSEC_INLINE */
|
||||
|
||||
static chcr_handler_func work_handlers[NUM_CPL_CMDS] = {
|
||||
[CPL_FW6_PLD] = cpl_fw6_pld_handler,
|
||||
#ifdef CONFIG_CHELSIO_TLS_DEVICE
|
||||
[CPL_ACT_OPEN_RPL] = chcr_ktls_cpl_act_open_rpl,
|
||||
[CPL_SET_TCB_RPL] = chcr_ktls_cpl_set_tcb_rpl,
|
||||
#endif
|
||||
};
|
||||
|
||||
static struct cxgb4_uld_info chcr_uld_info = {
|
||||
@ -45,9 +53,9 @@ static struct cxgb4_uld_info chcr_uld_info = {
|
||||
.add = chcr_uld_add,
|
||||
.state_change = chcr_uld_state_change,
|
||||
.rx_handler = chcr_uld_rx_handler,
|
||||
#ifdef CONFIG_CHELSIO_IPSEC_INLINE
|
||||
#if defined(CONFIG_CHELSIO_IPSEC_INLINE) || defined(CONFIG_CHELSIO_TLS_DEVICE)
|
||||
.tx_handler = chcr_uld_tx_handler,
|
||||
#endif /* CONFIG_CHELSIO_IPSEC_INLINE */
|
||||
#endif /* CONFIG_CHELSIO_IPSEC_INLINE || CONFIG_CHELSIO_TLS_DEVICE */
|
||||
};
|
||||
|
||||
static void detach_work_fn(struct work_struct *work)
|
||||
@ -125,8 +133,6 @@ static void chcr_dev_init(struct uld_ctx *u_ctx)
|
||||
atomic_set(&dev->inflight, 0);
|
||||
mutex_lock(&drv_data.drv_mutex);
|
||||
list_add_tail(&u_ctx->entry, &drv_data.inact_dev);
|
||||
if (!drv_data.last_dev)
|
||||
drv_data.last_dev = u_ctx;
|
||||
mutex_unlock(&drv_data.drv_mutex);
|
||||
}
|
||||
|
||||
@ -150,14 +156,13 @@ static int chcr_dev_move(struct uld_ctx *u_ctx)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int cpl_fw6_pld_handler(struct chcr_dev *dev,
|
||||
static int cpl_fw6_pld_handler(struct adapter *adap,
|
||||
unsigned char *input)
|
||||
{
|
||||
struct crypto_async_request *req;
|
||||
struct cpl_fw6_pld *fw6_pld;
|
||||
u32 ack_err_status = 0;
|
||||
int error_status = 0;
|
||||
struct adapter *adap = padap(dev);
|
||||
|
||||
fw6_pld = (struct cpl_fw6_pld *)input;
|
||||
req = (struct crypto_async_request *)(uintptr_t)be64_to_cpu(
|
||||
@ -201,10 +206,11 @@ static void *chcr_uld_add(const struct cxgb4_lld_info *lld)
|
||||
}
|
||||
u_ctx->lldi = *lld;
|
||||
chcr_dev_init(u_ctx);
|
||||
#ifdef CONFIG_CHELSIO_IPSEC_INLINE
|
||||
if (lld->crypto & ULP_CRYPTO_IPSEC_INLINE)
|
||||
chcr_add_xfrmops(lld);
|
||||
#endif /* CONFIG_CHELSIO_IPSEC_INLINE */
|
||||
|
||||
#ifdef CONFIG_CHELSIO_TLS_DEVICE
|
||||
if (lld->ulp_crypto & ULP_CRYPTO_KTLS_INLINE)
|
||||
chcr_enable_ktls(padap(&u_ctx->dev));
|
||||
#endif
|
||||
out:
|
||||
return u_ctx;
|
||||
}
|
||||
@ -214,26 +220,37 @@ int chcr_uld_rx_handler(void *handle, const __be64 *rsp,
|
||||
{
|
||||
struct uld_ctx *u_ctx = (struct uld_ctx *)handle;
|
||||
struct chcr_dev *dev = &u_ctx->dev;
|
||||
struct adapter *adap = padap(dev);
|
||||
const struct cpl_fw6_pld *rpl = (struct cpl_fw6_pld *)rsp;
|
||||
|
||||
if (rpl->opcode != CPL_FW6_PLD) {
|
||||
pr_err("Unsupported opcode\n");
|
||||
if (!work_handlers[rpl->opcode]) {
|
||||
pr_err("Unsupported opcode %d received\n", rpl->opcode);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (!pgl)
|
||||
work_handlers[rpl->opcode](dev, (unsigned char *)&rsp[1]);
|
||||
work_handlers[rpl->opcode](adap, (unsigned char *)&rsp[1]);
|
||||
else
|
||||
work_handlers[rpl->opcode](dev, pgl->va);
|
||||
work_handlers[rpl->opcode](adap, pgl->va);
|
||||
return 0;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_CHELSIO_IPSEC_INLINE
|
||||
#if defined(CONFIG_CHELSIO_IPSEC_INLINE) || defined(CONFIG_CHELSIO_TLS_DEVICE)
|
||||
int chcr_uld_tx_handler(struct sk_buff *skb, struct net_device *dev)
|
||||
{
|
||||
/* In case if skb's decrypted bit is set, it's nic tls packet, else it's
|
||||
* ipsec packet.
|
||||
*/
|
||||
#ifdef CONFIG_CHELSIO_TLS_DEVICE
|
||||
if (skb->decrypted)
|
||||
return chcr_ktls_xmit(skb, dev);
|
||||
#endif
|
||||
#ifdef CONFIG_CHELSIO_IPSEC_INLINE
|
||||
return chcr_ipsec_xmit(skb, dev);
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
#endif /* CONFIG_CHELSIO_IPSEC_INLINE */
|
||||
#endif /* CONFIG_CHELSIO_IPSEC_INLINE || CONFIG_CHELSIO_TLS_DEVICE */
|
||||
|
||||
static void chcr_detach_device(struct uld_ctx *u_ctx)
|
||||
{
|
||||
@ -280,6 +297,24 @@ static int chcr_uld_state_change(void *handle, enum cxgb4_state state)
|
||||
return ret;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_CHELSIO_IPSEC_INLINE
|
||||
static void update_netdev_features(void)
|
||||
{
|
||||
struct uld_ctx *u_ctx, *tmp;
|
||||
|
||||
mutex_lock(&drv_data.drv_mutex);
|
||||
list_for_each_entry_safe(u_ctx, tmp, &drv_data.inact_dev, entry) {
|
||||
if (u_ctx->lldi.crypto & ULP_CRYPTO_IPSEC_INLINE)
|
||||
chcr_add_xfrmops(&u_ctx->lldi);
|
||||
}
|
||||
list_for_each_entry_safe(u_ctx, tmp, &drv_data.act_dev, entry) {
|
||||
if (u_ctx->lldi.crypto & ULP_CRYPTO_IPSEC_INLINE)
|
||||
chcr_add_xfrmops(&u_ctx->lldi);
|
||||
}
|
||||
mutex_unlock(&drv_data.drv_mutex);
|
||||
}
|
||||
#endif /* CONFIG_CHELSIO_IPSEC_INLINE */
|
||||
|
||||
static int __init chcr_crypto_init(void)
|
||||
{
|
||||
INIT_LIST_HEAD(&drv_data.act_dev);
|
||||
@ -289,6 +324,12 @@ static int __init chcr_crypto_init(void)
|
||||
drv_data.last_dev = NULL;
|
||||
cxgb4_register_uld(CXGB4_ULD_CRYPTO, &chcr_uld_info);
|
||||
|
||||
#ifdef CONFIG_CHELSIO_IPSEC_INLINE
|
||||
rtnl_lock();
|
||||
update_netdev_features();
|
||||
rtnl_unlock();
|
||||
#endif /* CONFIG_CHELSIO_IPSEC_INLINE */
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -304,12 +345,20 @@ static void __exit chcr_crypto_exit(void)
|
||||
list_for_each_entry_safe(u_ctx, tmp, &drv_data.act_dev, entry) {
|
||||
adap = padap(&u_ctx->dev);
|
||||
memset(&adap->chcr_stats, 0, sizeof(adap->chcr_stats));
|
||||
#ifdef CONFIG_CHELSIO_TLS_DEVICE
|
||||
if (u_ctx->lldi.ulp_crypto & ULP_CRYPTO_KTLS_INLINE)
|
||||
chcr_disable_ktls(adap);
|
||||
#endif
|
||||
list_del(&u_ctx->entry);
|
||||
kfree(u_ctx);
|
||||
}
|
||||
list_for_each_entry_safe(u_ctx, tmp, &drv_data.inact_dev, entry) {
|
||||
adap = padap(&u_ctx->dev);
|
||||
memset(&adap->chcr_stats, 0, sizeof(adap->chcr_stats));
|
||||
#ifdef CONFIG_CHELSIO_TLS_DEVICE
|
||||
if (u_ctx->lldi.ulp_crypto & ULP_CRYPTO_KTLS_INLINE)
|
||||
chcr_disable_ktls(adap);
|
||||
#endif
|
||||
list_del(&u_ctx->entry);
|
||||
kfree(u_ctx);
|
||||
}
|
||||
|
@ -222,4 +222,11 @@ int chcr_handle_resp(struct crypto_async_request *req, unsigned char *input,
|
||||
int err);
|
||||
int chcr_ipsec_xmit(struct sk_buff *skb, struct net_device *dev);
|
||||
void chcr_add_xfrmops(const struct cxgb4_lld_info *lld);
|
||||
#ifdef CONFIG_CHELSIO_TLS_DEVICE
|
||||
void chcr_enable_ktls(struct adapter *adap);
|
||||
void chcr_disable_ktls(struct adapter *adap);
|
||||
int chcr_ktls_cpl_act_open_rpl(struct adapter *adap, unsigned char *input);
|
||||
int chcr_ktls_cpl_set_tcb_rpl(struct adapter *adap, unsigned char *input);
|
||||
int chcr_ktls_xmit(struct sk_buff *skb, struct net_device *dev);
|
||||
#endif
|
||||
#endif /* __CHCR_CORE_H__ */
|
||||
|
@ -99,9 +99,7 @@ void chcr_add_xfrmops(const struct cxgb4_lld_info *lld)
|
||||
netdev->xfrmdev_ops = &chcr_xfrmdev_ops;
|
||||
netdev->hw_enc_features |= NETIF_F_HW_ESP;
|
||||
netdev->features |= NETIF_F_HW_ESP;
|
||||
rtnl_lock();
|
||||
netdev_change_features(netdev);
|
||||
rtnl_unlock();
|
||||
}
|
||||
}
|
||||
|
||||
|
2020
drivers/crypto/chelsio/chcr_ktls.c
Normal file
2020
drivers/crypto/chelsio/chcr_ktls.c
Normal file
File diff suppressed because it is too large
Load Diff
98
drivers/crypto/chelsio/chcr_ktls.h
Normal file
98
drivers/crypto/chelsio/chcr_ktls.h
Normal file
@ -0,0 +1,98 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-only */
|
||||
/* Copyright (C) 2020 Chelsio Communications. All rights reserved. */
|
||||
|
||||
#ifndef __CHCR_KTLS_H__
|
||||
#define __CHCR_KTLS_H__
|
||||
|
||||
#ifdef CONFIG_CHELSIO_TLS_DEVICE
|
||||
#include <net/tls.h>
|
||||
#include "cxgb4.h"
|
||||
#include "t4_msg.h"
|
||||
#include "t4_tcb.h"
|
||||
#include "l2t.h"
|
||||
#include "chcr_common.h"
|
||||
#include "cxgb4_uld.h"
|
||||
|
||||
#define CHCR_TCB_STATE_CLOSED 0
|
||||
#define CHCR_KTLS_KEY_CTX_LEN 16
|
||||
#define CHCR_SET_TCB_FIELD_LEN sizeof(struct cpl_set_tcb_field)
|
||||
#define CHCR_PLAIN_TX_DATA_LEN (sizeof(struct fw_ulptx_wr) +\
|
||||
sizeof(struct ulp_txpkt) +\
|
||||
sizeof(struct ulptx_idata) +\
|
||||
sizeof(struct cpl_tx_data))
|
||||
|
||||
#define CHCR_KTLS_WR_SIZE (CHCR_PLAIN_TX_DATA_LEN +\
|
||||
sizeof(struct cpl_tx_sec_pdu))
|
||||
|
||||
enum chcr_ktls_conn_state {
|
||||
KTLS_CONN_CLOSED,
|
||||
KTLS_CONN_ACT_OPEN_REQ,
|
||||
KTLS_CONN_ACT_OPEN_RPL,
|
||||
KTLS_CONN_SET_TCB_REQ,
|
||||
KTLS_CONN_SET_TCB_RPL,
|
||||
KTLS_CONN_TX_READY,
|
||||
};
|
||||
|
||||
struct chcr_ktls_info {
|
||||
struct sock *sk;
|
||||
spinlock_t lock; /* state machine lock */
|
||||
struct ktls_key_ctx key_ctx;
|
||||
struct adapter *adap;
|
||||
struct l2t_entry *l2te;
|
||||
struct net_device *netdev;
|
||||
u64 iv;
|
||||
u64 record_no;
|
||||
int tid;
|
||||
int atid;
|
||||
int rx_qid;
|
||||
u32 iv_size;
|
||||
u32 prev_seq;
|
||||
u32 prev_ack;
|
||||
u32 salt_size;
|
||||
u32 key_ctx_len;
|
||||
u32 scmd0_seqno_numivs;
|
||||
u32 scmd0_ivgen_hdrlen;
|
||||
u32 tcp_start_seq_number;
|
||||
u32 scmd0_short_seqno_numivs;
|
||||
u32 scmd0_short_ivgen_hdrlen;
|
||||
enum chcr_ktls_conn_state connection_state;
|
||||
u16 prev_win;
|
||||
u8 tx_chan;
|
||||
u8 smt_idx;
|
||||
u8 port_id;
|
||||
u8 ip_family;
|
||||
u8 first_qset;
|
||||
};
|
||||
|
||||
struct chcr_ktls_ofld_ctx_tx {
|
||||
struct tls_offload_context_tx base;
|
||||
struct chcr_ktls_info *chcr_info;
|
||||
};
|
||||
|
||||
static inline struct chcr_ktls_ofld_ctx_tx *
|
||||
chcr_get_ktls_tx_context(struct tls_context *tls_ctx)
|
||||
{
|
||||
BUILD_BUG_ON(sizeof(struct chcr_ktls_ofld_ctx_tx) >
|
||||
TLS_OFFLOAD_CONTEXT_SIZE_TX);
|
||||
return container_of(tls_offload_ctx_tx(tls_ctx),
|
||||
struct chcr_ktls_ofld_ctx_tx,
|
||||
base);
|
||||
}
|
||||
|
||||
static inline int chcr_get_first_rx_qid(struct adapter *adap)
|
||||
{
|
||||
/* u_ctx is saved in adap, fetch it */
|
||||
struct uld_ctx *u_ctx = adap->uld[CXGB4_ULD_CRYPTO].handle;
|
||||
|
||||
if (!u_ctx)
|
||||
return -1;
|
||||
return u_ctx->lldi.rxq_ids[0];
|
||||
}
|
||||
|
||||
void chcr_enable_ktls(struct adapter *adap);
|
||||
void chcr_disable_ktls(struct adapter *adap);
|
||||
int chcr_ktls_cpl_act_open_rpl(struct adapter *adap, unsigned char *input);
|
||||
int chcr_ktls_cpl_set_tcb_rpl(struct adapter *adap, unsigned char *input);
|
||||
int chcr_ktls_xmit(struct sk_buff *skb, struct net_device *dev);
|
||||
#endif /* CONFIG_CHELSIO_TLS_DEVICE */
|
||||
#endif /* __CHCR_KTLS_H__ */
|
@ -445,6 +445,7 @@ void chtls_destroy_sock(struct sock *sk)
|
||||
chtls_purge_write_queue(sk);
|
||||
free_tls_keyid(sk);
|
||||
kref_put(&csk->kref, chtls_sock_release);
|
||||
csk->cdev = NULL;
|
||||
sk->sk_prot = &tcp_prot;
|
||||
sk->sk_prot->destroy(sk);
|
||||
}
|
||||
@ -759,8 +760,10 @@ static void chtls_release_resources(struct sock *sk)
|
||||
csk->l2t_entry = NULL;
|
||||
}
|
||||
|
||||
cxgb4_remove_tid(tids, csk->port_id, tid, sk->sk_family);
|
||||
sock_put(sk);
|
||||
if (sk->sk_state != TCP_SYN_SENT) {
|
||||
cxgb4_remove_tid(tids, csk->port_id, tid, sk->sk_family);
|
||||
sock_put(sk);
|
||||
}
|
||||
}
|
||||
|
||||
static void chtls_conn_done(struct sock *sk)
|
||||
@ -1716,6 +1719,9 @@ static void chtls_peer_close(struct sock *sk, struct sk_buff *skb)
|
||||
{
|
||||
struct chtls_sock *csk = rcu_dereference_sk_user_data(sk);
|
||||
|
||||
if (csk_flag_nochk(csk, CSK_ABORT_RPL_PENDING))
|
||||
goto out;
|
||||
|
||||
sk->sk_shutdown |= RCV_SHUTDOWN;
|
||||
sock_set_flag(sk, SOCK_DONE);
|
||||
|
||||
@ -1748,6 +1754,7 @@ static void chtls_peer_close(struct sock *sk, struct sk_buff *skb)
|
||||
else
|
||||
sk_wake_async(sk, SOCK_WAKE_WAITD, POLL_IN);
|
||||
}
|
||||
out:
|
||||
kfree_skb(skb);
|
||||
}
|
||||
|
||||
@ -1758,6 +1765,10 @@ static void chtls_close_con_rpl(struct sock *sk, struct sk_buff *skb)
|
||||
struct tcp_sock *tp;
|
||||
|
||||
csk = rcu_dereference_sk_user_data(sk);
|
||||
|
||||
if (csk_flag_nochk(csk, CSK_ABORT_RPL_PENDING))
|
||||
goto out;
|
||||
|
||||
tp = tcp_sk(sk);
|
||||
|
||||
tp->snd_una = ntohl(rpl->snd_nxt) - 1; /* exclude FIN */
|
||||
@ -1787,6 +1798,7 @@ static void chtls_close_con_rpl(struct sock *sk, struct sk_buff *skb)
|
||||
default:
|
||||
pr_info("close_con_rpl in bad state %d\n", sk->sk_state);
|
||||
}
|
||||
out:
|
||||
kfree_skb(skb);
|
||||
}
|
||||
|
||||
@ -1896,6 +1908,7 @@ static void chtls_send_abort_rpl(struct sock *sk, struct sk_buff *skb,
|
||||
}
|
||||
|
||||
set_abort_rpl_wr(reply_skb, tid, status);
|
||||
kfree_skb(skb);
|
||||
set_wr_txq(reply_skb, CPL_PRIORITY_DATA, queue);
|
||||
if (csk_conn_inline(csk)) {
|
||||
struct l2t_entry *e = csk->l2t_entry;
|
||||
@ -1906,7 +1919,6 @@ static void chtls_send_abort_rpl(struct sock *sk, struct sk_buff *skb,
|
||||
}
|
||||
}
|
||||
cxgb4_ofld_send(cdev->lldi->ports[0], reply_skb);
|
||||
kfree_skb(skb);
|
||||
}
|
||||
|
||||
/*
|
||||
@ -2008,7 +2020,8 @@ static void chtls_abort_req_rss(struct sock *sk, struct sk_buff *skb)
|
||||
chtls_conn_done(sk);
|
||||
}
|
||||
|
||||
chtls_send_abort_rpl(sk, skb, csk->cdev, rst_status, queue);
|
||||
chtls_send_abort_rpl(sk, skb, BLOG_SKB_CB(skb)->cdev,
|
||||
rst_status, queue);
|
||||
}
|
||||
|
||||
static void chtls_abort_rpl_rss(struct sock *sk, struct sk_buff *skb)
|
||||
@ -2042,6 +2055,7 @@ static int chtls_conn_cpl(struct chtls_dev *cdev, struct sk_buff *skb)
|
||||
struct cpl_peer_close *req = cplhdr(skb) + RSS_HDR;
|
||||
void (*fn)(struct sock *sk, struct sk_buff *skb);
|
||||
unsigned int hwtid = GET_TID(req);
|
||||
struct chtls_sock *csk;
|
||||
struct sock *sk;
|
||||
u8 opcode;
|
||||
|
||||
@ -2051,6 +2065,8 @@ static int chtls_conn_cpl(struct chtls_dev *cdev, struct sk_buff *skb)
|
||||
if (!sk)
|
||||
goto rel_skb;
|
||||
|
||||
csk = sk->sk_user_data;
|
||||
|
||||
switch (opcode) {
|
||||
case CPL_PEER_CLOSE:
|
||||
fn = chtls_peer_close;
|
||||
@ -2059,6 +2075,11 @@ static int chtls_conn_cpl(struct chtls_dev *cdev, struct sk_buff *skb)
|
||||
fn = chtls_close_con_rpl;
|
||||
break;
|
||||
case CPL_ABORT_REQ_RSS:
|
||||
/*
|
||||
* Save the offload device in the skb, we may process this
|
||||
* message after the socket has closed.
|
||||
*/
|
||||
BLOG_SKB_CB(skb)->cdev = csk->cdev;
|
||||
fn = chtls_abort_req_rss;
|
||||
break;
|
||||
case CPL_ABORT_RPL_RSS:
|
||||
|
@ -902,14 +902,6 @@ static int chtls_skb_copy_to_page_nocache(struct sock *sk,
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Read TLS header to find content type and data length */
|
||||
static int tls_header_read(struct tls_hdr *thdr, struct iov_iter *from)
|
||||
{
|
||||
if (copy_from_iter(thdr, sizeof(*thdr), from) != sizeof(*thdr))
|
||||
return -EFAULT;
|
||||
return (__force int)cpu_to_be16(thdr->length);
|
||||
}
|
||||
|
||||
static int csk_mem_free(struct chtls_dev *cdev, struct sock *sk)
|
||||
{
|
||||
return (cdev->max_host_sndbuf - sk->sk_wmem_queued);
|
||||
@ -981,6 +973,37 @@ do_interrupted:
|
||||
goto do_rm_wq;
|
||||
}
|
||||
|
||||
static int chtls_proccess_cmsg(struct sock *sk, struct msghdr *msg,
|
||||
unsigned char *record_type)
|
||||
{
|
||||
struct cmsghdr *cmsg;
|
||||
int rc = -EINVAL;
|
||||
|
||||
for_each_cmsghdr(cmsg, msg) {
|
||||
if (!CMSG_OK(msg, cmsg))
|
||||
return -EINVAL;
|
||||
if (cmsg->cmsg_level != SOL_TLS)
|
||||
continue;
|
||||
|
||||
switch (cmsg->cmsg_type) {
|
||||
case TLS_SET_RECORD_TYPE:
|
||||
if (cmsg->cmsg_len < CMSG_LEN(sizeof(*record_type)))
|
||||
return -EINVAL;
|
||||
|
||||
if (msg->msg_flags & MSG_MORE)
|
||||
return -EINVAL;
|
||||
|
||||
*record_type = *(unsigned char *)CMSG_DATA(cmsg);
|
||||
rc = 0;
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
|
||||
return rc;
|
||||
}
|
||||
|
||||
int chtls_sendmsg(struct sock *sk, struct msghdr *msg, size_t size)
|
||||
{
|
||||
struct chtls_sock *csk = rcu_dereference_sk_user_data(sk);
|
||||
@ -1022,15 +1045,21 @@ int chtls_sendmsg(struct sock *sk, struct msghdr *msg, size_t size)
|
||||
goto wait_for_sndbuf;
|
||||
|
||||
if (is_tls_tx(csk) && !csk->tlshws.txleft) {
|
||||
struct tls_hdr hdr;
|
||||
unsigned char record_type = TLS_RECORD_TYPE_DATA;
|
||||
|
||||
recordsz = tls_header_read(&hdr, &msg->msg_iter);
|
||||
size -= TLS_HEADER_LENGTH;
|
||||
copied += TLS_HEADER_LENGTH;
|
||||
if (unlikely(msg->msg_controllen)) {
|
||||
err = chtls_proccess_cmsg(sk, msg,
|
||||
&record_type);
|
||||
if (err)
|
||||
goto out_err;
|
||||
}
|
||||
|
||||
recordsz = size;
|
||||
csk->tlshws.txleft = recordsz;
|
||||
csk->tlshws.type = hdr.type;
|
||||
csk->tlshws.type = record_type;
|
||||
|
||||
if (skb)
|
||||
ULP_SKB_CB(skb)->ulp.tls.type = hdr.type;
|
||||
ULP_SKB_CB(skb)->ulp.tls.type = record_type;
|
||||
}
|
||||
|
||||
if (!skb || (ULP_SKB_CB(skb)->flags & ULPCB_FLAG_NO_APPEND) ||
|
||||
@ -1521,6 +1550,22 @@ found_ok_skb:
|
||||
}
|
||||
}
|
||||
}
|
||||
/* Set record type if not already done. For a non-data record,
|
||||
* do not proceed if record type could not be copied.
|
||||
*/
|
||||
if (ULP_SKB_CB(skb)->flags & ULPCB_FLAG_TLS_HDR) {
|
||||
struct tls_hdr *thdr = (struct tls_hdr *)skb->data;
|
||||
int cerr = 0;
|
||||
|
||||
cerr = put_cmsg(msg, SOL_TLS, TLS_GET_RECORD_TYPE,
|
||||
sizeof(thdr->type), &thdr->type);
|
||||
|
||||
if (cerr && thdr->type != TLS_RECORD_TYPE_DATA)
|
||||
return -EIO;
|
||||
/* don't send tls header, skip copy */
|
||||
goto skip_copy;
|
||||
}
|
||||
|
||||
if (skb_copy_datagram_msg(skb, offset, msg, avail)) {
|
||||
if (!copied) {
|
||||
copied = -EFAULT;
|
||||
|
@ -3570,7 +3570,8 @@ static void mlx5_ib_set_rule_source_port(struct mlx5_ib_dev *dev,
|
||||
misc = MLX5_ADDR_OF(fte_match_param, spec->match_criteria,
|
||||
misc_parameters_2);
|
||||
|
||||
MLX5_SET_TO_ONES(fte_match_set_misc2, misc, metadata_reg_c_0);
|
||||
MLX5_SET(fte_match_set_misc2, misc, metadata_reg_c_0,
|
||||
mlx5_eswitch_get_vport_metadata_mask());
|
||||
} else {
|
||||
misc = MLX5_ADDR_OF(fte_match_param, spec->match_value,
|
||||
misc_parameters);
|
||||
@ -6391,6 +6392,7 @@ static int mlx5_ib_stage_init_init(struct mlx5_ib_dev *dev)
|
||||
spin_lock_init(&dev->reset_flow_resource_lock);
|
||||
xa_init(&dev->odp_mkeys);
|
||||
xa_init(&dev->sig_mrs);
|
||||
spin_lock_init(&dev->mkey_lock);
|
||||
|
||||
spin_lock_init(&dev->dm.lock);
|
||||
dev->dm.dev = mdev;
|
||||
|
@ -994,6 +994,11 @@ struct mlx5_ib_dev {
|
||||
/* sync used page count stats
|
||||
*/
|
||||
struct mlx5_ib_resources devr;
|
||||
|
||||
/* protect mkey key part */
|
||||
spinlock_t mkey_lock;
|
||||
u8 mkey_key;
|
||||
|
||||
struct mlx5_mr_cache cache;
|
||||
struct timer_list delay_timer;
|
||||
/* Prevents soft lock on massive reg MRs */
|
||||
|
@ -47,6 +47,46 @@ enum {
|
||||
|
||||
#define MLX5_UMR_ALIGN 2048
|
||||
|
||||
static void
|
||||
create_mkey_callback(int status, struct mlx5_async_work *context);
|
||||
|
||||
static void
|
||||
assign_mkey_variant(struct mlx5_ib_dev *dev, struct mlx5_core_mkey *mkey,
|
||||
u32 *in)
|
||||
{
|
||||
void *mkc;
|
||||
u8 key;
|
||||
|
||||
spin_lock_irq(&dev->mkey_lock);
|
||||
key = dev->mkey_key++;
|
||||
spin_unlock_irq(&dev->mkey_lock);
|
||||
|
||||
mkc = MLX5_ADDR_OF(create_mkey_in, in, memory_key_mkey_entry);
|
||||
MLX5_SET(mkc, mkc, mkey_7_0, key);
|
||||
mkey->key = key;
|
||||
}
|
||||
|
||||
static int
|
||||
mlx5_ib_create_mkey(struct mlx5_ib_dev *dev, struct mlx5_core_mkey *mkey,
|
||||
u32 *in, int inlen)
|
||||
{
|
||||
assign_mkey_variant(dev, mkey, in);
|
||||
return mlx5_core_create_mkey(dev->mdev, mkey, in, inlen);
|
||||
}
|
||||
|
||||
static int
|
||||
mlx5_ib_create_mkey_cb(struct mlx5_ib_dev *dev,
|
||||
struct mlx5_core_mkey *mkey,
|
||||
struct mlx5_async_ctx *async_ctx,
|
||||
u32 *in, int inlen, u32 *out, int outlen,
|
||||
struct mlx5_async_work *context)
|
||||
{
|
||||
MLX5_SET(create_mkey_in, in, opcode, MLX5_CMD_OP_CREATE_MKEY);
|
||||
assign_mkey_variant(dev, mkey, in);
|
||||
return mlx5_cmd_exec_cb(async_ctx, in, inlen, out, outlen,
|
||||
create_mkey_callback, context);
|
||||
}
|
||||
|
||||
static void clean_mr(struct mlx5_ib_dev *dev, struct mlx5_ib_mr *mr);
|
||||
static void dereg_mr(struct mlx5_ib_dev *dev, struct mlx5_ib_mr *mr);
|
||||
static int mr_cache_max_order(struct mlx5_ib_dev *dev);
|
||||
@ -79,7 +119,7 @@ static bool use_umr_mtt_update(struct mlx5_ib_mr *mr, u64 start, u64 length)
|
||||
length + (start & (MLX5_ADAPTER_PAGE_SIZE - 1));
|
||||
}
|
||||
|
||||
static void reg_mr_callback(int status, struct mlx5_async_work *context)
|
||||
static void create_mkey_callback(int status, struct mlx5_async_work *context)
|
||||
{
|
||||
struct mlx5_ib_mr *mr =
|
||||
container_of(context, struct mlx5_ib_mr, cb_work);
|
||||
@ -87,7 +127,6 @@ static void reg_mr_callback(int status, struct mlx5_async_work *context)
|
||||
struct mlx5_mr_cache *cache = &dev->cache;
|
||||
int c = order2idx(dev, mr->order);
|
||||
struct mlx5_cache_ent *ent = &cache->ent[c];
|
||||
u8 key;
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&ent->lock, flags);
|
||||
@ -102,10 +141,8 @@ static void reg_mr_callback(int status, struct mlx5_async_work *context)
|
||||
}
|
||||
|
||||
mr->mmkey.type = MLX5_MKEY_MR;
|
||||
spin_lock_irqsave(&dev->mdev->priv.mkey_lock, flags);
|
||||
key = dev->mdev->priv.mkey_key++;
|
||||
spin_unlock_irqrestore(&dev->mdev->priv.mkey_lock, flags);
|
||||
mr->mmkey.key = mlx5_idx_to_mkey(MLX5_GET(create_mkey_out, mr->out, mkey_index)) | key;
|
||||
mr->mmkey.key |= mlx5_idx_to_mkey(
|
||||
MLX5_GET(create_mkey_out, mr->out, mkey_index));
|
||||
|
||||
cache->last_add = jiffies;
|
||||
|
||||
@ -163,10 +200,10 @@ static int add_keys(struct mlx5_ib_dev *dev, int c, int num)
|
||||
spin_lock_irq(&ent->lock);
|
||||
ent->pending++;
|
||||
spin_unlock_irq(&ent->lock);
|
||||
err = mlx5_core_create_mkey_cb(dev->mdev, &mr->mmkey,
|
||||
err = mlx5_ib_create_mkey_cb(dev, &mr->mmkey,
|
||||
&dev->async_ctx, in, inlen,
|
||||
mr->out, sizeof(mr->out),
|
||||
reg_mr_callback, &mr->cb_work);
|
||||
&mr->cb_work);
|
||||
if (err) {
|
||||
spin_lock_irq(&ent->lock);
|
||||
ent->pending--;
|
||||
@ -685,7 +722,6 @@ struct ib_mr *mlx5_ib_get_dma_mr(struct ib_pd *pd, int acc)
|
||||
{
|
||||
struct mlx5_ib_dev *dev = to_mdev(pd->device);
|
||||
int inlen = MLX5_ST_SZ_BYTES(create_mkey_in);
|
||||
struct mlx5_core_dev *mdev = dev->mdev;
|
||||
struct mlx5_ib_mr *mr;
|
||||
void *mkc;
|
||||
u32 *in;
|
||||
@ -707,7 +743,7 @@ struct ib_mr *mlx5_ib_get_dma_mr(struct ib_pd *pd, int acc)
|
||||
MLX5_SET(mkc, mkc, length64, 1);
|
||||
set_mkc_access_pd_addr_fields(mkc, acc, 0, pd);
|
||||
|
||||
err = mlx5_core_create_mkey(mdev, &mr->mmkey, in, inlen);
|
||||
err = mlx5_ib_create_mkey(dev, &mr->mmkey, in, inlen);
|
||||
if (err)
|
||||
goto err_in;
|
||||
|
||||
@ -1097,7 +1133,7 @@ static struct mlx5_ib_mr *reg_create(struct ib_mr *ibmr, struct ib_pd *pd,
|
||||
get_octo_len(virt_addr, length, page_shift));
|
||||
}
|
||||
|
||||
err = mlx5_core_create_mkey(dev->mdev, &mr->mmkey, in, inlen);
|
||||
err = mlx5_ib_create_mkey(dev, &mr->mmkey, in, inlen);
|
||||
if (err) {
|
||||
mlx5_ib_warn(dev, "create mkey failed\n");
|
||||
goto err_2;
|
||||
@ -1137,7 +1173,6 @@ static struct ib_mr *mlx5_ib_get_dm_mr(struct ib_pd *pd, u64 start_addr,
|
||||
{
|
||||
struct mlx5_ib_dev *dev = to_mdev(pd->device);
|
||||
int inlen = MLX5_ST_SZ_BYTES(create_mkey_in);
|
||||
struct mlx5_core_dev *mdev = dev->mdev;
|
||||
struct mlx5_ib_mr *mr;
|
||||
void *mkc;
|
||||
u32 *in;
|
||||
@ -1160,7 +1195,7 @@ static struct ib_mr *mlx5_ib_get_dm_mr(struct ib_pd *pd, u64 start_addr,
|
||||
MLX5_SET64(mkc, mkc, len, length);
|
||||
set_mkc_access_pd_addr_fields(mkc, acc, start_addr, pd);
|
||||
|
||||
err = mlx5_core_create_mkey(mdev, &mr->mmkey, in, inlen);
|
||||
err = mlx5_ib_create_mkey(dev, &mr->mmkey, in, inlen);
|
||||
if (err)
|
||||
goto err_in;
|
||||
|
||||
@ -1638,7 +1673,7 @@ static int _mlx5_alloc_mkey_descs(struct ib_pd *pd, struct mlx5_ib_mr *mr,
|
||||
|
||||
mlx5_set_umr_free_mkey(pd, in, ndescs, access_mode, page_shift);
|
||||
|
||||
err = mlx5_core_create_mkey(dev->mdev, &mr->mmkey, in, inlen);
|
||||
err = mlx5_ib_create_mkey(dev, &mr->mmkey, in, inlen);
|
||||
if (err)
|
||||
goto err_free_descs;
|
||||
|
||||
@ -1905,7 +1940,7 @@ struct ib_mw *mlx5_ib_alloc_mw(struct ib_pd *pd, enum ib_mw_type type,
|
||||
MLX5_SET(mkc, mkc, en_rinval, !!((type == IB_MW_TYPE_2)));
|
||||
MLX5_SET(mkc, mkc, qpn, 0xffffff);
|
||||
|
||||
err = mlx5_core_create_mkey(dev->mdev, &mw->mmkey, in, inlen);
|
||||
err = mlx5_ib_create_mkey(dev, &mw->mmkey, in, inlen);
|
||||
if (err)
|
||||
goto free;
|
||||
|
||||
|
@ -213,6 +213,8 @@ static int ipoib_get_link_ksettings(struct net_device *netdev,
|
||||
}
|
||||
|
||||
static const struct ethtool_ops ipoib_ethtool_ops = {
|
||||
.supported_coalesce_params = ETHTOOL_COALESCE_RX_USECS |
|
||||
ETHTOOL_COALESCE_RX_MAX_FRAMES,
|
||||
.get_link_ksettings = ipoib_get_link_ksettings,
|
||||
.get_drvinfo = ipoib_get_drvinfo,
|
||||
.get_coalesce = ipoib_get_coalesce,
|
||||
|
@ -259,6 +259,19 @@ config GENEVE
|
||||
To compile this driver as a module, choose M here: the module
|
||||
will be called geneve.
|
||||
|
||||
config BAREUDP
|
||||
tristate "Bare UDP Encapsulation"
|
||||
depends on INET
|
||||
depends on IPV6 || !IPV6
|
||||
select NET_UDP_TUNNEL
|
||||
select GRO_CELLS
|
||||
help
|
||||
This adds a bare UDP tunnel module for tunnelling different
|
||||
kinds of traffic like MPLS, IP, etc. inside a UDP tunnel.
|
||||
|
||||
To compile this driver as a module, choose M here: the module
|
||||
will be called bareudp.
|
||||
|
||||
config GTP
|
||||
tristate "GPRS Tunneling Protocol datapath (GTP-U)"
|
||||
depends on INET
|
||||
@ -432,6 +445,8 @@ source "drivers/net/fddi/Kconfig"
|
||||
|
||||
source "drivers/net/hippi/Kconfig"
|
||||
|
||||
source "drivers/net/ipa/Kconfig"
|
||||
|
||||
config NET_SB1000
|
||||
tristate "General Instruments Surfboard 1000"
|
||||
depends on PNP
|
||||
|
@ -29,6 +29,7 @@ obj-$(CONFIG_VETH) += veth.o
|
||||
obj-$(CONFIG_VIRTIO_NET) += virtio_net.o
|
||||
obj-$(CONFIG_VXLAN) += vxlan.o
|
||||
obj-$(CONFIG_GENEVE) += geneve.o
|
||||
obj-$(CONFIG_BAREUDP) += bareudp.o
|
||||
obj-$(CONFIG_GTP) += gtp.o
|
||||
obj-$(CONFIG_NLMON) += nlmon.o
|
||||
obj-$(CONFIG_NET_VRF) += vrf.o
|
||||
@ -46,6 +47,7 @@ obj-$(CONFIG_ETHERNET) += ethernet/
|
||||
obj-$(CONFIG_FDDI) += fddi/
|
||||
obj-$(CONFIG_HIPPI) += hippi/
|
||||
obj-$(CONFIG_HAMRADIO) += hamradio/
|
||||
obj-$(CONFIG_QCOM_IPA) += ipa/
|
||||
obj-$(CONFIG_PLIP) += plip/
|
||||
obj-$(CONFIG_PPP) += ppp/
|
||||
obj-$(CONFIG_PPP_ASYNC) += ppp/
|
||||
|
817
drivers/net/bareudp.c
Normal file
817
drivers/net/bareudp.c
Normal file
@ -0,0 +1,817 @@
|
||||
// SPDX-License-Identifier: GPL-2.0
|
||||
/* Bareudp: UDP tunnel encasulation for different Payload types like
|
||||
* MPLS, NSH, IP, etc.
|
||||
* Copyright (c) 2019 Nokia, Inc.
|
||||
* Authors: Martin Varghese, <martin.varghese@nokia.com>
|
||||
*/
|
||||
|
||||
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
||||
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/etherdevice.h>
|
||||
#include <linux/hash.h>
|
||||
#include <net/dst_metadata.h>
|
||||
#include <net/gro_cells.h>
|
||||
#include <net/rtnetlink.h>
|
||||
#include <net/protocol.h>
|
||||
#include <net/ip6_tunnel.h>
|
||||
#include <net/ip_tunnels.h>
|
||||
#include <net/udp_tunnel.h>
|
||||
#include <net/bareudp.h>
|
||||
|
||||
#define BAREUDP_BASE_HLEN sizeof(struct udphdr)
|
||||
#define BAREUDP_IPV4_HLEN (sizeof(struct iphdr) + \
|
||||
sizeof(struct udphdr))
|
||||
#define BAREUDP_IPV6_HLEN (sizeof(struct ipv6hdr) + \
|
||||
sizeof(struct udphdr))
|
||||
|
||||
static bool log_ecn_error = true;
|
||||
module_param(log_ecn_error, bool, 0644);
|
||||
MODULE_PARM_DESC(log_ecn_error, "Log packets received with corrupted ECN");
|
||||
|
||||
/* per-network namespace private data for this module */
|
||||
|
||||
static unsigned int bareudp_net_id;
|
||||
|
||||
struct bareudp_net {
|
||||
struct list_head bareudp_list;
|
||||
};
|
||||
|
||||
/* Pseudo network device */
|
||||
struct bareudp_dev {
|
||||
struct net *net; /* netns for packet i/o */
|
||||
struct net_device *dev; /* netdev for bareudp tunnel */
|
||||
__be16 ethertype;
|
||||
__be16 port;
|
||||
u16 sport_min;
|
||||
bool multi_proto_mode;
|
||||
struct socket __rcu *sock;
|
||||
struct list_head next; /* bareudp node on namespace list */
|
||||
struct gro_cells gro_cells;
|
||||
};
|
||||
|
||||
static int bareudp_udp_encap_recv(struct sock *sk, struct sk_buff *skb)
|
||||
{
|
||||
struct metadata_dst *tun_dst = NULL;
|
||||
struct pcpu_sw_netstats *stats;
|
||||
struct bareudp_dev *bareudp;
|
||||
unsigned short family;
|
||||
unsigned int len;
|
||||
__be16 proto;
|
||||
void *oiph;
|
||||
int err;
|
||||
|
||||
bareudp = rcu_dereference_sk_user_data(sk);
|
||||
if (!bareudp)
|
||||
goto drop;
|
||||
|
||||
if (skb->protocol == htons(ETH_P_IP))
|
||||
family = AF_INET;
|
||||
else
|
||||
family = AF_INET6;
|
||||
|
||||
if (bareudp->ethertype == htons(ETH_P_IP)) {
|
||||
struct iphdr *iphdr;
|
||||
|
||||
iphdr = (struct iphdr *)(skb->data + BAREUDP_BASE_HLEN);
|
||||
if (iphdr->version == 4) {
|
||||
proto = bareudp->ethertype;
|
||||
} else if (bareudp->multi_proto_mode && (iphdr->version == 6)) {
|
||||
proto = htons(ETH_P_IPV6);
|
||||
} else {
|
||||
bareudp->dev->stats.rx_dropped++;
|
||||
goto drop;
|
||||
}
|
||||
} else if (bareudp->ethertype == htons(ETH_P_MPLS_UC)) {
|
||||
struct iphdr *tunnel_hdr;
|
||||
|
||||
tunnel_hdr = (struct iphdr *)skb_network_header(skb);
|
||||
if (tunnel_hdr->version == 4) {
|
||||
if (!ipv4_is_multicast(tunnel_hdr->daddr)) {
|
||||
proto = bareudp->ethertype;
|
||||
} else if (bareudp->multi_proto_mode &&
|
||||
ipv4_is_multicast(tunnel_hdr->daddr)) {
|
||||
proto = htons(ETH_P_MPLS_MC);
|
||||
} else {
|
||||
bareudp->dev->stats.rx_dropped++;
|
||||
goto drop;
|
||||
}
|
||||
} else {
|
||||
int addr_type;
|
||||
struct ipv6hdr *tunnel_hdr_v6;
|
||||
|
||||
tunnel_hdr_v6 = (struct ipv6hdr *)skb_network_header(skb);
|
||||
addr_type =
|
||||
ipv6_addr_type((struct in6_addr *)&tunnel_hdr_v6->daddr);
|
||||
if (!(addr_type & IPV6_ADDR_MULTICAST)) {
|
||||
proto = bareudp->ethertype;
|
||||
} else if (bareudp->multi_proto_mode &&
|
||||
(addr_type & IPV6_ADDR_MULTICAST)) {
|
||||
proto = htons(ETH_P_MPLS_MC);
|
||||
} else {
|
||||
bareudp->dev->stats.rx_dropped++;
|
||||
goto drop;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
proto = bareudp->ethertype;
|
||||
}
|
||||
|
||||
if (iptunnel_pull_header(skb, BAREUDP_BASE_HLEN,
|
||||
proto,
|
||||
!net_eq(bareudp->net,
|
||||
dev_net(bareudp->dev)))) {
|
||||
bareudp->dev->stats.rx_dropped++;
|
||||
goto drop;
|
||||
}
|
||||
|
||||
tun_dst = udp_tun_rx_dst(skb, family, TUNNEL_KEY, 0, 0);
|
||||
if (!tun_dst) {
|
||||
bareudp->dev->stats.rx_dropped++;
|
||||
goto drop;
|
||||
}
|
||||
skb_dst_set(skb, &tun_dst->dst);
|
||||
skb->dev = bareudp->dev;
|
||||
oiph = skb_network_header(skb);
|
||||
skb_reset_network_header(skb);
|
||||
|
||||
if (family == AF_INET)
|
||||
err = IP_ECN_decapsulate(oiph, skb);
|
||||
#if IS_ENABLED(CONFIG_IPV6)
|
||||
else
|
||||
err = IP6_ECN_decapsulate(oiph, skb);
|
||||
#endif
|
||||
|
||||
if (unlikely(err)) {
|
||||
if (log_ecn_error) {
|
||||
if (family == AF_INET)
|
||||
net_info_ratelimited("non-ECT from %pI4 "
|
||||
"with TOS=%#x\n",
|
||||
&((struct iphdr *)oiph)->saddr,
|
||||
((struct iphdr *)oiph)->tos);
|
||||
#if IS_ENABLED(CONFIG_IPV6)
|
||||
else
|
||||
net_info_ratelimited("non-ECT from %pI6\n",
|
||||
&((struct ipv6hdr *)oiph)->saddr);
|
||||
#endif
|
||||
}
|
||||
if (err > 1) {
|
||||
++bareudp->dev->stats.rx_frame_errors;
|
||||
++bareudp->dev->stats.rx_errors;
|
||||
goto drop;
|
||||
}
|
||||
}
|
||||
|
||||
len = skb->len;
|
||||
err = gro_cells_receive(&bareudp->gro_cells, skb);
|
||||
if (likely(err == NET_RX_SUCCESS)) {
|
||||
stats = this_cpu_ptr(bareudp->dev->tstats);
|
||||
u64_stats_update_begin(&stats->syncp);
|
||||
stats->rx_packets++;
|
||||
stats->rx_bytes += len;
|
||||
u64_stats_update_end(&stats->syncp);
|
||||
}
|
||||
return 0;
|
||||
drop:
|
||||
/* Consume bad packet */
|
||||
kfree_skb(skb);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bareudp_err_lookup(struct sock *sk, struct sk_buff *skb)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bareudp_init(struct net_device *dev)
|
||||
{
|
||||
struct bareudp_dev *bareudp = netdev_priv(dev);
|
||||
int err;
|
||||
|
||||
dev->tstats = netdev_alloc_pcpu_stats(struct pcpu_sw_netstats);
|
||||
if (!dev->tstats)
|
||||
return -ENOMEM;
|
||||
|
||||
err = gro_cells_init(&bareudp->gro_cells, dev);
|
||||
if (err) {
|
||||
free_percpu(dev->tstats);
|
||||
return err;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void bareudp_uninit(struct net_device *dev)
|
||||
{
|
||||
struct bareudp_dev *bareudp = netdev_priv(dev);
|
||||
|
||||
gro_cells_destroy(&bareudp->gro_cells);
|
||||
free_percpu(dev->tstats);
|
||||
}
|
||||
|
||||
static struct socket *bareudp_create_sock(struct net *net, __be16 port)
|
||||
{
|
||||
struct udp_port_cfg udp_conf;
|
||||
struct socket *sock;
|
||||
int err;
|
||||
|
||||
memset(&udp_conf, 0, sizeof(udp_conf));
|
||||
#if IS_ENABLED(CONFIG_IPV6)
|
||||
udp_conf.family = AF_INET6;
|
||||
#else
|
||||
udp_conf.family = AF_INET;
|
||||
#endif
|
||||
udp_conf.local_udp_port = port;
|
||||
/* Open UDP socket */
|
||||
err = udp_sock_create(net, &udp_conf, &sock);
|
||||
if (err < 0)
|
||||
return ERR_PTR(err);
|
||||
|
||||
return sock;
|
||||
}
|
||||
|
||||
/* Create new listen socket if needed */
|
||||
static int bareudp_socket_create(struct bareudp_dev *bareudp, __be16 port)
|
||||
{
|
||||
struct udp_tunnel_sock_cfg tunnel_cfg;
|
||||
struct socket *sock;
|
||||
|
||||
sock = bareudp_create_sock(bareudp->net, port);
|
||||
if (IS_ERR(sock))
|
||||
return PTR_ERR(sock);
|
||||
|
||||
/* Mark socket as an encapsulation socket */
|
||||
memset(&tunnel_cfg, 0, sizeof(tunnel_cfg));
|
||||
tunnel_cfg.sk_user_data = bareudp;
|
||||
tunnel_cfg.encap_type = 1;
|
||||
tunnel_cfg.encap_rcv = bareudp_udp_encap_recv;
|
||||
tunnel_cfg.encap_err_lookup = bareudp_err_lookup;
|
||||
tunnel_cfg.encap_destroy = NULL;
|
||||
setup_udp_tunnel_sock(bareudp->net, sock, &tunnel_cfg);
|
||||
|
||||
/* As the setup_udp_tunnel_sock does not call udp_encap_enable if the
|
||||
* socket type is v6 an explicit call to udp_encap_enable is needed.
|
||||
*/
|
||||
if (sock->sk->sk_family == AF_INET6)
|
||||
udp_encap_enable();
|
||||
|
||||
rcu_assign_pointer(bareudp->sock, sock);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bareudp_open(struct net_device *dev)
|
||||
{
|
||||
struct bareudp_dev *bareudp = netdev_priv(dev);
|
||||
int ret = 0;
|
||||
|
||||
ret = bareudp_socket_create(bareudp, bareudp->port);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void bareudp_sock_release(struct bareudp_dev *bareudp)
|
||||
{
|
||||
struct socket *sock;
|
||||
|
||||
sock = bareudp->sock;
|
||||
rcu_assign_pointer(bareudp->sock, NULL);
|
||||
synchronize_net();
|
||||
udp_tunnel_sock_release(sock);
|
||||
}
|
||||
|
||||
static int bareudp_stop(struct net_device *dev)
|
||||
{
|
||||
struct bareudp_dev *bareudp = netdev_priv(dev);
|
||||
|
||||
bareudp_sock_release(bareudp);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bareudp_xmit_skb(struct sk_buff *skb, struct net_device *dev,
|
||||
struct bareudp_dev *bareudp,
|
||||
const struct ip_tunnel_info *info)
|
||||
{
|
||||
bool xnet = !net_eq(bareudp->net, dev_net(bareudp->dev));
|
||||
bool use_cache = ip_tunnel_dst_cache_usable(skb, info);
|
||||
struct socket *sock = rcu_dereference(bareudp->sock);
|
||||
bool udp_sum = !!(info->key.tun_flags & TUNNEL_CSUM);
|
||||
const struct ip_tunnel_key *key = &info->key;
|
||||
struct rtable *rt;
|
||||
__be16 sport, df;
|
||||
int min_headroom;
|
||||
__u8 tos, ttl;
|
||||
__be32 saddr;
|
||||
int err;
|
||||
|
||||
if (!sock)
|
||||
return -ESHUTDOWN;
|
||||
|
||||
rt = ip_route_output_tunnel(skb, dev, bareudp->net, &saddr, info,
|
||||
IPPROTO_UDP, use_cache);
|
||||
|
||||
if (IS_ERR(rt))
|
||||
return PTR_ERR(rt);
|
||||
|
||||
skb_tunnel_check_pmtu(skb, &rt->dst,
|
||||
BAREUDP_IPV4_HLEN + info->options_len);
|
||||
|
||||
sport = udp_flow_src_port(bareudp->net, skb,
|
||||
bareudp->sport_min, USHRT_MAX,
|
||||
true);
|
||||
tos = ip_tunnel_ecn_encap(key->tos, ip_hdr(skb), skb);
|
||||
ttl = key->ttl;
|
||||
df = key->tun_flags & TUNNEL_DONT_FRAGMENT ? htons(IP_DF) : 0;
|
||||
skb_scrub_packet(skb, xnet);
|
||||
|
||||
err = -ENOSPC;
|
||||
if (!skb_pull(skb, skb_network_offset(skb)))
|
||||
goto free_dst;
|
||||
|
||||
min_headroom = LL_RESERVED_SPACE(rt->dst.dev) + rt->dst.header_len +
|
||||
BAREUDP_BASE_HLEN + info->options_len + sizeof(struct iphdr);
|
||||
|
||||
err = skb_cow_head(skb, min_headroom);
|
||||
if (unlikely(err))
|
||||
goto free_dst;
|
||||
|
||||
err = udp_tunnel_handle_offloads(skb, udp_sum);
|
||||
if (err)
|
||||
goto free_dst;
|
||||
|
||||
skb_set_inner_protocol(skb, bareudp->ethertype);
|
||||
udp_tunnel_xmit_skb(rt, sock->sk, skb, saddr, info->key.u.ipv4.dst,
|
||||
tos, ttl, df, sport, bareudp->port,
|
||||
!net_eq(bareudp->net, dev_net(bareudp->dev)),
|
||||
!(info->key.tun_flags & TUNNEL_CSUM));
|
||||
return 0;
|
||||
|
||||
free_dst:
|
||||
dst_release(&rt->dst);
|
||||
return err;
|
||||
}
|
||||
|
||||
#if IS_ENABLED(CONFIG_IPV6)
|
||||
static int bareudp6_xmit_skb(struct sk_buff *skb, struct net_device *dev,
|
||||
struct bareudp_dev *bareudp,
|
||||
const struct ip_tunnel_info *info)
|
||||
{
|
||||
bool xnet = !net_eq(bareudp->net, dev_net(bareudp->dev));
|
||||
bool use_cache = ip_tunnel_dst_cache_usable(skb, info);
|
||||
struct socket *sock = rcu_dereference(bareudp->sock);
|
||||
bool udp_sum = !!(info->key.tun_flags & TUNNEL_CSUM);
|
||||
const struct ip_tunnel_key *key = &info->key;
|
||||
struct dst_entry *dst = NULL;
|
||||
struct in6_addr saddr, daddr;
|
||||
int min_headroom;
|
||||
__u8 prio, ttl;
|
||||
__be16 sport;
|
||||
int err;
|
||||
|
||||
if (!sock)
|
||||
return -ESHUTDOWN;
|
||||
|
||||
dst = ip6_dst_lookup_tunnel(skb, dev, bareudp->net, sock, &saddr, info,
|
||||
IPPROTO_UDP, use_cache);
|
||||
if (IS_ERR(dst))
|
||||
return PTR_ERR(dst);
|
||||
|
||||
skb_tunnel_check_pmtu(skb, dst, BAREUDP_IPV6_HLEN + info->options_len);
|
||||
|
||||
sport = udp_flow_src_port(bareudp->net, skb,
|
||||
bareudp->sport_min, USHRT_MAX,
|
||||
true);
|
||||
prio = ip_tunnel_ecn_encap(key->tos, ip_hdr(skb), skb);
|
||||
ttl = key->ttl;
|
||||
|
||||
skb_scrub_packet(skb, xnet);
|
||||
|
||||
err = -ENOSPC;
|
||||
if (!skb_pull(skb, skb_network_offset(skb)))
|
||||
goto free_dst;
|
||||
|
||||
min_headroom = LL_RESERVED_SPACE(dst->dev) + dst->header_len +
|
||||
BAREUDP_BASE_HLEN + info->options_len + sizeof(struct iphdr);
|
||||
|
||||
err = skb_cow_head(skb, min_headroom);
|
||||
if (unlikely(err))
|
||||
goto free_dst;
|
||||
|
||||
err = udp_tunnel_handle_offloads(skb, udp_sum);
|
||||
if (err)
|
||||
goto free_dst;
|
||||
|
||||
daddr = info->key.u.ipv6.dst;
|
||||
udp_tunnel6_xmit_skb(dst, sock->sk, skb, dev,
|
||||
&saddr, &daddr, prio, ttl,
|
||||
info->key.label, sport, bareudp->port,
|
||||
!(info->key.tun_flags & TUNNEL_CSUM));
|
||||
return 0;
|
||||
|
||||
free_dst:
|
||||
dst_release(dst);
|
||||
return err;
|
||||
}
|
||||
#endif
|
||||
|
||||
static netdev_tx_t bareudp_xmit(struct sk_buff *skb, struct net_device *dev)
|
||||
{
|
||||
struct bareudp_dev *bareudp = netdev_priv(dev);
|
||||
struct ip_tunnel_info *info = NULL;
|
||||
int err;
|
||||
|
||||
if (skb->protocol != bareudp->ethertype) {
|
||||
if (!bareudp->multi_proto_mode ||
|
||||
(skb->protocol != htons(ETH_P_MPLS_MC) &&
|
||||
skb->protocol != htons(ETH_P_IPV6))) {
|
||||
err = -EINVAL;
|
||||
goto tx_error;
|
||||
}
|
||||
}
|
||||
|
||||
info = skb_tunnel_info(skb);
|
||||
if (unlikely(!info || !(info->mode & IP_TUNNEL_INFO_TX))) {
|
||||
err = -EINVAL;
|
||||
goto tx_error;
|
||||
}
|
||||
|
||||
rcu_read_lock();
|
||||
#if IS_ENABLED(CONFIG_IPV6)
|
||||
if (info->mode & IP_TUNNEL_INFO_IPV6)
|
||||
err = bareudp6_xmit_skb(skb, dev, bareudp, info);
|
||||
else
|
||||
#endif
|
||||
err = bareudp_xmit_skb(skb, dev, bareudp, info);
|
||||
|
||||
rcu_read_unlock();
|
||||
|
||||
if (likely(!err))
|
||||
return NETDEV_TX_OK;
|
||||
tx_error:
|
||||
dev_kfree_skb(skb);
|
||||
|
||||
if (err == -ELOOP)
|
||||
dev->stats.collisions++;
|
||||
else if (err == -ENETUNREACH)
|
||||
dev->stats.tx_carrier_errors++;
|
||||
|
||||
dev->stats.tx_errors++;
|
||||
return NETDEV_TX_OK;
|
||||
}
|
||||
|
||||
static int bareudp_fill_metadata_dst(struct net_device *dev,
|
||||
struct sk_buff *skb)
|
||||
{
|
||||
struct ip_tunnel_info *info = skb_tunnel_info(skb);
|
||||
struct bareudp_dev *bareudp = netdev_priv(dev);
|
||||
bool use_cache;
|
||||
|
||||
use_cache = ip_tunnel_dst_cache_usable(skb, info);
|
||||
|
||||
if (ip_tunnel_info_af(info) == AF_INET) {
|
||||
struct rtable *rt;
|
||||
__be32 saddr;
|
||||
|
||||
rt = ip_route_output_tunnel(skb, dev, bareudp->net, &saddr,
|
||||
info, IPPROTO_UDP, use_cache);
|
||||
if (IS_ERR(rt))
|
||||
return PTR_ERR(rt);
|
||||
|
||||
ip_rt_put(rt);
|
||||
info->key.u.ipv4.src = saddr;
|
||||
#if IS_ENABLED(CONFIG_IPV6)
|
||||
} else if (ip_tunnel_info_af(info) == AF_INET6) {
|
||||
struct dst_entry *dst;
|
||||
struct in6_addr saddr;
|
||||
struct socket *sock = rcu_dereference(bareudp->sock);
|
||||
|
||||
dst = ip6_dst_lookup_tunnel(skb, dev, bareudp->net, sock,
|
||||
&saddr, info, IPPROTO_UDP,
|
||||
use_cache);
|
||||
if (IS_ERR(dst))
|
||||
return PTR_ERR(dst);
|
||||
|
||||
dst_release(dst);
|
||||
info->key.u.ipv6.src = saddr;
|
||||
#endif
|
||||
} else {
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
info->key.tp_src = udp_flow_src_port(bareudp->net, skb,
|
||||
bareudp->sport_min,
|
||||
USHRT_MAX, true);
|
||||
info->key.tp_dst = bareudp->port;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct net_device_ops bareudp_netdev_ops = {
|
||||
.ndo_init = bareudp_init,
|
||||
.ndo_uninit = bareudp_uninit,
|
||||
.ndo_open = bareudp_open,
|
||||
.ndo_stop = bareudp_stop,
|
||||
.ndo_start_xmit = bareudp_xmit,
|
||||
.ndo_get_stats64 = ip_tunnel_get_stats64,
|
||||
.ndo_fill_metadata_dst = bareudp_fill_metadata_dst,
|
||||
};
|
||||
|
||||
static const struct nla_policy bareudp_policy[IFLA_BAREUDP_MAX + 1] = {
|
||||
[IFLA_BAREUDP_PORT] = { .type = NLA_U16 },
|
||||
[IFLA_BAREUDP_ETHERTYPE] = { .type = NLA_U16 },
|
||||
[IFLA_BAREUDP_SRCPORT_MIN] = { .type = NLA_U16 },
|
||||
[IFLA_BAREUDP_MULTIPROTO_MODE] = { .type = NLA_FLAG },
|
||||
};
|
||||
|
||||
/* Info for udev, that this is a virtual tunnel endpoint */
|
||||
static struct device_type bareudp_type = {
|
||||
.name = "bareudp",
|
||||
};
|
||||
|
||||
/* Initialize the device structure. */
|
||||
static void bareudp_setup(struct net_device *dev)
|
||||
{
|
||||
dev->netdev_ops = &bareudp_netdev_ops;
|
||||
dev->needs_free_netdev = true;
|
||||
SET_NETDEV_DEVTYPE(dev, &bareudp_type);
|
||||
dev->features |= NETIF_F_SG | NETIF_F_HW_CSUM;
|
||||
dev->features |= NETIF_F_RXCSUM;
|
||||
dev->features |= NETIF_F_GSO_SOFTWARE;
|
||||
dev->hw_features |= NETIF_F_SG | NETIF_F_HW_CSUM | NETIF_F_RXCSUM;
|
||||
dev->hw_features |= NETIF_F_GSO_SOFTWARE;
|
||||
dev->hard_header_len = 0;
|
||||
dev->addr_len = 0;
|
||||
dev->mtu = ETH_DATA_LEN;
|
||||
dev->min_mtu = IPV4_MIN_MTU;
|
||||
dev->max_mtu = IP_MAX_MTU - BAREUDP_BASE_HLEN;
|
||||
dev->type = ARPHRD_NONE;
|
||||
netif_keep_dst(dev);
|
||||
dev->priv_flags |= IFF_NO_QUEUE;
|
||||
dev->flags = IFF_POINTOPOINT | IFF_NOARP | IFF_MULTICAST;
|
||||
}
|
||||
|
||||
static int bareudp_validate(struct nlattr *tb[], struct nlattr *data[],
|
||||
struct netlink_ext_ack *extack)
|
||||
{
|
||||
if (!data) {
|
||||
NL_SET_ERR_MSG(extack,
|
||||
"Not enough attributes provided to perform the operation");
|
||||
return -EINVAL;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bareudp2info(struct nlattr *data[], struct bareudp_conf *conf,
|
||||
struct netlink_ext_ack *extack)
|
||||
{
|
||||
if (!data[IFLA_BAREUDP_PORT]) {
|
||||
NL_SET_ERR_MSG(extack, "port not specified");
|
||||
return -EINVAL;
|
||||
}
|
||||
if (!data[IFLA_BAREUDP_ETHERTYPE]) {
|
||||
NL_SET_ERR_MSG(extack, "ethertype not specified");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (data[IFLA_BAREUDP_PORT])
|
||||
conf->port = nla_get_u16(data[IFLA_BAREUDP_PORT]);
|
||||
|
||||
if (data[IFLA_BAREUDP_ETHERTYPE])
|
||||
conf->ethertype = nla_get_u16(data[IFLA_BAREUDP_ETHERTYPE]);
|
||||
|
||||
if (data[IFLA_BAREUDP_SRCPORT_MIN])
|
||||
conf->sport_min = nla_get_u16(data[IFLA_BAREUDP_SRCPORT_MIN]);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct bareudp_dev *bareudp_find_dev(struct bareudp_net *bn,
|
||||
const struct bareudp_conf *conf)
|
||||
{
|
||||
struct bareudp_dev *bareudp, *t = NULL;
|
||||
|
||||
list_for_each_entry(bareudp, &bn->bareudp_list, next) {
|
||||
if (conf->port == bareudp->port)
|
||||
t = bareudp;
|
||||
}
|
||||
return t;
|
||||
}
|
||||
|
||||
static int bareudp_configure(struct net *net, struct net_device *dev,
|
||||
struct bareudp_conf *conf)
|
||||
{
|
||||
struct bareudp_net *bn = net_generic(net, bareudp_net_id);
|
||||
struct bareudp_dev *t, *bareudp = netdev_priv(dev);
|
||||
int err;
|
||||
|
||||
bareudp->net = net;
|
||||
bareudp->dev = dev;
|
||||
t = bareudp_find_dev(bn, conf);
|
||||
if (t)
|
||||
return -EBUSY;
|
||||
|
||||
if (conf->multi_proto_mode &&
|
||||
(conf->ethertype != htons(ETH_P_MPLS_UC) &&
|
||||
conf->ethertype != htons(ETH_P_IP)))
|
||||
return -EINVAL;
|
||||
|
||||
bareudp->port = conf->port;
|
||||
bareudp->ethertype = conf->ethertype;
|
||||
bareudp->sport_min = conf->sport_min;
|
||||
bareudp->multi_proto_mode = conf->multi_proto_mode;
|
||||
err = register_netdevice(dev);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
list_add(&bareudp->next, &bn->bareudp_list);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bareudp_link_config(struct net_device *dev,
|
||||
struct nlattr *tb[])
|
||||
{
|
||||
int err;
|
||||
|
||||
if (tb[IFLA_MTU]) {
|
||||
err = dev_set_mtu(dev, nla_get_u32(tb[IFLA_MTU]));
|
||||
if (err)
|
||||
return err;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bareudp_newlink(struct net *net, struct net_device *dev,
|
||||
struct nlattr *tb[], struct nlattr *data[],
|
||||
struct netlink_ext_ack *extack)
|
||||
{
|
||||
struct bareudp_conf conf;
|
||||
int err;
|
||||
|
||||
err = bareudp2info(data, &conf, extack);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
err = bareudp_configure(net, dev, &conf);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
err = bareudp_link_config(dev, tb);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void bareudp_dellink(struct net_device *dev, struct list_head *head)
|
||||
{
|
||||
struct bareudp_dev *bareudp = netdev_priv(dev);
|
||||
|
||||
list_del(&bareudp->next);
|
||||
unregister_netdevice_queue(dev, head);
|
||||
}
|
||||
|
||||
static size_t bareudp_get_size(const struct net_device *dev)
|
||||
{
|
||||
return nla_total_size(sizeof(__be16)) + /* IFLA_BAREUDP_PORT */
|
||||
nla_total_size(sizeof(__be16)) + /* IFLA_BAREUDP_ETHERTYPE */
|
||||
nla_total_size(sizeof(__u16)) + /* IFLA_BAREUDP_SRCPORT_MIN */
|
||||
nla_total_size(0) + /* IFLA_BAREUDP_MULTIPROTO_MODE */
|
||||
0;
|
||||
}
|
||||
|
||||
static int bareudp_fill_info(struct sk_buff *skb, const struct net_device *dev)
|
||||
{
|
||||
struct bareudp_dev *bareudp = netdev_priv(dev);
|
||||
|
||||
if (nla_put_be16(skb, IFLA_BAREUDP_PORT, bareudp->port))
|
||||
goto nla_put_failure;
|
||||
if (nla_put_be16(skb, IFLA_BAREUDP_ETHERTYPE, bareudp->ethertype))
|
||||
goto nla_put_failure;
|
||||
if (nla_put_u16(skb, IFLA_BAREUDP_SRCPORT_MIN, bareudp->sport_min))
|
||||
goto nla_put_failure;
|
||||
if (bareudp->multi_proto_mode &&
|
||||
nla_put_flag(skb, IFLA_BAREUDP_MULTIPROTO_MODE))
|
||||
goto nla_put_failure;
|
||||
|
||||
return 0;
|
||||
|
||||
nla_put_failure:
|
||||
return -EMSGSIZE;
|
||||
}
|
||||
|
||||
static struct rtnl_link_ops bareudp_link_ops __read_mostly = {
|
||||
.kind = "bareudp",
|
||||
.maxtype = IFLA_BAREUDP_MAX,
|
||||
.policy = bareudp_policy,
|
||||
.priv_size = sizeof(struct bareudp_dev),
|
||||
.setup = bareudp_setup,
|
||||
.validate = bareudp_validate,
|
||||
.newlink = bareudp_newlink,
|
||||
.dellink = bareudp_dellink,
|
||||
.get_size = bareudp_get_size,
|
||||
.fill_info = bareudp_fill_info,
|
||||
};
|
||||
|
||||
struct net_device *bareudp_dev_create(struct net *net, const char *name,
|
||||
u8 name_assign_type,
|
||||
struct bareudp_conf *conf)
|
||||
{
|
||||
struct nlattr *tb[IFLA_MAX + 1];
|
||||
struct net_device *dev;
|
||||
LIST_HEAD(list_kill);
|
||||
int err;
|
||||
|
||||
memset(tb, 0, sizeof(tb));
|
||||
dev = rtnl_create_link(net, name, name_assign_type,
|
||||
&bareudp_link_ops, tb, NULL);
|
||||
if (IS_ERR(dev))
|
||||
return dev;
|
||||
|
||||
err = bareudp_configure(net, dev, conf);
|
||||
if (err) {
|
||||
free_netdev(dev);
|
||||
return ERR_PTR(err);
|
||||
}
|
||||
err = dev_set_mtu(dev, IP_MAX_MTU - BAREUDP_BASE_HLEN);
|
||||
if (err)
|
||||
goto err;
|
||||
|
||||
err = rtnl_configure_link(dev, NULL);
|
||||
if (err < 0)
|
||||
goto err;
|
||||
|
||||
return dev;
|
||||
err:
|
||||
bareudp_dellink(dev, &list_kill);
|
||||
unregister_netdevice_many(&list_kill);
|
||||
return ERR_PTR(err);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(bareudp_dev_create);
|
||||
|
||||
static __net_init int bareudp_init_net(struct net *net)
|
||||
{
|
||||
struct bareudp_net *bn = net_generic(net, bareudp_net_id);
|
||||
|
||||
INIT_LIST_HEAD(&bn->bareudp_list);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void bareudp_destroy_tunnels(struct net *net, struct list_head *head)
|
||||
{
|
||||
struct bareudp_net *bn = net_generic(net, bareudp_net_id);
|
||||
struct bareudp_dev *bareudp, *next;
|
||||
|
||||
list_for_each_entry_safe(bareudp, next, &bn->bareudp_list, next)
|
||||
unregister_netdevice_queue(bareudp->dev, head);
|
||||
}
|
||||
|
||||
static void __net_exit bareudp_exit_batch_net(struct list_head *net_list)
|
||||
{
|
||||
struct net *net;
|
||||
LIST_HEAD(list);
|
||||
|
||||
rtnl_lock();
|
||||
list_for_each_entry(net, net_list, exit_list)
|
||||
bareudp_destroy_tunnels(net, &list);
|
||||
|
||||
/* unregister the devices gathered above */
|
||||
unregister_netdevice_many(&list);
|
||||
rtnl_unlock();
|
||||
}
|
||||
|
||||
static struct pernet_operations bareudp_net_ops = {
|
||||
.init = bareudp_init_net,
|
||||
.exit_batch = bareudp_exit_batch_net,
|
||||
.id = &bareudp_net_id,
|
||||
.size = sizeof(struct bareudp_net),
|
||||
};
|
||||
|
||||
static int __init bareudp_init_module(void)
|
||||
{
|
||||
int rc;
|
||||
|
||||
rc = register_pernet_subsys(&bareudp_net_ops);
|
||||
if (rc)
|
||||
goto out1;
|
||||
|
||||
rc = rtnl_link_register(&bareudp_link_ops);
|
||||
if (rc)
|
||||
goto out2;
|
||||
|
||||
return 0;
|
||||
out2:
|
||||
unregister_pernet_subsys(&bareudp_net_ops);
|
||||
out1:
|
||||
return rc;
|
||||
}
|
||||
late_initcall(bareudp_init_module);
|
||||
|
||||
static void __exit bareudp_cleanup_module(void)
|
||||
{
|
||||
rtnl_link_unregister(&bareudp_link_ops);
|
||||
unregister_pernet_subsys(&bareudp_net_ops);
|
||||
}
|
||||
module_exit(bareudp_cleanup_module);
|
||||
|
||||
MODULE_ALIAS_RTNL_LINK("bareudp");
|
||||
MODULE_LICENSE("GPL");
|
||||
MODULE_AUTHOR("Martin Varghese <martin.varghese@nokia.com>");
|
||||
MODULE_DESCRIPTION("Interface driver for UDP encapsulated traffic");
|
@ -1265,7 +1265,7 @@ static rx_handler_result_t bond_handle_frame(struct sk_buff **pskb)
|
||||
skb->dev = bond->dev;
|
||||
|
||||
if (BOND_MODE(bond) == BOND_MODE_ALB &&
|
||||
bond->dev->priv_flags & IFF_BRIDGE_PORT &&
|
||||
netif_is_bridge_port(bond->dev) &&
|
||||
skb->pkt_type == PACKET_HOST) {
|
||||
|
||||
if (unlikely(skb_cow_head(skb,
|
||||
@ -4370,7 +4370,6 @@ static void bond_ethtool_get_drvinfo(struct net_device *bond_dev,
|
||||
struct ethtool_drvinfo *drvinfo)
|
||||
{
|
||||
strlcpy(drvinfo->driver, DRV_NAME, sizeof(drvinfo->driver));
|
||||
strlcpy(drvinfo->version, DRV_VERSION, sizeof(drvinfo->version));
|
||||
snprintf(drvinfo->fw_version, sizeof(drvinfo->fw_version), "%d",
|
||||
BOND_ABI_VERSION);
|
||||
}
|
||||
@ -5008,8 +5007,6 @@ static int __init bonding_init(void)
|
||||
int i;
|
||||
int res;
|
||||
|
||||
pr_info("%s", bond_version);
|
||||
|
||||
res = bond_check_params(&bonding_defaults);
|
||||
if (res)
|
||||
goto out;
|
||||
@ -5064,6 +5061,5 @@ static void __exit bonding_exit(void)
|
||||
module_init(bonding_init);
|
||||
module_exit(bonding_exit);
|
||||
MODULE_LICENSE("GPL");
|
||||
MODULE_VERSION(DRV_VERSION);
|
||||
MODULE_DESCRIPTION(DRV_DESCRIPTION ", v" DRV_VERSION);
|
||||
MODULE_DESCRIPTION(DRV_DESCRIPTION);
|
||||
MODULE_AUTHOR("Thomas Davis, tadavis@lbl.gov and many others");
|
||||
|
@ -14,12 +14,11 @@
|
||||
|
||||
#ifndef _BONDING_PRIV_H
|
||||
#define _BONDING_PRIV_H
|
||||
#include <linux/vermagic.h>
|
||||
|
||||
#define DRV_VERSION "3.7.1"
|
||||
#define DRV_RELDATE "April 27, 2011"
|
||||
#define DRV_NAME "bonding"
|
||||
#define DRV_DESCRIPTION "Ethernet Channel Bonding Driver"
|
||||
|
||||
#define bond_version DRV_DESCRIPTION ": v" DRV_VERSION " (" DRV_RELDATE ")\n"
|
||||
#define bond_version DRV_DESCRIPTION ": v" UTS_RELEASE "\n"
|
||||
|
||||
#endif
|
||||
|
@ -348,11 +348,8 @@ static void slcan_write_wakeup(struct tty_struct *tty)
|
||||
|
||||
rcu_read_lock();
|
||||
sl = rcu_dereference(tty->disc_data);
|
||||
if (!sl)
|
||||
goto out;
|
||||
|
||||
schedule_work(&sl->tx_work);
|
||||
out:
|
||||
if (sl)
|
||||
schedule_work(&sl->tx_work);
|
||||
rcu_read_unlock();
|
||||
}
|
||||
|
||||
|
@ -681,7 +681,9 @@ int b53_configure_vlan(struct dsa_switch *ds)
|
||||
{
|
||||
struct b53_device *dev = ds->priv;
|
||||
struct b53_vlan vl = { 0 };
|
||||
struct b53_vlan *v;
|
||||
int i, def_vid;
|
||||
u16 vid;
|
||||
|
||||
def_vid = b53_default_pvid(dev);
|
||||
|
||||
@ -699,8 +701,18 @@ int b53_configure_vlan(struct dsa_switch *ds)
|
||||
b53_write16(dev, B53_VLAN_PAGE,
|
||||
B53_VLAN_PORT_DEF_TAG(i), def_vid);
|
||||
|
||||
if (!is5325(dev) && !is5365(dev))
|
||||
b53_set_jumbo(dev, dev->enable_jumbo, false);
|
||||
/* Upon initial call we have not set-up any VLANs, but upon
|
||||
* system resume, we need to restore all VLAN entries.
|
||||
*/
|
||||
for (vid = def_vid; vid < dev->num_vlans; vid++) {
|
||||
v = &dev->vlans[vid];
|
||||
|
||||
if (!v->members)
|
||||
continue;
|
||||
|
||||
b53_set_vlan_entry(dev, vid, v);
|
||||
b53_fast_age_vlan(dev, vid);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -807,8 +819,6 @@ static int b53_phy_write16(struct dsa_switch *ds, int addr, int reg, u16 val)
|
||||
static int b53_reset_switch(struct b53_device *priv)
|
||||
{
|
||||
/* reset vlans */
|
||||
priv->enable_jumbo = false;
|
||||
|
||||
memset(priv->vlans, 0, sizeof(*priv->vlans) * priv->num_vlans);
|
||||
memset(priv->ports, 0, sizeof(*priv->ports) * priv->num_ports);
|
||||
|
||||
@ -1289,7 +1299,9 @@ EXPORT_SYMBOL(b53_phylink_mac_link_down);
|
||||
void b53_phylink_mac_link_up(struct dsa_switch *ds, int port,
|
||||
unsigned int mode,
|
||||
phy_interface_t interface,
|
||||
struct phy_device *phydev)
|
||||
struct phy_device *phydev,
|
||||
int speed, int duplex,
|
||||
bool tx_pause, bool rx_pause)
|
||||
{
|
||||
struct b53_device *dev = ds->priv;
|
||||
|
||||
@ -1343,6 +1355,14 @@ int b53_vlan_prepare(struct dsa_switch *ds, int port,
|
||||
if ((is5325(dev) || is5365(dev)) && vlan->vid_begin == 0)
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
/* Port 7 on 7278 connects to the ASP's UniMAC which is not capable of
|
||||
* receiving VLAN tagged frames at all, we can still allow the port to
|
||||
* be configured for egress untagged.
|
||||
*/
|
||||
if (dev->chip_id == BCM7278_DEVICE_ID && port == 7 &&
|
||||
!(vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED))
|
||||
return -EINVAL;
|
||||
|
||||
if (vlan->vid_end > dev->num_vlans)
|
||||
return -ERANGE;
|
||||
|
||||
@ -1708,6 +1728,12 @@ int b53_br_join(struct dsa_switch *ds, int port, struct net_device *br)
|
||||
u16 pvlan, reg;
|
||||
unsigned int i;
|
||||
|
||||
/* On 7278, port 7 which connects to the ASP should only receive
|
||||
* traffic from matching CFP rules.
|
||||
*/
|
||||
if (dev->chip_id == BCM7278_DEVICE_ID && port == 7)
|
||||
return -EINVAL;
|
||||
|
||||
/* Make this port leave the all VLANs join since we will have proper
|
||||
* VLAN entries from now on
|
||||
*/
|
||||
@ -2063,6 +2089,26 @@ int b53_set_mac_eee(struct dsa_switch *ds, int port, struct ethtool_eee *e)
|
||||
}
|
||||
EXPORT_SYMBOL(b53_set_mac_eee);
|
||||
|
||||
static int b53_change_mtu(struct dsa_switch *ds, int port, int mtu)
|
||||
{
|
||||
struct b53_device *dev = ds->priv;
|
||||
bool enable_jumbo;
|
||||
bool allow_10_100;
|
||||
|
||||
if (is5325(dev) || is5365(dev))
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
enable_jumbo = (mtu >= JMS_MIN_SIZE);
|
||||
allow_10_100 = (dev->chip_id == BCM583XX_DEVICE_ID);
|
||||
|
||||
return b53_set_jumbo(dev, enable_jumbo, allow_10_100);
|
||||
}
|
||||
|
||||
static int b53_get_max_mtu(struct dsa_switch *ds, int port)
|
||||
{
|
||||
return JMS_MAX_SIZE;
|
||||
}
|
||||
|
||||
static const struct dsa_switch_ops b53_switch_ops = {
|
||||
.get_tag_protocol = b53_get_tag_protocol,
|
||||
.setup = b53_setup,
|
||||
@ -2100,6 +2146,8 @@ static const struct dsa_switch_ops b53_switch_ops = {
|
||||
.port_mdb_prepare = b53_mdb_prepare,
|
||||
.port_mdb_add = b53_mdb_add,
|
||||
.port_mdb_del = b53_mdb_del,
|
||||
.port_max_mtu = b53_get_max_mtu,
|
||||
.port_change_mtu = b53_change_mtu,
|
||||
};
|
||||
|
||||
struct b53_chip_data {
|
||||
|
@ -338,7 +338,9 @@ void b53_phylink_mac_link_down(struct dsa_switch *ds, int port,
|
||||
void b53_phylink_mac_link_up(struct dsa_switch *ds, int port,
|
||||
unsigned int mode,
|
||||
phy_interface_t interface,
|
||||
struct phy_device *phydev);
|
||||
struct phy_device *phydev,
|
||||
int speed, int duplex,
|
||||
bool tx_pause, bool rx_pause);
|
||||
int b53_vlan_filtering(struct dsa_switch *ds, int port, bool vlan_filtering);
|
||||
int b53_vlan_prepare(struct dsa_switch *ds, int port,
|
||||
const struct switchdev_obj_port_vlan *vlan);
|
||||
|
@ -178,9 +178,17 @@ static int bcm_sf2_port_setup(struct dsa_switch *ds, int port,
|
||||
core_writel(priv, reg, CORE_DIS_LEARN);
|
||||
|
||||
/* Enable Broadcom tags for that port if requested */
|
||||
if (priv->brcm_tag_mask & BIT(port))
|
||||
if (priv->brcm_tag_mask & BIT(port)) {
|
||||
b53_brcm_hdr_setup(ds, port);
|
||||
|
||||
/* Disable learning on ASP port */
|
||||
if (port == 7) {
|
||||
reg = core_readl(priv, CORE_DIS_LEARN);
|
||||
reg |= BIT(port);
|
||||
core_writel(priv, reg, CORE_DIS_LEARN);
|
||||
}
|
||||
}
|
||||
|
||||
/* Configure Traffic Class to QoS mapping, allow each priority to map
|
||||
* to a different queue number
|
||||
*/
|
||||
@ -648,7 +656,9 @@ static void bcm_sf2_sw_mac_link_down(struct dsa_switch *ds, int port,
|
||||
static void bcm_sf2_sw_mac_link_up(struct dsa_switch *ds, int port,
|
||||
unsigned int mode,
|
||||
phy_interface_t interface,
|
||||
struct phy_device *phydev)
|
||||
struct phy_device *phydev,
|
||||
int speed, int duplex,
|
||||
bool tx_pause, bool rx_pause)
|
||||
{
|
||||
struct bcm_sf2_priv *priv = bcm_sf2_to_priv(ds);
|
||||
struct ethtool_eee *p = &priv->dev->ports[port].eee;
|
||||
|
@ -13,6 +13,8 @@
|
||||
#include <net/dsa.h>
|
||||
#include <linux/bitmap.h>
|
||||
#include <net/flow_offload.h>
|
||||
#include <net/switchdev.h>
|
||||
#include <uapi/linux/if_bridge.h>
|
||||
|
||||
#include "bcm_sf2.h"
|
||||
#include "bcm_sf2_regs.h"
|
||||
@ -261,16 +263,27 @@ static int bcm_sf2_cfp_act_pol_set(struct bcm_sf2_priv *priv,
|
||||
static void bcm_sf2_cfp_slice_ipv4(struct bcm_sf2_priv *priv,
|
||||
struct flow_dissector_key_ipv4_addrs *addrs,
|
||||
struct flow_dissector_key_ports *ports,
|
||||
unsigned int slice_num,
|
||||
const __be16 vlan_tci,
|
||||
unsigned int slice_num, u8 num_udf,
|
||||
bool mask)
|
||||
{
|
||||
u32 reg, offset;
|
||||
|
||||
/* UDF_Valid[7:0] [31:24]
|
||||
* S-Tag [23:8]
|
||||
* C-Tag [7:0]
|
||||
*/
|
||||
reg = udf_lower_bits(num_udf) << 24 | be16_to_cpu(vlan_tci) >> 8;
|
||||
if (mask)
|
||||
core_writel(priv, reg, CORE_CFP_MASK_PORT(5));
|
||||
else
|
||||
core_writel(priv, reg, CORE_CFP_DATA_PORT(5));
|
||||
|
||||
/* C-Tag [31:24]
|
||||
* UDF_n_A8 [23:8]
|
||||
* UDF_n_A7 [7:0]
|
||||
*/
|
||||
reg = 0;
|
||||
reg = (u32)(be16_to_cpu(vlan_tci) & 0xff) << 24;
|
||||
if (mask)
|
||||
offset = CORE_CFP_MASK_PORT(4);
|
||||
else
|
||||
@ -336,6 +349,7 @@ static int bcm_sf2_cfp_ipv4_rule_set(struct bcm_sf2_priv *priv, int port,
|
||||
struct ethtool_rx_flow_spec *fs)
|
||||
{
|
||||
struct ethtool_rx_flow_spec_input input = {};
|
||||
__be16 vlan_tci = 0 , vlan_m_tci = 0xffff;
|
||||
const struct cfp_udf_layout *layout;
|
||||
unsigned int slice_num, rule_index;
|
||||
struct ethtool_rx_flow_rule *flow;
|
||||
@ -360,6 +374,12 @@ static int bcm_sf2_cfp_ipv4_rule_set(struct bcm_sf2_priv *priv, int port,
|
||||
|
||||
ip_frag = !!(be32_to_cpu(fs->h_ext.data[0]) & 1);
|
||||
|
||||
/* Extract VLAN TCI */
|
||||
if (fs->flow_type & FLOW_EXT) {
|
||||
vlan_tci = fs->h_ext.vlan_tci;
|
||||
vlan_m_tci = fs->m_ext.vlan_tci;
|
||||
}
|
||||
|
||||
/* Locate the first rule available */
|
||||
if (fs->location == RX_CLS_LOC_ANY)
|
||||
rule_index = find_first_zero_bit(priv->cfp.used,
|
||||
@ -421,18 +441,11 @@ static int bcm_sf2_cfp_ipv4_rule_set(struct bcm_sf2_priv *priv, int port,
|
||||
core_writel(priv, layout->udfs[slice_num].mask_value |
|
||||
udf_upper_bits(num_udf), CORE_CFP_MASK_PORT(6));
|
||||
|
||||
/* UDF_Valid[7:0] [31:24]
|
||||
* S-Tag [23:8]
|
||||
* C-Tag [7:0]
|
||||
*/
|
||||
core_writel(priv, udf_lower_bits(num_udf) << 24, CORE_CFP_DATA_PORT(5));
|
||||
|
||||
/* Mask all but valid UDFs */
|
||||
core_writel(priv, udf_lower_bits(num_udf) << 24, CORE_CFP_MASK_PORT(5));
|
||||
|
||||
/* Program the match and the mask */
|
||||
bcm_sf2_cfp_slice_ipv4(priv, ipv4.key, ports.key, slice_num, false);
|
||||
bcm_sf2_cfp_slice_ipv4(priv, ipv4.mask, ports.mask, SLICE_NUM_MASK, true);
|
||||
bcm_sf2_cfp_slice_ipv4(priv, ipv4.key, ports.key, vlan_tci,
|
||||
slice_num, num_udf, false);
|
||||
bcm_sf2_cfp_slice_ipv4(priv, ipv4.mask, ports.mask, vlan_m_tci,
|
||||
SLICE_NUM_MASK, num_udf, true);
|
||||
|
||||
/* Insert into TCAM now */
|
||||
bcm_sf2_cfp_rule_addr_set(priv, rule_index);
|
||||
@ -468,17 +481,29 @@ out_err_flow_rule:
|
||||
|
||||
static void bcm_sf2_cfp_slice_ipv6(struct bcm_sf2_priv *priv,
|
||||
const __be32 *ip6_addr, const __be16 port,
|
||||
unsigned int slice_num,
|
||||
const __be16 vlan_tci,
|
||||
unsigned int slice_num, u32 udf_bits,
|
||||
bool mask)
|
||||
{
|
||||
u32 reg, tmp, val, offset;
|
||||
|
||||
/* UDF_Valid[7:0] [31:24]
|
||||
* S-Tag [23:8]
|
||||
* C-Tag [7:0]
|
||||
*/
|
||||
reg = udf_bits << 24 | be16_to_cpu(vlan_tci) >> 8;
|
||||
if (mask)
|
||||
core_writel(priv, reg, CORE_CFP_MASK_PORT(5));
|
||||
else
|
||||
core_writel(priv, reg, CORE_CFP_DATA_PORT(5));
|
||||
|
||||
/* C-Tag [31:24]
|
||||
* UDF_n_B8 [23:8] (port)
|
||||
* UDF_n_B7 (upper) [7:0] (addr[15:8])
|
||||
*/
|
||||
reg = be32_to_cpu(ip6_addr[3]);
|
||||
val = (u32)be16_to_cpu(port) << 8 | ((reg >> 8) & 0xff);
|
||||
val |= (u32)(be16_to_cpu(vlan_tci) & 0xff) << 24;
|
||||
if (mask)
|
||||
offset = CORE_CFP_MASK_PORT(4);
|
||||
else
|
||||
@ -587,6 +612,11 @@ static int bcm_sf2_cfp_rule_cmp(struct bcm_sf2_priv *priv, int port,
|
||||
|
||||
ret = memcmp(&rule->fs.h_u, &fs->h_u, fs_size);
|
||||
ret |= memcmp(&rule->fs.m_u, &fs->m_u, fs_size);
|
||||
/* Compare VLAN TCI values as well */
|
||||
if (rule->fs.flow_type & FLOW_EXT) {
|
||||
ret |= rule->fs.h_ext.vlan_tci != fs->h_ext.vlan_tci;
|
||||
ret |= rule->fs.m_ext.vlan_tci != fs->m_ext.vlan_tci;
|
||||
}
|
||||
if (ret == 0)
|
||||
break;
|
||||
}
|
||||
@ -600,6 +630,7 @@ static int bcm_sf2_cfp_ipv6_rule_set(struct bcm_sf2_priv *priv, int port,
|
||||
struct ethtool_rx_flow_spec *fs)
|
||||
{
|
||||
struct ethtool_rx_flow_spec_input input = {};
|
||||
__be16 vlan_tci = 0, vlan_m_tci = 0xffff;
|
||||
unsigned int slice_num, rule_index[2];
|
||||
const struct cfp_udf_layout *layout;
|
||||
struct ethtool_rx_flow_rule *flow;
|
||||
@ -623,6 +654,12 @@ static int bcm_sf2_cfp_ipv6_rule_set(struct bcm_sf2_priv *priv, int port,
|
||||
|
||||
ip_frag = !!(be32_to_cpu(fs->h_ext.data[0]) & 1);
|
||||
|
||||
/* Extract VLAN TCI */
|
||||
if (fs->flow_type & FLOW_EXT) {
|
||||
vlan_tci = fs->h_ext.vlan_tci;
|
||||
vlan_m_tci = fs->m_ext.vlan_tci;
|
||||
}
|
||||
|
||||
layout = &udf_tcpip6_layout;
|
||||
slice_num = bcm_sf2_get_slice_number(layout, 0);
|
||||
if (slice_num == UDF_NUM_SLICES)
|
||||
@ -704,20 +741,13 @@ static int bcm_sf2_cfp_ipv6_rule_set(struct bcm_sf2_priv *priv, int port,
|
||||
reg = layout->udfs[slice_num].mask_value | udf_upper_bits(num_udf);
|
||||
core_writel(priv, reg, CORE_CFP_MASK_PORT(6));
|
||||
|
||||
/* UDF_Valid[7:0] [31:24]
|
||||
* S-Tag [23:8]
|
||||
* C-Tag [7:0]
|
||||
*/
|
||||
core_writel(priv, udf_lower_bits(num_udf) << 24, CORE_CFP_DATA_PORT(5));
|
||||
|
||||
/* Mask all but valid UDFs */
|
||||
core_writel(priv, udf_lower_bits(num_udf) << 24, CORE_CFP_MASK_PORT(5));
|
||||
|
||||
/* Slice the IPv6 source address and port */
|
||||
bcm_sf2_cfp_slice_ipv6(priv, ipv6.key->src.in6_u.u6_addr32,
|
||||
ports.key->src, slice_num, false);
|
||||
ports.key->src, vlan_tci, slice_num,
|
||||
udf_lower_bits(num_udf), false);
|
||||
bcm_sf2_cfp_slice_ipv6(priv, ipv6.mask->src.in6_u.u6_addr32,
|
||||
ports.mask->src, SLICE_NUM_MASK, true);
|
||||
ports.mask->src, vlan_m_tci, SLICE_NUM_MASK,
|
||||
udf_lower_bits(num_udf), true);
|
||||
|
||||
/* Insert into TCAM now because we need to insert a second rule */
|
||||
bcm_sf2_cfp_rule_addr_set(priv, rule_index[0]);
|
||||
@ -768,16 +798,12 @@ static int bcm_sf2_cfp_ipv6_rule_set(struct bcm_sf2_priv *priv, int port,
|
||||
udf_lower_bits(num_udf) << 8;
|
||||
core_writel(priv, reg, CORE_CFP_MASK_PORT(6));
|
||||
|
||||
/* Don't care */
|
||||
core_writel(priv, 0, CORE_CFP_DATA_PORT(5));
|
||||
|
||||
/* Mask all */
|
||||
core_writel(priv, 0, CORE_CFP_MASK_PORT(5));
|
||||
|
||||
bcm_sf2_cfp_slice_ipv6(priv, ipv6.key->dst.in6_u.u6_addr32,
|
||||
ports.key->dst, slice_num, false);
|
||||
ports.key->dst, 0, slice_num,
|
||||
0, false);
|
||||
bcm_sf2_cfp_slice_ipv6(priv, ipv6.mask->dst.in6_u.u6_addr32,
|
||||
ports.key->dst, SLICE_NUM_MASK, true);
|
||||
ports.key->dst, 0, SLICE_NUM_MASK,
|
||||
0, true);
|
||||
|
||||
/* Insert into TCAM now */
|
||||
bcm_sf2_cfp_rule_addr_set(priv, rule_index[1]);
|
||||
@ -823,7 +849,9 @@ static int bcm_sf2_cfp_rule_insert(struct dsa_switch *ds, int port,
|
||||
struct bcm_sf2_priv *priv = bcm_sf2_to_priv(ds);
|
||||
s8 cpu_port = dsa_to_port(ds, port)->cpu_dp->index;
|
||||
__u64 ring_cookie = fs->ring_cookie;
|
||||
struct switchdev_obj_port_vlan vlan;
|
||||
unsigned int queue_num, port_num;
|
||||
u16 vid;
|
||||
int ret;
|
||||
|
||||
/* This rule is a Wake-on-LAN filter and we must specifically
|
||||
@ -843,6 +871,34 @@ static int bcm_sf2_cfp_rule_insert(struct dsa_switch *ds, int port,
|
||||
dsa_is_cpu_port(ds, port_num)) ||
|
||||
port_num >= priv->hw_params.num_ports)
|
||||
return -EINVAL;
|
||||
|
||||
/* If the rule is matching a particular VLAN, make sure that we honor
|
||||
* the matching and have it tagged or untagged on the destination port,
|
||||
* we do this on egress with a VLAN entry. The egress tagging attribute
|
||||
* is expected to be provided in h_ext.data[1] bit 0. A 1 means untagged,
|
||||
* a 0 means tagged.
|
||||
*/
|
||||
if (fs->flow_type & FLOW_EXT) {
|
||||
/* We cannot support matching multiple VLAN IDs yet */
|
||||
if ((be16_to_cpu(fs->m_ext.vlan_tci) & VLAN_VID_MASK) !=
|
||||
VLAN_VID_MASK)
|
||||
return -EINVAL;
|
||||
|
||||
vid = be16_to_cpu(fs->h_ext.vlan_tci) & VLAN_VID_MASK;
|
||||
vlan.vid_begin = vid;
|
||||
vlan.vid_end = vid;
|
||||
if (cpu_to_be32(fs->h_ext.data[1]) & 1)
|
||||
vlan.flags = BRIDGE_VLAN_INFO_UNTAGGED;
|
||||
else
|
||||
vlan.flags = 0;
|
||||
|
||||
ret = ds->ops->port_vlan_prepare(ds, port_num, &vlan);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ds->ops->port_vlan_add(ds, port_num, &vlan);
|
||||
}
|
||||
|
||||
/*
|
||||
* We have a small oddity where Port 6 just does not have a
|
||||
* valid bit here (so we substract by one).
|
||||
@ -878,21 +934,22 @@ static int bcm_sf2_cfp_rule_set(struct dsa_switch *ds, int port,
|
||||
int ret = -EINVAL;
|
||||
|
||||
/* Check for unsupported extensions */
|
||||
if ((fs->flow_type & FLOW_EXT) && (fs->m_ext.vlan_etype ||
|
||||
fs->m_ext.data[1]))
|
||||
if (fs->flow_type & FLOW_MAC_EXT)
|
||||
return -EINVAL;
|
||||
|
||||
if (fs->location != RX_CLS_LOC_ANY && fs->location >= CFP_NUM_RULES)
|
||||
return -EINVAL;
|
||||
|
||||
if (fs->location != RX_CLS_LOC_ANY &&
|
||||
test_bit(fs->location, priv->cfp.used))
|
||||
return -EBUSY;
|
||||
|
||||
if (fs->location != RX_CLS_LOC_ANY &&
|
||||
fs->location > bcm_sf2_cfp_rule_size(priv))
|
||||
return -EINVAL;
|
||||
|
||||
if ((fs->flow_type & FLOW_EXT) &&
|
||||
!(ds->ops->port_vlan_prepare || ds->ops->port_vlan_add ||
|
||||
ds->ops->port_vlan_del))
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
if (fs->location != RX_CLS_LOC_ANY &&
|
||||
test_bit(fs->location, priv->cfp.used))
|
||||
return -EBUSY;
|
||||
|
||||
ret = bcm_sf2_cfp_rule_cmp(priv, port, fs);
|
||||
if (ret == 0)
|
||||
return -EEXIST;
|
||||
@ -973,7 +1030,7 @@ static int bcm_sf2_cfp_rule_del(struct bcm_sf2_priv *priv, int port, u32 loc)
|
||||
struct cfp_rule *rule;
|
||||
int ret;
|
||||
|
||||
if (loc >= CFP_NUM_RULES)
|
||||
if (loc > bcm_sf2_cfp_rule_size(priv))
|
||||
return -EINVAL;
|
||||
|
||||
/* Refuse deleting unused rules, and those that are not unique since
|
||||
|
@ -1517,7 +1517,9 @@ static void gswip_phylink_mac_link_down(struct dsa_switch *ds, int port,
|
||||
static void gswip_phylink_mac_link_up(struct dsa_switch *ds, int port,
|
||||
unsigned int mode,
|
||||
phy_interface_t interface,
|
||||
struct phy_device *phydev)
|
||||
struct phy_device *phydev,
|
||||
int speed, int duplex,
|
||||
bool tx_pause, bool rx_pause)
|
||||
{
|
||||
struct gswip_priv *priv = ds->priv;
|
||||
|
||||
|
@ -1,5 +1,6 @@
|
||||
# SPDX-License-Identifier: GPL-2.0-only
|
||||
config NET_DSA_MICROCHIP_KSZ_COMMON
|
||||
select NET_DSA_TAG_KSZ
|
||||
tristate
|
||||
|
||||
menuconfig NET_DSA_MICROCHIP_KSZ9477
|
||||
|
@ -67,7 +67,7 @@ static void port_r_cnt(struct ksz_device *dev, int port)
|
||||
static void ksz_mib_read_work(struct work_struct *work)
|
||||
{
|
||||
struct ksz_device *dev = container_of(work, struct ksz_device,
|
||||
mib_read);
|
||||
mib_read.work);
|
||||
struct ksz_port_mib *mib;
|
||||
struct ksz_port *p;
|
||||
int i;
|
||||
@ -93,32 +93,24 @@ static void ksz_mib_read_work(struct work_struct *work)
|
||||
p->read = false;
|
||||
mutex_unlock(&mib->cnt_mutex);
|
||||
}
|
||||
}
|
||||
|
||||
static void mib_monitor(struct timer_list *t)
|
||||
{
|
||||
struct ksz_device *dev = from_timer(dev, t, mib_read_timer);
|
||||
|
||||
mod_timer(&dev->mib_read_timer, jiffies + dev->mib_read_interval);
|
||||
schedule_work(&dev->mib_read);
|
||||
schedule_delayed_work(&dev->mib_read, dev->mib_read_interval);
|
||||
}
|
||||
|
||||
void ksz_init_mib_timer(struct ksz_device *dev)
|
||||
{
|
||||
int i;
|
||||
|
||||
INIT_DELAYED_WORK(&dev->mib_read, ksz_mib_read_work);
|
||||
|
||||
/* Read MIB counters every 30 seconds to avoid overflow. */
|
||||
dev->mib_read_interval = msecs_to_jiffies(30000);
|
||||
|
||||
INIT_WORK(&dev->mib_read, ksz_mib_read_work);
|
||||
timer_setup(&dev->mib_read_timer, mib_monitor, 0);
|
||||
|
||||
for (i = 0; i < dev->mib_port_cnt; i++)
|
||||
dev->dev_ops->port_init_cnt(dev, i);
|
||||
|
||||
/* Start the timer 2 seconds later. */
|
||||
dev->mib_read_timer.expires = jiffies + msecs_to_jiffies(2000);
|
||||
add_timer(&dev->mib_read_timer);
|
||||
schedule_delayed_work(&dev->mib_read, msecs_to_jiffies(2000));
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(ksz_init_mib_timer);
|
||||
|
||||
@ -152,7 +144,7 @@ void ksz_adjust_link(struct dsa_switch *ds, int port,
|
||||
/* Read all MIB counters when the link is going down. */
|
||||
if (!phydev->link) {
|
||||
p->read = true;
|
||||
schedule_work(&dev->mib_read);
|
||||
schedule_delayed_work(&dev->mib_read, 0);
|
||||
}
|
||||
mutex_lock(&dev->dev_mutex);
|
||||
if (!phydev->link)
|
||||
@ -477,10 +469,8 @@ EXPORT_SYMBOL(ksz_switch_register);
|
||||
void ksz_switch_remove(struct ksz_device *dev)
|
||||
{
|
||||
/* timer started */
|
||||
if (dev->mib_read_timer.expires) {
|
||||
del_timer_sync(&dev->mib_read_timer);
|
||||
flush_work(&dev->mib_read);
|
||||
}
|
||||
if (dev->mib_read_interval)
|
||||
cancel_delayed_work_sync(&dev->mib_read);
|
||||
|
||||
dev->dev_ops->exit(dev);
|
||||
dsa_unregister_switch(dev->ds);
|
||||
|
@ -80,8 +80,7 @@ struct ksz_device {
|
||||
struct vlan_table *vlan_cache;
|
||||
|
||||
struct ksz_port *ports;
|
||||
struct timer_list mib_read_timer;
|
||||
struct work_struct mib_read;
|
||||
struct delayed_work mib_read;
|
||||
unsigned long mib_read_interval;
|
||||
u16 br_member;
|
||||
u16 member;
|
||||
|
@ -563,17 +563,6 @@ mt7530_mib_reset(struct dsa_switch *ds)
|
||||
mt7530_write(priv, MT7530_MIB_CCR, CCR_MIB_ACTIVATE);
|
||||
}
|
||||
|
||||
static void
|
||||
mt7530_port_set_status(struct mt7530_priv *priv, int port, int enable)
|
||||
{
|
||||
u32 mask = PMCR_TX_EN | PMCR_RX_EN | PMCR_FORCE_LNK;
|
||||
|
||||
if (enable)
|
||||
mt7530_set(priv, MT7530_PMCR_P(port), mask);
|
||||
else
|
||||
mt7530_clear(priv, MT7530_PMCR_P(port), mask);
|
||||
}
|
||||
|
||||
static int mt7530_phy_read(struct dsa_switch *ds, int port, int regnum)
|
||||
{
|
||||
struct mt7530_priv *priv = ds->priv;
|
||||
@ -750,7 +739,7 @@ mt7530_port_enable(struct dsa_switch *ds, int port,
|
||||
priv->ports[port].enable = true;
|
||||
mt7530_rmw(priv, MT7530_PCR_P(port), PCR_MATRIX_MASK,
|
||||
priv->ports[port].pm);
|
||||
mt7530_port_set_status(priv, port, 0);
|
||||
mt7530_clear(priv, MT7530_PMCR_P(port), PMCR_LINK_SETTINGS_MASK);
|
||||
|
||||
mutex_unlock(&priv->reg_mutex);
|
||||
|
||||
@ -773,7 +762,7 @@ mt7530_port_disable(struct dsa_switch *ds, int port)
|
||||
priv->ports[port].enable = false;
|
||||
mt7530_rmw(priv, MT7530_PCR_P(port), PCR_MATRIX_MASK,
|
||||
PCR_MATRIX_CLR);
|
||||
mt7530_port_set_status(priv, port, 0);
|
||||
mt7530_clear(priv, MT7530_PMCR_P(port), PMCR_LINK_SETTINGS_MASK);
|
||||
|
||||
mutex_unlock(&priv->reg_mutex);
|
||||
}
|
||||
@ -1222,6 +1211,64 @@ mt7530_port_vlan_del(struct dsa_switch *ds, int port,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int mt7530_port_mirror_add(struct dsa_switch *ds, int port,
|
||||
struct dsa_mall_mirror_tc_entry *mirror,
|
||||
bool ingress)
|
||||
{
|
||||
struct mt7530_priv *priv = ds->priv;
|
||||
u32 val;
|
||||
|
||||
/* Check for existent entry */
|
||||
if ((ingress ? priv->mirror_rx : priv->mirror_tx) & BIT(port))
|
||||
return -EEXIST;
|
||||
|
||||
val = mt7530_read(priv, MT7530_MFC);
|
||||
|
||||
/* MT7530 only supports one monitor port */
|
||||
if (val & MIRROR_EN && MIRROR_PORT(val) != mirror->to_local_port)
|
||||
return -EEXIST;
|
||||
|
||||
val |= MIRROR_EN;
|
||||
val &= ~MIRROR_MASK;
|
||||
val |= mirror->to_local_port;
|
||||
mt7530_write(priv, MT7530_MFC, val);
|
||||
|
||||
val = mt7530_read(priv, MT7530_PCR_P(port));
|
||||
if (ingress) {
|
||||
val |= PORT_RX_MIR;
|
||||
priv->mirror_rx |= BIT(port);
|
||||
} else {
|
||||
val |= PORT_TX_MIR;
|
||||
priv->mirror_tx |= BIT(port);
|
||||
}
|
||||
mt7530_write(priv, MT7530_PCR_P(port), val);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void mt7530_port_mirror_del(struct dsa_switch *ds, int port,
|
||||
struct dsa_mall_mirror_tc_entry *mirror)
|
||||
{
|
||||
struct mt7530_priv *priv = ds->priv;
|
||||
u32 val;
|
||||
|
||||
val = mt7530_read(priv, MT7530_PCR_P(port));
|
||||
if (mirror->ingress) {
|
||||
val &= ~PORT_RX_MIR;
|
||||
priv->mirror_rx &= ~BIT(port);
|
||||
} else {
|
||||
val &= ~PORT_TX_MIR;
|
||||
priv->mirror_tx &= ~BIT(port);
|
||||
}
|
||||
mt7530_write(priv, MT7530_PCR_P(port), val);
|
||||
|
||||
if (!priv->mirror_rx && !priv->mirror_tx) {
|
||||
val = mt7530_read(priv, MT7530_MFC);
|
||||
val &= ~MIRROR_EN;
|
||||
mt7530_write(priv, MT7530_MFC, val);
|
||||
}
|
||||
}
|
||||
|
||||
static enum dsa_tag_protocol
|
||||
mtk_get_tag_protocol(struct dsa_switch *ds, int port,
|
||||
enum dsa_tag_protocol mp)
|
||||
@ -1441,8 +1488,7 @@ static void mt7530_phylink_mac_config(struct dsa_switch *ds, int port,
|
||||
|
||||
mcr_cur = mt7530_read(priv, MT7530_PMCR_P(port));
|
||||
mcr_new = mcr_cur;
|
||||
mcr_new &= ~(PMCR_FORCE_SPEED_1000 | PMCR_FORCE_SPEED_100 |
|
||||
PMCR_FORCE_FDX | PMCR_TX_FC_EN | PMCR_RX_FC_EN);
|
||||
mcr_new &= ~PMCR_LINK_SETTINGS_MASK;
|
||||
mcr_new |= PMCR_IFG_XMIT(1) | PMCR_MAC_MODE | PMCR_BACKOFF_EN |
|
||||
PMCR_BACKPR_EN | PMCR_FORCE_MODE;
|
||||
|
||||
@ -1450,22 +1496,6 @@ static void mt7530_phylink_mac_config(struct dsa_switch *ds, int port,
|
||||
if (port == 5 && dsa_is_user_port(ds, 5))
|
||||
mcr_new |= PMCR_EXT_PHY;
|
||||
|
||||
switch (state->speed) {
|
||||
case SPEED_1000:
|
||||
mcr_new |= PMCR_FORCE_SPEED_1000;
|
||||
break;
|
||||
case SPEED_100:
|
||||
mcr_new |= PMCR_FORCE_SPEED_100;
|
||||
break;
|
||||
}
|
||||
if (state->duplex == DUPLEX_FULL) {
|
||||
mcr_new |= PMCR_FORCE_FDX;
|
||||
if (state->pause & MLO_PAUSE_TX)
|
||||
mcr_new |= PMCR_TX_FC_EN;
|
||||
if (state->pause & MLO_PAUSE_RX)
|
||||
mcr_new |= PMCR_RX_FC_EN;
|
||||
}
|
||||
|
||||
if (mcr_new != mcr_cur)
|
||||
mt7530_write(priv, MT7530_PMCR_P(port), mcr_new);
|
||||
}
|
||||
@ -1476,17 +1506,38 @@ static void mt7530_phylink_mac_link_down(struct dsa_switch *ds, int port,
|
||||
{
|
||||
struct mt7530_priv *priv = ds->priv;
|
||||
|
||||
mt7530_port_set_status(priv, port, 0);
|
||||
mt7530_clear(priv, MT7530_PMCR_P(port), PMCR_LINK_SETTINGS_MASK);
|
||||
}
|
||||
|
||||
static void mt7530_phylink_mac_link_up(struct dsa_switch *ds, int port,
|
||||
unsigned int mode,
|
||||
phy_interface_t interface,
|
||||
struct phy_device *phydev)
|
||||
struct phy_device *phydev,
|
||||
int speed, int duplex,
|
||||
bool tx_pause, bool rx_pause)
|
||||
{
|
||||
struct mt7530_priv *priv = ds->priv;
|
||||
u32 mcr;
|
||||
|
||||
mt7530_port_set_status(priv, port, 1);
|
||||
mcr = PMCR_RX_EN | PMCR_TX_EN | PMCR_FORCE_LNK;
|
||||
|
||||
switch (speed) {
|
||||
case SPEED_1000:
|
||||
mcr |= PMCR_FORCE_SPEED_1000;
|
||||
break;
|
||||
case SPEED_100:
|
||||
mcr |= PMCR_FORCE_SPEED_100;
|
||||
break;
|
||||
}
|
||||
if (duplex == DUPLEX_FULL) {
|
||||
mcr |= PMCR_FORCE_FDX;
|
||||
if (tx_pause)
|
||||
mcr |= PMCR_TX_FC_EN;
|
||||
if (rx_pause)
|
||||
mcr |= PMCR_RX_FC_EN;
|
||||
}
|
||||
|
||||
mt7530_set(priv, MT7530_PMCR_P(port), mcr);
|
||||
}
|
||||
|
||||
static void mt7530_phylink_validate(struct dsa_switch *ds, int port,
|
||||
@ -1611,6 +1662,8 @@ static const struct dsa_switch_ops mt7530_switch_ops = {
|
||||
.port_vlan_prepare = mt7530_port_vlan_prepare,
|
||||
.port_vlan_add = mt7530_port_vlan_add,
|
||||
.port_vlan_del = mt7530_port_vlan_del,
|
||||
.port_mirror_add = mt7530_port_mirror_add,
|
||||
.port_mirror_del = mt7530_port_mirror_del,
|
||||
.phylink_validate = mt7530_phylink_validate,
|
||||
.phylink_mac_link_state = mt7530_phylink_mac_link_state,
|
||||
.phylink_mac_config = mt7530_phylink_mac_config,
|
||||
|
@ -36,6 +36,9 @@ enum {
|
||||
#define CPU_EN BIT(7)
|
||||
#define CPU_PORT(x) ((x) << 4)
|
||||
#define CPU_MASK (0xf << 4)
|
||||
#define MIRROR_EN BIT(3)
|
||||
#define MIRROR_PORT(x) ((x) & 0x7)
|
||||
#define MIRROR_MASK 0x7
|
||||
|
||||
/* Registers for address table access */
|
||||
#define MT7530_ATA1 0x74
|
||||
@ -141,6 +144,8 @@ enum mt7530_stp_state {
|
||||
|
||||
/* Register for port control */
|
||||
#define MT7530_PCR_P(x) (0x2004 + ((x) * 0x100))
|
||||
#define PORT_TX_MIR BIT(9)
|
||||
#define PORT_RX_MIR BIT(8)
|
||||
#define PORT_VLAN(x) ((x) & 0x3)
|
||||
|
||||
enum mt7530_port_mode {
|
||||
@ -201,6 +206,10 @@ enum mt7530_vlan_port_attr {
|
||||
#define PMCR_FORCE_LNK BIT(0)
|
||||
#define PMCR_SPEED_MASK (PMCR_FORCE_SPEED_100 | \
|
||||
PMCR_FORCE_SPEED_1000)
|
||||
#define PMCR_LINK_SETTINGS_MASK (PMCR_TX_EN | PMCR_FORCE_SPEED_1000 | \
|
||||
PMCR_RX_EN | PMCR_FORCE_SPEED_100 | \
|
||||
PMCR_TX_FC_EN | PMCR_RX_FC_EN | \
|
||||
PMCR_FORCE_FDX | PMCR_FORCE_LNK)
|
||||
|
||||
#define MT7530_PMSR_P(x) (0x3008 + (x) * 0x100)
|
||||
#define PMSR_EEE1G BIT(7)
|
||||
@ -460,6 +469,8 @@ struct mt7530_priv {
|
||||
phy_interface_t p6_interface;
|
||||
phy_interface_t p5_interface;
|
||||
unsigned int p5_intf_sel;
|
||||
u8 mirror_rx;
|
||||
u8 mirror_tx;
|
||||
|
||||
struct mt7530_port ports[MT7530_NUM_PORTS];
|
||||
/* protect among processes for registers access*/
|
||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user