Merge branch 'netlink-protocol-specs'
Jakub Kicinski says: ==================== Netlink protocol specs I think the Netlink proto specs are far along enough to merge. Filling in all attribute types and quirks will be an ongoing effort but we have enough to cover FOU so it's somewhat complete. I fully intend to continue polishing the code but at the same time I'd like to start helping others base their work on the specs (e.g. DPLL) and need to start working on some new families myself. That's the progress / motivation for merging. The RFC [1] has more of a high level blurb, plus I created a lot of documentation, I'm not going to repeat it here. There was also the talk at LPC [2]. [1] https://lore.kernel.org/all/20220811022304.583300-1-kuba@kernel.org/ [2] https://youtu.be/9QkXIQXkaQk?t=2562 v2: https://lore.kernel.org/all/20220930023418.1346263-1-kuba@kernel.org/ v3: https://lore.kernel.org/all/20230119003613.111778-1-kuba@kernel.org/1 v4: - spec improvements (patch 2) - Python cleanup (patch 3) - rename auto-gen files and use the right comment style ==================== Link: https://lore.kernel.org/r/20230120175041.342573-1-kuba@kernel.org Signed-off-by: Paolo Abeni <pabeni@redhat.com>
This commit is contained in:
commit
c554520f2c
|
@ -127,6 +127,7 @@ Documents that don't fit elsewhere or which have yet to be categorized.
|
|||
:maxdepth: 1
|
||||
|
||||
librs
|
||||
netlink
|
||||
|
||||
.. only:: subproject and html
|
||||
|
||||
|
|
|
@ -0,0 +1,101 @@
|
|||
.. SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
.. _kernel_netlink:
|
||||
|
||||
===================================
|
||||
Netlink notes for kernel developers
|
||||
===================================
|
||||
|
||||
General guidance
|
||||
================
|
||||
|
||||
Attribute enums
|
||||
---------------
|
||||
|
||||
Older families often define "null" attributes and commands with value
|
||||
of ``0`` and named ``unspec``. This is supported (``type: unused``)
|
||||
but should be avoided in new families. The ``unspec`` enum values are
|
||||
not used in practice, so just set the value of the first attribute to ``1``.
|
||||
|
||||
Message enums
|
||||
-------------
|
||||
|
||||
Use the same command IDs for requests and replies. This makes it easier
|
||||
to match them up, and we have plenty of ID space.
|
||||
|
||||
Use separate command IDs for notifications. This makes it easier to
|
||||
sort the notifications from replies (and present them to the user
|
||||
application via a different API than replies).
|
||||
|
||||
Answer requests
|
||||
---------------
|
||||
|
||||
Older families do not reply to all of the commands, especially NEW / ADD
|
||||
commands. User only gets information whether the operation succeeded or
|
||||
not via the ACK. Try to find useful data to return. Once the command is
|
||||
added whether it replies with a full message or only an ACK is uAPI and
|
||||
cannot be changed. It's better to err on the side of replying.
|
||||
|
||||
Specifically NEW and ADD commands should reply with information identifying
|
||||
the created object such as the allocated object's ID (without having to
|
||||
resort to using ``NLM_F_ECHO``).
|
||||
|
||||
NLM_F_ECHO
|
||||
----------
|
||||
|
||||
Make sure to pass the request info to genl_notify() to allow ``NLM_F_ECHO``
|
||||
to take effect. This is useful for programs that need precise feedback
|
||||
from the kernel (for example for logging purposes).
|
||||
|
||||
Support dump consistency
|
||||
------------------------
|
||||
|
||||
If iterating over objects during dump may skip over objects or repeat
|
||||
them - make sure to report dump inconsistency with ``NLM_F_DUMP_INTR``.
|
||||
This is usually implemented by maintaining a generation id for the
|
||||
structure and recording it in the ``seq`` member of struct netlink_callback.
|
||||
|
||||
Netlink specification
|
||||
=====================
|
||||
|
||||
Documentation of the Netlink specification parts which are only relevant
|
||||
to the kernel space.
|
||||
|
||||
Globals
|
||||
-------
|
||||
|
||||
kernel-policy
|
||||
~~~~~~~~~~~~~
|
||||
|
||||
Defines if the kernel validation policy is per operation (``per-op``)
|
||||
or for the entire family (``global``). New families should use ``per-op``
|
||||
(default) to be able to narrow down the attributes accepted by a specific
|
||||
command.
|
||||
|
||||
checks
|
||||
------
|
||||
|
||||
Documentation for the ``checks`` sub-sections of attribute specs.
|
||||
|
||||
unterminated-ok
|
||||
~~~~~~~~~~~~~~~
|
||||
|
||||
Accept strings without the null-termination (for legacy families only).
|
||||
Switches from the ``NLA_NUL_STRING`` to ``NLA_STRING`` policy type.
|
||||
|
||||
max-len
|
||||
~~~~~~~
|
||||
|
||||
Defines max length for a binary or string attribute (corresponding
|
||||
to the ``len`` member of struct nla_policy). For string attributes terminating
|
||||
null character is not counted towards ``max-len``.
|
||||
|
||||
The field may either be a literal integer value or a name of a defined
|
||||
constant. String types may reduce the constant by one
|
||||
(i.e. specify ``max-len: CONST - 1``) to reserve space for the terminating
|
||||
character so implementations should recognize such pattern.
|
||||
|
||||
min-len
|
||||
~~~~~~~
|
||||
|
||||
Similar to ``max-len`` but defines minimum length.
|
|
@ -0,0 +1,333 @@
|
|||
# SPDX-License-Identifier: GPL-2.0
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://kernel.org/schemas/netlink/genetlink-c.yaml#
|
||||
$schema: https://json-schema.org/draft-07/schema
|
||||
|
||||
# Common defines
|
||||
$defs:
|
||||
uint:
|
||||
type: integer
|
||||
minimum: 0
|
||||
len-or-define:
|
||||
type: [ string, integer ]
|
||||
pattern: ^[0-9A-Za-z_]+( - 1)?$
|
||||
minimum: 0
|
||||
|
||||
# Schema for specs
|
||||
title: Protocol
|
||||
description: Specification of a genetlink protocol
|
||||
type: object
|
||||
required: [ name, doc, attribute-sets, operations ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
description: Name of the genetlink family.
|
||||
type: string
|
||||
doc:
|
||||
type: string
|
||||
version:
|
||||
description: Generic Netlink family version. Default is 1.
|
||||
type: integer
|
||||
minimum: 1
|
||||
protocol:
|
||||
description: Schema compatibility level. Default is "genetlink".
|
||||
enum: [ genetlink, genetlink-c ]
|
||||
# Start genetlink-c
|
||||
uapi-header:
|
||||
description: Path to the uAPI header, default is linux/${family-name}.h
|
||||
type: string
|
||||
c-family-name:
|
||||
description: Name of the define for the family name.
|
||||
type: string
|
||||
c-version-name:
|
||||
description: Name of the define for the verion of the family.
|
||||
type: string
|
||||
max-by-define:
|
||||
description: Makes the number of attributes and commands be specified by a define, not an enum value.
|
||||
type: boolean
|
||||
# End genetlink-c
|
||||
|
||||
definitions:
|
||||
description: List of type and constant definitions (enums, flags, defines).
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
required: [ type, name ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
type: string
|
||||
header:
|
||||
description: For C-compatible languages, header which already defines this value.
|
||||
type: string
|
||||
type:
|
||||
enum: [ const, enum, flags ]
|
||||
doc:
|
||||
type: string
|
||||
# For const
|
||||
value:
|
||||
description: For const - the value.
|
||||
type: [ string, integer ]
|
||||
# For enum and flags
|
||||
value-start:
|
||||
description: For enum or flags the literal initializer for the first value.
|
||||
type: [ string, integer ]
|
||||
entries:
|
||||
description: For enum or flags array of values.
|
||||
type: array
|
||||
items:
|
||||
oneOf:
|
||||
- type: string
|
||||
- type: object
|
||||
required: [ name ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
type: string
|
||||
value:
|
||||
type: integer
|
||||
doc:
|
||||
type: string
|
||||
render-max:
|
||||
description: Render the max members for this enum.
|
||||
type: boolean
|
||||
# Start genetlink-c
|
||||
enum-name:
|
||||
description: Name for enum, if empty no name will be used.
|
||||
type: [ string, "null" ]
|
||||
name-prefix:
|
||||
description: For enum the prefix of the values, optional.
|
||||
type: string
|
||||
# End genetlink-c
|
||||
|
||||
attribute-sets:
|
||||
description: Definition of attribute spaces for this family.
|
||||
type: array
|
||||
items:
|
||||
description: Definition of a single attribute space.
|
||||
type: object
|
||||
required: [ name, attributes ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
description: |
|
||||
Name used when referring to this space in other definitions, not used outside of the spec.
|
||||
type: string
|
||||
name-prefix:
|
||||
description: |
|
||||
Prefix for the C enum name of the attributes. Default family[name]-set[name]-a-
|
||||
type: string
|
||||
enum-name:
|
||||
description: Name for the enum type of the attribute.
|
||||
type: string
|
||||
doc:
|
||||
description: Documentation of the space.
|
||||
type: string
|
||||
subset-of:
|
||||
description: |
|
||||
Name of another space which this is a logical part of. Sub-spaces can be used to define
|
||||
a limited group of attributes which are used in a nest.
|
||||
type: string
|
||||
# Start genetlink-c
|
||||
attr-cnt-name:
|
||||
description: The explicit name for constant holding the count of attributes (last attr + 1).
|
||||
type: string
|
||||
attr-max-name:
|
||||
description: The explicit name for last member of attribute enum.
|
||||
type: string
|
||||
# End genetlink-c
|
||||
attributes:
|
||||
description: List of attributes in the space.
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
required: [ name, type ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
type: string
|
||||
type: &attr-type
|
||||
enum: [ unused, pad, flag, binary, u8, u16, u32, u64, s32, s64,
|
||||
string, nest, array-nest, nest-type-value ]
|
||||
doc:
|
||||
description: Documentation of the attribute.
|
||||
type: string
|
||||
value:
|
||||
description: Value for the enum item representing this attribute in the uAPI.
|
||||
$ref: '#/$defs/uint'
|
||||
type-value:
|
||||
description: Name of the value extracted from the type of a nest-type-value attribute.
|
||||
type: array
|
||||
items:
|
||||
type: string
|
||||
byte-order:
|
||||
enum: [ little-endian, big-endian ]
|
||||
multi-attr:
|
||||
type: boolean
|
||||
nested-attributes:
|
||||
description: Name of the space (sub-space) used inside the attribute.
|
||||
type: string
|
||||
enum:
|
||||
description: Name of the enum type used for the attribute.
|
||||
type: string
|
||||
enum-as-flags:
|
||||
description: |
|
||||
Treat the enum as flags. In most cases enum is either used as flags or as values.
|
||||
Sometimes, however, both forms are necessary, in which case header contains the enum
|
||||
form while specific attributes may request to convert the values into a bitfield.
|
||||
type: boolean
|
||||
checks:
|
||||
description: Kernel input validation.
|
||||
type: object
|
||||
additionalProperties: False
|
||||
properties:
|
||||
flags-mask:
|
||||
description: Name of the flags constant on which to base mask (unsigned scalar types only).
|
||||
type: string
|
||||
min:
|
||||
description: Min value for an integer attribute.
|
||||
type: integer
|
||||
min-len:
|
||||
description: Min length for a binary attribute.
|
||||
$ref: '#/$defs/len-or-define'
|
||||
max-len:
|
||||
description: Max length for a string or a binary attribute.
|
||||
$ref: '#/$defs/len-or-define'
|
||||
sub-type: *attr-type
|
||||
|
||||
# Make sure name-prefix does not appear in subsets (subsets inherit naming)
|
||||
dependencies:
|
||||
name-prefix:
|
||||
not:
|
||||
required: [ subset-of ]
|
||||
subset-of:
|
||||
not:
|
||||
required: [ name-prefix ]
|
||||
|
||||
operations:
|
||||
description: Operations supported by the protocol.
|
||||
type: object
|
||||
required: [ list ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
enum-model:
|
||||
description: |
|
||||
The model of assigning values to the operations.
|
||||
"unified" is the recommended model where all message types belong
|
||||
to a single enum.
|
||||
"directional" has the messages sent to the kernel and from the kernel
|
||||
enumerated separately.
|
||||
"notify-split" has the notifications and request-response types in
|
||||
different enums.
|
||||
enum: [ unified, directional, notify-split ]
|
||||
name-prefix:
|
||||
description: |
|
||||
Prefix for the C enum name of the command. The name is formed by concatenating
|
||||
the prefix with the upper case name of the command, with dashes replaced by underscores.
|
||||
type: string
|
||||
enum-name:
|
||||
description: Name for the enum type with commands.
|
||||
type: string
|
||||
async-prefix:
|
||||
description: Same as name-prefix but used to render notifications and events to separate enum.
|
||||
type: string
|
||||
async-enum:
|
||||
description: Name for the enum type with notifications/events.
|
||||
type: string
|
||||
list:
|
||||
description: List of commands
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
additionalProperties: False
|
||||
required: [ name, doc ]
|
||||
properties:
|
||||
name:
|
||||
description: Name of the operation, also defining its C enum value in uAPI.
|
||||
type: string
|
||||
doc:
|
||||
description: Documentation for the command.
|
||||
type: string
|
||||
value:
|
||||
description: Value for the enum in the uAPI.
|
||||
$ref: '#/$defs/uint'
|
||||
attribute-set:
|
||||
description: |
|
||||
Attribute space from which attributes directly in the requests and replies
|
||||
to this command are defined.
|
||||
type: string
|
||||
flags: &cmd_flags
|
||||
description: Command flags.
|
||||
type: array
|
||||
items:
|
||||
enum: [ admin-perm ]
|
||||
dont-validate:
|
||||
description: Kernel attribute validation flags.
|
||||
type: array
|
||||
items:
|
||||
enum: [ strict, dump ]
|
||||
do: &subop-type
|
||||
description: Main command handler.
|
||||
type: object
|
||||
additionalProperties: False
|
||||
properties:
|
||||
request: &subop-attr-list
|
||||
description: Definition of the request message for a given command.
|
||||
type: object
|
||||
additionalProperties: False
|
||||
properties:
|
||||
attributes:
|
||||
description: |
|
||||
Names of attributes from the attribute-set (not full attribute
|
||||
definitions, just names).
|
||||
type: array
|
||||
items:
|
||||
type: string
|
||||
reply: *subop-attr-list
|
||||
pre:
|
||||
description: Hook for a function to run before the main callback (pre_doit or start).
|
||||
type: string
|
||||
post:
|
||||
description: Hook for a function to run after the main callback (post_doit or done).
|
||||
type: string
|
||||
dump: *subop-type
|
||||
notify:
|
||||
description: Name of the command sharing the reply type with this notification.
|
||||
type: string
|
||||
event:
|
||||
type: object
|
||||
additionalProperties: False
|
||||
properties:
|
||||
attributes:
|
||||
description: Explicit list of the attributes for the notification.
|
||||
type: array
|
||||
items:
|
||||
type: string
|
||||
mcgrp:
|
||||
description: Name of the multicast group generating given notification.
|
||||
type: string
|
||||
mcast-groups:
|
||||
description: List of multicast groups.
|
||||
type: object
|
||||
required: [ list ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
list:
|
||||
description: List of groups.
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
required: [ name ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
description: |
|
||||
The name for the group, used to form the define and the value of the define.
|
||||
type: string
|
||||
# Start genetlink-c
|
||||
c-define-name:
|
||||
description: Override for the name of the define in C uAPI.
|
||||
type: string
|
||||
# End genetlink-c
|
||||
flags: *cmd_flags
|
|
@ -0,0 +1,356 @@
|
|||
# SPDX-License-Identifier: GPL-2.0
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://kernel.org/schemas/netlink/genetlink-legacy.yaml#
|
||||
$schema: https://json-schema.org/draft-07/schema
|
||||
|
||||
# Common defines
|
||||
$defs:
|
||||
uint:
|
||||
type: integer
|
||||
minimum: 0
|
||||
len-or-define:
|
||||
type: [ string, integer ]
|
||||
pattern: ^[0-9A-Za-z_]+( - 1)?$
|
||||
minimum: 0
|
||||
|
||||
# Schema for specs
|
||||
title: Protocol
|
||||
description: Specification of a genetlink protocol
|
||||
type: object
|
||||
required: [ name, doc, attribute-sets, operations ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
description: Name of the genetlink family.
|
||||
type: string
|
||||
doc:
|
||||
type: string
|
||||
version:
|
||||
description: Generic Netlink family version. Default is 1.
|
||||
type: integer
|
||||
minimum: 1
|
||||
protocol:
|
||||
description: Schema compatibility level. Default is "genetlink".
|
||||
enum: [ genetlink, genetlink-c, genetlink-legacy ] # Trim
|
||||
# Start genetlink-c
|
||||
uapi-header:
|
||||
description: Path to the uAPI header, default is linux/${family-name}.h
|
||||
type: string
|
||||
c-family-name:
|
||||
description: Name of the define for the family name.
|
||||
type: string
|
||||
c-version-name:
|
||||
description: Name of the define for the verion of the family.
|
||||
type: string
|
||||
max-by-define:
|
||||
description: Makes the number of attributes and commands be specified by a define, not an enum value.
|
||||
type: boolean
|
||||
# End genetlink-c
|
||||
# Start genetlink-legacy
|
||||
kernel-policy:
|
||||
description: |
|
||||
Defines if the input policy in the kernel is global, per-operation, or split per operation type.
|
||||
Default is split.
|
||||
enum: [ split, per-op, global ]
|
||||
# End genetlink-legacy
|
||||
|
||||
definitions:
|
||||
description: List of type and constant definitions (enums, flags, defines).
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
required: [ type, name ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
type: string
|
||||
header:
|
||||
description: For C-compatible languages, header which already defines this value.
|
||||
type: string
|
||||
type:
|
||||
enum: [ const, enum, flags, struct ] # Trim
|
||||
doc:
|
||||
type: string
|
||||
# For const
|
||||
value:
|
||||
description: For const - the value.
|
||||
type: [ string, integer ]
|
||||
# For enum and flags
|
||||
value-start:
|
||||
description: For enum or flags the literal initializer for the first value.
|
||||
type: [ string, integer ]
|
||||
entries:
|
||||
description: For enum or flags array of values.
|
||||
type: array
|
||||
items:
|
||||
oneOf:
|
||||
- type: string
|
||||
- type: object
|
||||
required: [ name ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
type: string
|
||||
value:
|
||||
type: integer
|
||||
doc:
|
||||
type: string
|
||||
render-max:
|
||||
description: Render the max members for this enum.
|
||||
type: boolean
|
||||
# Start genetlink-c
|
||||
enum-name:
|
||||
description: Name for enum, if empty no name will be used.
|
||||
type: [ string, "null" ]
|
||||
name-prefix:
|
||||
description: For enum the prefix of the values, optional.
|
||||
type: string
|
||||
# End genetlink-c
|
||||
# Start genetlink-legacy
|
||||
members:
|
||||
description: List of struct members. Only scalars and strings members allowed.
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
required: [ name, type ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
type: string
|
||||
type:
|
||||
enum: [ u8, u16, u32, u64, s8, s16, s32, s64, string ]
|
||||
len:
|
||||
$ref: '#/$defs/len-or-define'
|
||||
# End genetlink-legacy
|
||||
|
||||
attribute-sets:
|
||||
description: Definition of attribute spaces for this family.
|
||||
type: array
|
||||
items:
|
||||
description: Definition of a single attribute space.
|
||||
type: object
|
||||
required: [ name, attributes ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
description: |
|
||||
Name used when referring to this space in other definitions, not used outside of the spec.
|
||||
type: string
|
||||
name-prefix:
|
||||
description: |
|
||||
Prefix for the C enum name of the attributes. Default family[name]-set[name]-a-
|
||||
type: string
|
||||
enum-name:
|
||||
description: Name for the enum type of the attribute.
|
||||
type: string
|
||||
doc:
|
||||
description: Documentation of the space.
|
||||
type: string
|
||||
subset-of:
|
||||
description: |
|
||||
Name of another space which this is a logical part of. Sub-spaces can be used to define
|
||||
a limited group of attributes which are used in a nest.
|
||||
type: string
|
||||
# Start genetlink-c
|
||||
attr-cnt-name:
|
||||
description: The explicit name for constant holding the count of attributes (last attr + 1).
|
||||
type: string
|
||||
attr-max-name:
|
||||
description: The explicit name for last member of attribute enum.
|
||||
type: string
|
||||
# End genetlink-c
|
||||
attributes:
|
||||
description: List of attributes in the space.
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
required: [ name, type ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
type: string
|
||||
type: &attr-type
|
||||
enum: [ unused, pad, flag, binary, u8, u16, u32, u64, s32, s64,
|
||||
string, nest, array-nest, nest-type-value ]
|
||||
doc:
|
||||
description: Documentation of the attribute.
|
||||
type: string
|
||||
value:
|
||||
description: Value for the enum item representing this attribute in the uAPI.
|
||||
$ref: '#/$defs/uint'
|
||||
type-value:
|
||||
description: Name of the value extracted from the type of a nest-type-value attribute.
|
||||
type: array
|
||||
items:
|
||||
type: string
|
||||
byte-order:
|
||||
enum: [ little-endian, big-endian ]
|
||||
multi-attr:
|
||||
type: boolean
|
||||
nested-attributes:
|
||||
description: Name of the space (sub-space) used inside the attribute.
|
||||
type: string
|
||||
enum:
|
||||
description: Name of the enum type used for the attribute.
|
||||
type: string
|
||||
enum-as-flags:
|
||||
description: |
|
||||
Treat the enum as flags. In most cases enum is either used as flags or as values.
|
||||
Sometimes, however, both forms are necessary, in which case header contains the enum
|
||||
form while specific attributes may request to convert the values into a bitfield.
|
||||
type: boolean
|
||||
checks:
|
||||
description: Kernel input validation.
|
||||
type: object
|
||||
additionalProperties: False
|
||||
properties:
|
||||
flags-mask:
|
||||
description: Name of the flags constant on which to base mask (unsigned scalar types only).
|
||||
type: string
|
||||
min:
|
||||
description: Min value for an integer attribute.
|
||||
type: integer
|
||||
min-len:
|
||||
description: Min length for a binary attribute.
|
||||
$ref: '#/$defs/len-or-define'
|
||||
max-len:
|
||||
description: Max length for a string or a binary attribute.
|
||||
$ref: '#/$defs/len-or-define'
|
||||
sub-type: *attr-type
|
||||
|
||||
# Make sure name-prefix does not appear in subsets (subsets inherit naming)
|
||||
dependencies:
|
||||
name-prefix:
|
||||
not:
|
||||
required: [ subset-of ]
|
||||
subset-of:
|
||||
not:
|
||||
required: [ name-prefix ]
|
||||
|
||||
operations:
|
||||
description: Operations supported by the protocol.
|
||||
type: object
|
||||
required: [ list ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
enum-model:
|
||||
description: |
|
||||
The model of assigning values to the operations.
|
||||
"unified" is the recommended model where all message types belong
|
||||
to a single enum.
|
||||
"directional" has the messages sent to the kernel and from the kernel
|
||||
enumerated separately.
|
||||
"notify-split" has the notifications and request-response types in
|
||||
different enums.
|
||||
enum: [ unified, directional, notify-split ]
|
||||
name-prefix:
|
||||
description: |
|
||||
Prefix for the C enum name of the command. The name is formed by concatenating
|
||||
the prefix with the upper case name of the command, with dashes replaced by underscores.
|
||||
type: string
|
||||
enum-name:
|
||||
description: Name for the enum type with commands.
|
||||
type: string
|
||||
async-prefix:
|
||||
description: Same as name-prefix but used to render notifications and events to separate enum.
|
||||
type: string
|
||||
async-enum:
|
||||
description: Name for the enum type with notifications/events.
|
||||
type: string
|
||||
list:
|
||||
description: List of commands
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
additionalProperties: False
|
||||
required: [ name, doc ]
|
||||
properties:
|
||||
name:
|
||||
description: Name of the operation, also defining its C enum value in uAPI.
|
||||
type: string
|
||||
doc:
|
||||
description: Documentation for the command.
|
||||
type: string
|
||||
value:
|
||||
description: Value for the enum in the uAPI.
|
||||
$ref: '#/$defs/uint'
|
||||
attribute-set:
|
||||
description: |
|
||||
Attribute space from which attributes directly in the requests and replies
|
||||
to this command are defined.
|
||||
type: string
|
||||
flags: &cmd_flags
|
||||
description: Command flags.
|
||||
type: array
|
||||
items:
|
||||
enum: [ admin-perm ]
|
||||
dont-validate:
|
||||
description: Kernel attribute validation flags.
|
||||
type: array
|
||||
items:
|
||||
enum: [ strict, dump ]
|
||||
do: &subop-type
|
||||
description: Main command handler.
|
||||
type: object
|
||||
additionalProperties: False
|
||||
properties:
|
||||
request: &subop-attr-list
|
||||
description: Definition of the request message for a given command.
|
||||
type: object
|
||||
additionalProperties: False
|
||||
properties:
|
||||
attributes:
|
||||
description: |
|
||||
Names of attributes from the attribute-set (not full attribute
|
||||
definitions, just names).
|
||||
type: array
|
||||
items:
|
||||
type: string
|
||||
reply: *subop-attr-list
|
||||
pre:
|
||||
description: Hook for a function to run before the main callback (pre_doit or start).
|
||||
type: string
|
||||
post:
|
||||
description: Hook for a function to run after the main callback (post_doit or done).
|
||||
type: string
|
||||
dump: *subop-type
|
||||
notify:
|
||||
description: Name of the command sharing the reply type with this notification.
|
||||
type: string
|
||||
event:
|
||||
type: object
|
||||
additionalProperties: False
|
||||
properties:
|
||||
attributes:
|
||||
description: Explicit list of the attributes for the notification.
|
||||
type: array
|
||||
items:
|
||||
type: string
|
||||
mcgrp:
|
||||
description: Name of the multicast group generating given notification.
|
||||
type: string
|
||||
mcast-groups:
|
||||
description: List of multicast groups.
|
||||
type: object
|
||||
required: [ list ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
list:
|
||||
description: List of groups.
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
required: [ name ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
description: |
|
||||
The name for the group, used to form the define and the value of the define.
|
||||
type: string
|
||||
# Start genetlink-c
|
||||
c-define-name:
|
||||
description: Override for the name of the define in C uAPI.
|
||||
type: string
|
||||
# End genetlink-c
|
||||
flags: *cmd_flags
|
|
@ -0,0 +1,298 @@
|
|||
# SPDX-License-Identifier: GPL-2.0
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://kernel.org/schemas/netlink/genetlink-legacy.yaml#
|
||||
$schema: https://json-schema.org/draft-07/schema
|
||||
|
||||
# Common defines
|
||||
$defs:
|
||||
uint:
|
||||
type: integer
|
||||
minimum: 0
|
||||
len-or-define:
|
||||
type: [ string, integer ]
|
||||
pattern: ^[0-9A-Za-z_]+( - 1)?$
|
||||
minimum: 0
|
||||
|
||||
# Schema for specs
|
||||
title: Protocol
|
||||
description: Specification of a genetlink protocol
|
||||
type: object
|
||||
required: [ name, doc, attribute-sets, operations ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
description: Name of the genetlink family.
|
||||
type: string
|
||||
doc:
|
||||
type: string
|
||||
version:
|
||||
description: Generic Netlink family version. Default is 1.
|
||||
type: integer
|
||||
minimum: 1
|
||||
protocol:
|
||||
description: Schema compatibility level. Default is "genetlink".
|
||||
enum: [ genetlink ]
|
||||
|
||||
definitions:
|
||||
description: List of type and constant definitions (enums, flags, defines).
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
required: [ type, name ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
type: string
|
||||
header:
|
||||
description: For C-compatible languages, header which already defines this value.
|
||||
type: string
|
||||
type:
|
||||
enum: [ const, enum, flags ]
|
||||
doc:
|
||||
type: string
|
||||
# For const
|
||||
value:
|
||||
description: For const - the value.
|
||||
type: [ string, integer ]
|
||||
# For enum and flags
|
||||
value-start:
|
||||
description: For enum or flags the literal initializer for the first value.
|
||||
type: [ string, integer ]
|
||||
entries:
|
||||
description: For enum or flags array of values.
|
||||
type: array
|
||||
items:
|
||||
oneOf:
|
||||
- type: string
|
||||
- type: object
|
||||
required: [ name ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
type: string
|
||||
value:
|
||||
type: integer
|
||||
doc:
|
||||
type: string
|
||||
render-max:
|
||||
description: Render the max members for this enum.
|
||||
type: boolean
|
||||
|
||||
attribute-sets:
|
||||
description: Definition of attribute spaces for this family.
|
||||
type: array
|
||||
items:
|
||||
description: Definition of a single attribute space.
|
||||
type: object
|
||||
required: [ name, attributes ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
description: |
|
||||
Name used when referring to this space in other definitions, not used outside of the spec.
|
||||
type: string
|
||||
name-prefix:
|
||||
description: |
|
||||
Prefix for the C enum name of the attributes. Default family[name]-set[name]-a-
|
||||
type: string
|
||||
enum-name:
|
||||
description: Name for the enum type of the attribute.
|
||||
type: string
|
||||
doc:
|
||||
description: Documentation of the space.
|
||||
type: string
|
||||
subset-of:
|
||||
description: |
|
||||
Name of another space which this is a logical part of. Sub-spaces can be used to define
|
||||
a limited group of attributes which are used in a nest.
|
||||
type: string
|
||||
attributes:
|
||||
description: List of attributes in the space.
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
required: [ name, type ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
type: string
|
||||
type: &attr-type
|
||||
enum: [ unused, pad, flag, binary, u8, u16, u32, u64, s32, s64,
|
||||
string, nest, array-nest, nest-type-value ]
|
||||
doc:
|
||||
description: Documentation of the attribute.
|
||||
type: string
|
||||
value:
|
||||
description: Value for the enum item representing this attribute in the uAPI.
|
||||
$ref: '#/$defs/uint'
|
||||
type-value:
|
||||
description: Name of the value extracted from the type of a nest-type-value attribute.
|
||||
type: array
|
||||
items:
|
||||
type: string
|
||||
byte-order:
|
||||
enum: [ little-endian, big-endian ]
|
||||
multi-attr:
|
||||
type: boolean
|
||||
nested-attributes:
|
||||
description: Name of the space (sub-space) used inside the attribute.
|
||||
type: string
|
||||
enum:
|
||||
description: Name of the enum type used for the attribute.
|
||||
type: string
|
||||
enum-as-flags:
|
||||
description: |
|
||||
Treat the enum as flags. In most cases enum is either used as flags or as values.
|
||||
Sometimes, however, both forms are necessary, in which case header contains the enum
|
||||
form while specific attributes may request to convert the values into a bitfield.
|
||||
type: boolean
|
||||
checks:
|
||||
description: Kernel input validation.
|
||||
type: object
|
||||
additionalProperties: False
|
||||
properties:
|
||||
flags-mask:
|
||||
description: Name of the flags constant on which to base mask (unsigned scalar types only).
|
||||
type: string
|
||||
min:
|
||||
description: Min value for an integer attribute.
|
||||
type: integer
|
||||
min-len:
|
||||
description: Min length for a binary attribute.
|
||||
$ref: '#/$defs/len-or-define'
|
||||
max-len:
|
||||
description: Max length for a string or a binary attribute.
|
||||
$ref: '#/$defs/len-or-define'
|
||||
sub-type: *attr-type
|
||||
|
||||
# Make sure name-prefix does not appear in subsets (subsets inherit naming)
|
||||
dependencies:
|
||||
name-prefix:
|
||||
not:
|
||||
required: [ subset-of ]
|
||||
subset-of:
|
||||
not:
|
||||
required: [ name-prefix ]
|
||||
|
||||
operations:
|
||||
description: Operations supported by the protocol.
|
||||
type: object
|
||||
required: [ list ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
enum-model:
|
||||
description: |
|
||||
The model of assigning values to the operations.
|
||||
"unified" is the recommended model where all message types belong
|
||||
to a single enum.
|
||||
"directional" has the messages sent to the kernel and from the kernel
|
||||
enumerated separately.
|
||||
"notify-split" has the notifications and request-response types in
|
||||
different enums.
|
||||
enum: [ unified, directional, notify-split ]
|
||||
name-prefix:
|
||||
description: |
|
||||
Prefix for the C enum name of the command. The name is formed by concatenating
|
||||
the prefix with the upper case name of the command, with dashes replaced by underscores.
|
||||
type: string
|
||||
enum-name:
|
||||
description: Name for the enum type with commands.
|
||||
type: string
|
||||
async-prefix:
|
||||
description: Same as name-prefix but used to render notifications and events to separate enum.
|
||||
type: string
|
||||
async-enum:
|
||||
description: Name for the enum type with notifications/events.
|
||||
type: string
|
||||
list:
|
||||
description: List of commands
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
additionalProperties: False
|
||||
required: [ name, doc ]
|
||||
properties:
|
||||
name:
|
||||
description: Name of the operation, also defining its C enum value in uAPI.
|
||||
type: string
|
||||
doc:
|
||||
description: Documentation for the command.
|
||||
type: string
|
||||
value:
|
||||
description: Value for the enum in the uAPI.
|
||||
$ref: '#/$defs/uint'
|
||||
attribute-set:
|
||||
description: |
|
||||
Attribute space from which attributes directly in the requests and replies
|
||||
to this command are defined.
|
||||
type: string
|
||||
flags: &cmd_flags
|
||||
description: Command flags.
|
||||
type: array
|
||||
items:
|
||||
enum: [ admin-perm ]
|
||||
dont-validate:
|
||||
description: Kernel attribute validation flags.
|
||||
type: array
|
||||
items:
|
||||
enum: [ strict, dump ]
|
||||
do: &subop-type
|
||||
description: Main command handler.
|
||||
type: object
|
||||
additionalProperties: False
|
||||
properties:
|
||||
request: &subop-attr-list
|
||||
description: Definition of the request message for a given command.
|
||||
type: object
|
||||
additionalProperties: False
|
||||
properties:
|
||||
attributes:
|
||||
description: |
|
||||
Names of attributes from the attribute-set (not full attribute
|
||||
definitions, just names).
|
||||
type: array
|
||||
items:
|
||||
type: string
|
||||
reply: *subop-attr-list
|
||||
pre:
|
||||
description: Hook for a function to run before the main callback (pre_doit or start).
|
||||
type: string
|
||||
post:
|
||||
description: Hook for a function to run after the main callback (post_doit or done).
|
||||
type: string
|
||||
dump: *subop-type
|
||||
notify:
|
||||
description: Name of the command sharing the reply type with this notification.
|
||||
type: string
|
||||
event:
|
||||
type: object
|
||||
additionalProperties: False
|
||||
properties:
|
||||
attributes:
|
||||
description: Explicit list of the attributes for the notification.
|
||||
type: array
|
||||
items:
|
||||
type: string
|
||||
mcgrp:
|
||||
description: Name of the multicast group generating given notification.
|
||||
type: string
|
||||
mcast-groups:
|
||||
description: List of multicast groups.
|
||||
type: object
|
||||
required: [ list ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
list:
|
||||
description: List of groups.
|
||||
type: array
|
||||
items:
|
||||
type: object
|
||||
required: [ name ]
|
||||
additionalProperties: False
|
||||
properties:
|
||||
name:
|
||||
description: |
|
||||
The name for the group, used to form the define and the value of the define.
|
||||
type: string
|
||||
flags: *cmd_flags
|
|
@ -0,0 +1,128 @@
|
|||
name: fou
|
||||
|
||||
protocol: genetlink-legacy
|
||||
|
||||
doc: |
|
||||
Foo-over-UDP.
|
||||
|
||||
c-family-name: fou-genl-name
|
||||
c-version-name: fou-genl-version
|
||||
max-by-define: true
|
||||
kernel-policy: global
|
||||
|
||||
definitions:
|
||||
-
|
||||
type: enum
|
||||
name: encap_type
|
||||
name-prefix: fou-encap-
|
||||
enum-name:
|
||||
entries: [ unspec, direct, gue ]
|
||||
|
||||
attribute-sets:
|
||||
-
|
||||
name: fou
|
||||
name-prefix: fou-attr-
|
||||
attributes:
|
||||
-
|
||||
name: unspec
|
||||
type: unused
|
||||
-
|
||||
name: port
|
||||
type: u16
|
||||
byte-order: big-endian
|
||||
-
|
||||
name: af
|
||||
type: u8
|
||||
-
|
||||
name: ipproto
|
||||
type: u8
|
||||
-
|
||||
name: type
|
||||
type: u8
|
||||
-
|
||||
name: remcsum_nopartial
|
||||
type: flag
|
||||
-
|
||||
name: local_v4
|
||||
type: u32
|
||||
-
|
||||
name: local_v6
|
||||
type: binary
|
||||
checks:
|
||||
min-len: 16
|
||||
-
|
||||
name: peer_v4
|
||||
type: u32
|
||||
-
|
||||
name: peer_v6
|
||||
type: binary
|
||||
checks:
|
||||
min-len: 16
|
||||
-
|
||||
name: peer_port
|
||||
type: u16
|
||||
byte-order: big-endian
|
||||
-
|
||||
name: ifindex
|
||||
type: s32
|
||||
|
||||
operations:
|
||||
list:
|
||||
-
|
||||
name: unspec
|
||||
doc: unused
|
||||
|
||||
-
|
||||
name: add
|
||||
doc: Add port.
|
||||
attribute-set: fou
|
||||
|
||||
dont-validate: [ strict, dump ]
|
||||
flags: [ admin-perm ]
|
||||
|
||||
do:
|
||||
request: &all_attrs
|
||||
attributes:
|
||||
- port
|
||||
- ipproto
|
||||
- type
|
||||
- remcsum_nopartial
|
||||
- local_v4
|
||||
- peer_v4
|
||||
- local_v6
|
||||
- peer_v6
|
||||
- peer_port
|
||||
- ifindex
|
||||
|
||||
-
|
||||
name: del
|
||||
doc: Delete port.
|
||||
attribute-set: fou
|
||||
|
||||
dont-validate: [ strict, dump ]
|
||||
flags: [ admin-perm ]
|
||||
|
||||
do:
|
||||
request: &select_attrs
|
||||
attributes:
|
||||
- af
|
||||
- ifindex
|
||||
- port
|
||||
- peer_port
|
||||
- local_v4
|
||||
- peer_v4
|
||||
- local_v6
|
||||
- peer_v6
|
||||
|
||||
-
|
||||
name: get
|
||||
doc: Get tunnel info.
|
||||
attribute-set: fou
|
||||
dont-validate: [ strict, dump ]
|
||||
|
||||
do:
|
||||
request: *select_attrs
|
||||
reply: *all_attrs
|
||||
|
||||
dump:
|
||||
reply: *all_attrs
|
|
@ -0,0 +1,107 @@
|
|||
.. SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
==============================
|
||||
Netlink spec C code generation
|
||||
==============================
|
||||
|
||||
This document describes how Netlink specifications are used to render
|
||||
C code (uAPI, policies etc.). It also defines the additional properties
|
||||
allowed in older families by the ``genetlink-c`` protocol level,
|
||||
to control the naming.
|
||||
|
||||
For brevity this document refers to ``name`` properties of various
|
||||
objects by the object type. For example ``$attr`` is the value
|
||||
of ``name`` in an attribute, and ``$family`` is the name of the
|
||||
family (the global ``name`` property).
|
||||
|
||||
The upper case is used to denote literal values, e.g. ``$family-CMD``
|
||||
means the concatenation of ``$family``, a dash character, and the literal
|
||||
``CMD``.
|
||||
|
||||
The names of ``#defines`` and enum values are always converted to upper case,
|
||||
and with dashes (``-``) replaced by underscores (``_``).
|
||||
|
||||
If the constructed name is a C keyword, an extra underscore is
|
||||
appended (``do`` -> ``do_``).
|
||||
|
||||
Globals
|
||||
=======
|
||||
|
||||
``c-family-name`` controls the name of the ``#define`` for the family
|
||||
name, default is ``$family-FAMILY-NAME``.
|
||||
|
||||
``c-version-name`` controls the name of the ``#define`` for the version
|
||||
of the family, default is ``$family-FAMILY-VERSION``.
|
||||
|
||||
``max-by-define`` selects if max values for enums are defined as a
|
||||
``#define`` rather than inside the enum.
|
||||
|
||||
Definitions
|
||||
===========
|
||||
|
||||
Constants
|
||||
---------
|
||||
|
||||
Every constant is rendered as a ``#define``.
|
||||
The name of the constant is ``$family-$constant`` and the value
|
||||
is rendered as a string or integer according to its type in the spec.
|
||||
|
||||
Enums and flags
|
||||
---------------
|
||||
|
||||
Enums are named ``$family-$enum``. The full name can be set directly
|
||||
or suppressed by specifying the ``enum-name`` property.
|
||||
Default entry name is ``$family-$enum-$entry``.
|
||||
If ``name-prefix`` is specified it replaces the ``$family-$enum``
|
||||
portion of the entry name.
|
||||
|
||||
Boolean ``render-max`` controls creation of the max values
|
||||
(which are enabled by default for attribute enums).
|
||||
|
||||
Attributes
|
||||
==========
|
||||
|
||||
Each attribute set (excluding fractional sets) is rendered as an enum.
|
||||
|
||||
Attribute enums are traditionally unnamed in netlink headers.
|
||||
If naming is desired ``enum-name`` can be used to specify the name.
|
||||
|
||||
The default attribute name prefix is ``$family-A`` if the name of the set
|
||||
is the same as the name of the family and ``$family-A-$set`` if the names
|
||||
differ. The prefix can be overridden by the ``name-prefix`` property of a set.
|
||||
The rest of the section will refer to the prefix as ``$pfx``.
|
||||
|
||||
Attributes are named ``$pfx-$attribute``.
|
||||
|
||||
Attribute enums end with two special values ``__$pfx-MAX`` and ``$pfx-MAX``
|
||||
which are used for sizing attribute tables.
|
||||
These two names can be specified directly with the ``attr-cnt-name``
|
||||
and ``attr-max-name`` properties respectively.
|
||||
|
||||
If ``max-by-define`` is set to ``true`` at the global level ``attr-max-name``
|
||||
will be specified as a ``#define`` rather than an enum value.
|
||||
|
||||
Operations
|
||||
==========
|
||||
|
||||
Operations are named ``$family-CMD-$operation``.
|
||||
If ``name-prefix`` is specified it replaces the ``$family-CMD``
|
||||
portion of the name.
|
||||
|
||||
Similarly to attribute enums operation enums end with special count and max
|
||||
attributes. For operations those attributes can be renamed with
|
||||
``cmd-cnt-name`` and ``cmd-max-name``. Max will be a define if ``max-by-define``
|
||||
is ``true``.
|
||||
|
||||
Multicast groups
|
||||
================
|
||||
|
||||
Each multicast group gets a define rendered into the kernel uAPI header.
|
||||
The name of the define is ``$family-MCGRP-$group``, and can be overwritten
|
||||
with the ``c-define-name`` property.
|
||||
|
||||
Code generation
|
||||
===============
|
||||
|
||||
uAPI header is assumed to come from ``<linux/$family.h>`` in the default header
|
||||
search path. It can be changed using the ``uapi-header`` global property.
|
|
@ -0,0 +1,96 @@
|
|||
.. SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
=================================================================
|
||||
Netlink specification support for legacy Generic Netlink families
|
||||
=================================================================
|
||||
|
||||
This document describes the many additional quirks and properties
|
||||
required to describe older Generic Netlink families which form
|
||||
the ``genetlink-legacy`` protocol level.
|
||||
|
||||
The spec is a work in progress, some of the quirks are just documented
|
||||
for future reference.
|
||||
|
||||
Specification (defined)
|
||||
=======================
|
||||
|
||||
Attribute type nests
|
||||
--------------------
|
||||
|
||||
New Netlink families should use ``multi-attr`` to define arrays.
|
||||
Older families (e.g. ``genetlink`` control family) attempted to
|
||||
define array types reusing attribute type to carry information.
|
||||
|
||||
For reference the ``multi-attr`` array may look like this::
|
||||
|
||||
[ARRAY-ATTR]
|
||||
[INDEX (optionally)]
|
||||
[MEMBER1]
|
||||
[MEMBER2]
|
||||
[SOME-OTHER-ATTR]
|
||||
[ARRAY-ATTR]
|
||||
[INDEX (optionally)]
|
||||
[MEMBER1]
|
||||
[MEMBER2]
|
||||
|
||||
where ``ARRAY-ATTR`` is the array entry type.
|
||||
|
||||
array-nest
|
||||
~~~~~~~~~~
|
||||
|
||||
``array-nest`` creates the following structure::
|
||||
|
||||
[SOME-OTHER-ATTR]
|
||||
[ARRAY-ATTR]
|
||||
[ENTRY]
|
||||
[MEMBER1]
|
||||
[MEMBER2]
|
||||
[ENTRY]
|
||||
[MEMBER1]
|
||||
[MEMBER2]
|
||||
|
||||
It wraps the entire array in an extra attribute (hence limiting its size
|
||||
to 64kB). The ``ENTRY`` nests are special and have the index of the entry
|
||||
as their type instead of normal attribute type.
|
||||
|
||||
type-value
|
||||
~~~~~~~~~~
|
||||
|
||||
``type-value`` is a construct which uses attribute types to carry
|
||||
information about a single object (often used when array is dumped
|
||||
entry-by-entry).
|
||||
|
||||
``type-value`` can have multiple levels of nesting, for example
|
||||
genetlink's policy dumps create the following structures::
|
||||
|
||||
[POLICY-IDX]
|
||||
[ATTR-IDX]
|
||||
[POLICY-INFO-ATTR1]
|
||||
[POLICY-INFO-ATTR2]
|
||||
|
||||
Where the first level of nest has the policy index as it's attribute
|
||||
type, it contains a single nest which has the attribute index as its
|
||||
type. Inside the attr-index nest are the policy attributes. Modern
|
||||
Netlink families should have instead defined this as a flat structure,
|
||||
the nesting serves no good purpose here.
|
||||
|
||||
Other quirks (todo)
|
||||
===================
|
||||
|
||||
Structures
|
||||
----------
|
||||
|
||||
Legacy families can define C structures both to be used as the contents
|
||||
of an attribute and as a fixed message header. The plan is to define
|
||||
the structs in ``definitions`` and link the appropriate attrs.
|
||||
|
||||
Multi-message DO
|
||||
----------------
|
||||
|
||||
New Netlink families should never respond to a DO operation with multiple
|
||||
replies, with ``NLM_F_MULTI`` set. Use a filtered dump instead.
|
||||
|
||||
At the spec level we can define a ``dumps`` property for the ``do``,
|
||||
perhaps with values of ``combine`` and ``multi-object`` depending
|
||||
on how the parsing should be implemented (parse into a single reply
|
||||
vs list of objects i.e. pretty much a dump).
|
|
@ -10,3 +10,8 @@ Netlink documentation for users.
|
|||
:maxdepth: 2
|
||||
|
||||
intro
|
||||
specs
|
||||
c-code-gen
|
||||
genetlink-legacy
|
||||
|
||||
See also :ref:`Documentation/core-api/netlink.rst <kernel_netlink>`.
|
||||
|
|
|
@ -0,0 +1,422 @@
|
|||
.. SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
=========================================
|
||||
Netlink protocol specifications (in YAML)
|
||||
=========================================
|
||||
|
||||
Netlink protocol specifications are complete, machine readable descriptions of
|
||||
Netlink protocols written in YAML. The goal of the specifications is to allow
|
||||
separating Netlink parsing from user space logic and minimize the amount of
|
||||
hand written Netlink code for each new family, command, attribute.
|
||||
Netlink specs should be complete and not depend on any other spec
|
||||
or C header file, making it easy to use in languages which can't include
|
||||
kernel headers directly.
|
||||
|
||||
Internally kernel uses the YAML specs to generate:
|
||||
|
||||
- the C uAPI header
|
||||
- documentation of the protocol as a ReST file
|
||||
- policy tables for input attribute validation
|
||||
- operation tables
|
||||
|
||||
YAML specifications can be found under ``Documentation/netlink/specs/``
|
||||
|
||||
Compatibility levels
|
||||
====================
|
||||
|
||||
There are four schema levels for Netlink specs, from the simplest used
|
||||
by new families to the most complex covering all the quirks of the old ones.
|
||||
Each next level inherits the attributes of the previous level, meaning that
|
||||
user capable of parsing more complex ``genetlink`` schemas is also compatible
|
||||
with simpler ones. The levels are:
|
||||
|
||||
- ``genetlink`` - most streamlined, should be used by all new families
|
||||
- ``genetlink-c`` - superset of ``genetlink`` with extra attributes allowing
|
||||
customization of define and enum type and value names; this schema should
|
||||
be equivalent to ``genetlink`` for all implementations which don't interact
|
||||
directly with C uAPI headers
|
||||
- ``genetlink-legacy`` - Generic Netlink catch all schema supporting quirks of
|
||||
all old genetlink families, strange attribute formats, binary structures etc.
|
||||
- ``netlink-raw`` - catch all schema supporting pre-Generic Netlink protocols
|
||||
such as ``NETLINK_ROUTE``
|
||||
|
||||
The definition of the schemas (in ``jsonschema``) can be found
|
||||
under ``Documentation/netlink/``.
|
||||
|
||||
Schema structure
|
||||
================
|
||||
|
||||
YAML schema has the following conceptual sections:
|
||||
|
||||
- globals
|
||||
- definitions
|
||||
- attributes
|
||||
- operations
|
||||
- multicast groups
|
||||
|
||||
Most properties in the schema accept (or in fact require) a ``doc``
|
||||
sub-property documenting the defined object.
|
||||
|
||||
The following sections describe the properties of the most modern ``genetlink``
|
||||
schema. See the documentation of :doc:`genetlink-c <c-code-gen>`
|
||||
for information on how C names are derived from name properties.
|
||||
|
||||
genetlink
|
||||
=========
|
||||
|
||||
Globals
|
||||
-------
|
||||
|
||||
Attributes listed directly at the root level of the spec file.
|
||||
|
||||
name
|
||||
~~~~
|
||||
|
||||
Name of the family. Name identifies the family in a unique way, since
|
||||
the Family IDs are allocated dynamically.
|
||||
|
||||
version
|
||||
~~~~~~~
|
||||
|
||||
Generic Netlink family version, default is 1.
|
||||
|
||||
protocol
|
||||
~~~~~~~~
|
||||
|
||||
The schema level, default is ``genetlink``, which is the only value
|
||||
allowed for new ``genetlink`` families.
|
||||
|
||||
definitions
|
||||
-----------
|
||||
|
||||
Array of type and constant definitions.
|
||||
|
||||
name
|
||||
~~~~
|
||||
|
||||
Name of the type / constant.
|
||||
|
||||
type
|
||||
~~~~
|
||||
|
||||
One of the following types:
|
||||
|
||||
- const - a single, standalone constant
|
||||
- enum - defines an integer enumeration, with values for each entry
|
||||
incrementing by 1, (e.g. 0, 1, 2, 3)
|
||||
- flags - defines an integer enumeration, with values for each entry
|
||||
occupying a bit, starting from bit 0, (e.g. 1, 2, 4, 8)
|
||||
|
||||
value
|
||||
~~~~~
|
||||
|
||||
The value for the ``const``.
|
||||
|
||||
value-start
|
||||
~~~~~~~~~~~
|
||||
|
||||
The first value for ``enum`` and ``flags``, allows overriding the default
|
||||
start value of ``0`` (for ``enum``) and starting bit (for ``flags``).
|
||||
For ``flags`` ``value-start`` selects the starting bit, not the shifted value.
|
||||
|
||||
Sparse enumerations are not supported.
|
||||
|
||||
entries
|
||||
~~~~~~~
|
||||
|
||||
Array of names of the entries for ``enum`` and ``flags``.
|
||||
|
||||
header
|
||||
~~~~~~
|
||||
|
||||
For C-compatible languages, header which already defines this value.
|
||||
In case the definition is shared by multiple families (e.g. ``IFNAMSIZ``)
|
||||
code generators for C-compatible languages may prefer to add an appropriate
|
||||
include instead of rendering a new definition.
|
||||
|
||||
attribute-sets
|
||||
--------------
|
||||
|
||||
This property contains information about netlink attributes of the family.
|
||||
All families have at least one attribute set, most have multiple.
|
||||
``attribute-sets`` is an array, with each entry describing a single set.
|
||||
|
||||
Note that the spec is "flattened" and is not meant to visually resemble
|
||||
the format of the netlink messages (unlike certain ad-hoc documentation
|
||||
formats seen in kernel comments). In the spec subordinate attribute sets
|
||||
are not defined inline as a nest, but defined in a separate attribute set
|
||||
referred to with a ``nested-attributes`` property of the container.
|
||||
|
||||
Spec may also contain fractional sets - sets which contain a ``subset-of``
|
||||
property. Such sets describe a section of a full set, allowing narrowing down
|
||||
which attributes are allowed in a nest or refining the validation criteria.
|
||||
Fractional sets can only be used in nests. They are not rendered to the uAPI
|
||||
in any fashion.
|
||||
|
||||
name
|
||||
~~~~
|
||||
|
||||
Uniquely identifies the attribute set, operations and nested attributes
|
||||
refer to the sets by the ``name``.
|
||||
|
||||
subset-of
|
||||
~~~~~~~~~
|
||||
|
||||
Re-defines a portion of another set (a fractional set).
|
||||
Allows narrowing down fields and changing validation criteria
|
||||
or even types of attributes depending on the nest in which they
|
||||
are contained. The ``value`` of each attribute in the fractional
|
||||
set is implicitly the same as in the main set.
|
||||
|
||||
attributes
|
||||
~~~~~~~~~~
|
||||
|
||||
List of attributes in the set.
|
||||
|
||||
Attribute properties
|
||||
--------------------
|
||||
|
||||
name
|
||||
~~~~
|
||||
|
||||
Identifies the attribute, unique within the set.
|
||||
|
||||
type
|
||||
~~~~
|
||||
|
||||
Netlink attribute type, see :ref:`attr_types`.
|
||||
|
||||
.. _assign_val:
|
||||
|
||||
value
|
||||
~~~~~
|
||||
|
||||
Numerical attribute ID, used in serialized Netlink messages.
|
||||
The ``value`` property can be skipped, in which case the attribute ID
|
||||
will be the value of the previous attribute plus one (recursively)
|
||||
and ``0`` for the first attribute in the attribute set.
|
||||
|
||||
Note that the ``value`` of an attribute is defined only in its main set.
|
||||
|
||||
enum
|
||||
~~~~
|
||||
|
||||
For integer types specifies that values in the attribute belong
|
||||
to an ``enum`` or ``flags`` from the ``definitions`` section.
|
||||
|
||||
enum-as-flags
|
||||
~~~~~~~~~~~~~
|
||||
|
||||
Treat ``enum`` as ``flags`` regardless of its type in ``definitions``.
|
||||
When both ``enum`` and ``flags`` forms are needed ``definitions`` should
|
||||
contain an ``enum`` and attributes which need the ``flags`` form should
|
||||
use this attribute.
|
||||
|
||||
nested-attributes
|
||||
~~~~~~~~~~~~~~~~~
|
||||
|
||||
Identifies the attribute space for attributes nested within given attribute.
|
||||
Only valid for complex attributes which may have sub-attributes.
|
||||
|
||||
multi-attr (arrays)
|
||||
~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
Boolean property signifying that the attribute may be present multiple times.
|
||||
Allowing an attribute to repeat is the recommended way of implementing arrays
|
||||
(no extra nesting).
|
||||
|
||||
byte-order
|
||||
~~~~~~~~~~
|
||||
|
||||
For integer types specifies attribute byte order - ``little-endian``
|
||||
or ``big-endian``.
|
||||
|
||||
checks
|
||||
~~~~~~
|
||||
|
||||
Input validation constraints used by the kernel. User space should query
|
||||
the policy of the running kernel using Generic Netlink introspection,
|
||||
rather than depend on what is specified in the spec file.
|
||||
|
||||
The validation policy in the kernel is formed by combining the type
|
||||
definition (``type`` and ``nested-attributes``) and the ``checks``.
|
||||
|
||||
operations
|
||||
----------
|
||||
|
||||
This section describes messages passed between the kernel and the user space.
|
||||
There are three types of entries in this section - operations, notifications
|
||||
and events.
|
||||
|
||||
Operations describe the most common request - response communication. User
|
||||
sends a request and kernel replies. Each operation may contain any combination
|
||||
of the two modes familiar to netlink users - ``do`` and ``dump``.
|
||||
``do`` and ``dump`` in turn contain a combination of ``request`` and
|
||||
``response`` properties. If no explicit message with attributes is passed
|
||||
in a given direction (e.g. a ``dump`` which does not accept filter, or a ``do``
|
||||
of a SET operation to which the kernel responds with just the netlink error
|
||||
code) ``request`` or ``response`` section can be skipped.
|
||||
``request`` and ``response`` sections list the attributes allowed in a message.
|
||||
The list contains only the names of attributes from a set referred
|
||||
to by the ``attribute-set`` property.
|
||||
|
||||
Notifications and events both refer to the asynchronous messages sent by
|
||||
the kernel to members of a multicast group. The difference between the
|
||||
two is that a notification shares its contents with a GET operation
|
||||
(the name of the GET operation is specified in the ``notify`` property).
|
||||
This arrangement is commonly used for notifications about
|
||||
objects where the notification carries the full object definition.
|
||||
|
||||
Events are more focused and carry only a subset of information rather than full
|
||||
object state (a made up example would be a link state change event with just
|
||||
the interface name and the new link state). Events contain the ``event``
|
||||
property. Events are considered less idiomatic for netlink and notifications
|
||||
should be preferred.
|
||||
|
||||
list
|
||||
~~~~
|
||||
|
||||
The only property of ``operations`` for ``genetlink``, holds the list of
|
||||
operations, notifications etc.
|
||||
|
||||
Operation properties
|
||||
--------------------
|
||||
|
||||
name
|
||||
~~~~
|
||||
|
||||
Identifies the operation.
|
||||
|
||||
value
|
||||
~~~~~
|
||||
|
||||
Numerical message ID, used in serialized Netlink messages.
|
||||
The same enumeration rules are applied as to
|
||||
:ref:`attribute values<assign_val>`.
|
||||
|
||||
attribute-set
|
||||
~~~~~~~~~~~~~
|
||||
|
||||
Specifies the attribute set contained within the message.
|
||||
|
||||
do
|
||||
~~~
|
||||
|
||||
Specification for the ``doit`` request. Should contain ``request``, ``reply``
|
||||
or both of these properties, each holding a :ref:`attr_list`.
|
||||
|
||||
dump
|
||||
~~~~
|
||||
|
||||
Specification for the ``dumpit`` request. Should contain ``request``, ``reply``
|
||||
or both of these properties, each holding a :ref:`attr_list`.
|
||||
|
||||
notify
|
||||
~~~~~~
|
||||
|
||||
Designates the message as a notification. Contains the name of the operation
|
||||
(possibly the same as the operation holding this property) which shares
|
||||
the contents with the notification (``do``).
|
||||
|
||||
event
|
||||
~~~~~
|
||||
|
||||
Specification of attributes in the event, holds a :ref:`attr_list`.
|
||||
``event`` property is mutually exclusive with ``notify``.
|
||||
|
||||
mcgrp
|
||||
~~~~~
|
||||
|
||||
Used with ``event`` and ``notify``, specifies which multicast group
|
||||
message belongs to.
|
||||
|
||||
.. _attr_list:
|
||||
|
||||
Message attribute list
|
||||
----------------------
|
||||
|
||||
``request``, ``reply`` and ``event`` properties have a single ``attributes``
|
||||
property which holds the list of attribute names.
|
||||
|
||||
Messages can also define ``pre`` and ``post`` properties which will be rendered
|
||||
as ``pre_doit`` and ``post_doit`` calls in the kernel (these properties should
|
||||
be ignored by user space).
|
||||
|
||||
mcast-groups
|
||||
------------
|
||||
|
||||
This section lists the multicast groups of the family.
|
||||
|
||||
list
|
||||
~~~~
|
||||
|
||||
The only property of ``mcast-groups`` for ``genetlink``, holds the list
|
||||
of groups.
|
||||
|
||||
Multicast group properties
|
||||
--------------------------
|
||||
|
||||
name
|
||||
~~~~
|
||||
|
||||
Uniquely identifies the multicast group in the family. Similarly to
|
||||
Family ID, Multicast Group ID needs to be resolved at runtime, based
|
||||
on the name.
|
||||
|
||||
.. _attr_types:
|
||||
|
||||
Attribute types
|
||||
===============
|
||||
|
||||
This section describes the attribute types supported by the ``genetlink``
|
||||
compatibility level. Refer to documentation of different levels for additional
|
||||
attribute types.
|
||||
|
||||
Scalar integer types
|
||||
--------------------
|
||||
|
||||
Fixed-width integer types:
|
||||
``u8``, ``u16``, ``u32``, ``u64``, ``s8``, ``s16``, ``s32``, ``s64``.
|
||||
|
||||
Note that types smaller than 32 bit should be avoided as using them
|
||||
does not save any memory in Netlink messages (due to alignment).
|
||||
See :ref:`pad_type` for padding of 64 bit attributes.
|
||||
|
||||
The payload of the attribute is the integer in host order unless ``byte-order``
|
||||
specifies otherwise.
|
||||
|
||||
.. _pad_type:
|
||||
|
||||
pad
|
||||
---
|
||||
|
||||
Special attribute type used for padding attributes which require alignment
|
||||
bigger than standard 4B alignment required by netlink (e.g. 64 bit integers).
|
||||
There can only be a single attribute of the ``pad`` type in any attribute set
|
||||
and it should be automatically used for padding when needed.
|
||||
|
||||
flag
|
||||
----
|
||||
|
||||
Attribute with no payload, its presence is the entire information.
|
||||
|
||||
binary
|
||||
------
|
||||
|
||||
Raw binary data attribute, the contents are opaque to generic code.
|
||||
|
||||
string
|
||||
------
|
||||
|
||||
Character string. Unless ``checks`` has ``unterminated-ok`` set to ``true``
|
||||
the string is required to be null terminated.
|
||||
``max-len`` in ``checks`` indicates the longest possible string,
|
||||
if not present the length of the string is unbounded.
|
||||
|
||||
Note that ``max-len`` does not count the terminating character.
|
||||
|
||||
nest
|
||||
----
|
||||
|
||||
Attribute containing other (nested) attributes.
|
||||
``nested-attributes`` specifies which attribute set is used inside.
|
|
@ -14562,8 +14562,10 @@ Q: https://patchwork.kernel.org/project/netdevbpf/list/
|
|||
B: mailto:netdev@vger.kernel.org
|
||||
T: git git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net.git
|
||||
T: git git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net-next.git
|
||||
F: Documentation/core-api/netlink.rst
|
||||
F: Documentation/networking/
|
||||
F: Documentation/process/maintainer-netdev.rst
|
||||
F: Documentation/userspace-api/netlink/
|
||||
F: include/linux/in.h
|
||||
F: include/linux/net.h
|
||||
F: include/linux/netdevice.h
|
||||
|
@ -14575,6 +14577,7 @@ F: include/uapi/linux/netdevice.h
|
|||
F: lib/net_utils.c
|
||||
F: lib/random32.c
|
||||
F: net/
|
||||
F: tools/net/
|
||||
F: tools/testing/selftests/net/
|
||||
|
||||
NETWORKING [IPSEC]
|
||||
|
|
|
@ -1,31 +1,36 @@
|
|||
/* SPDX-License-Identifier: GPL-2.0 WITH Linux-syscall-note */
|
||||
/* fou.h - FOU Interface */
|
||||
/* Do not edit directly, auto-generated from: */
|
||||
/* Documentation/netlink/specs/fou.yaml */
|
||||
/* YNL-GEN uapi header */
|
||||
|
||||
#ifndef _UAPI_LINUX_FOU_H
|
||||
#define _UAPI_LINUX_FOU_H
|
||||
|
||||
/* NETLINK_GENERIC related info
|
||||
*/
|
||||
#define FOU_GENL_NAME "fou"
|
||||
#define FOU_GENL_VERSION 0x1
|
||||
#define FOU_GENL_VERSION 1
|
||||
|
||||
enum {
|
||||
FOU_ENCAP_UNSPEC,
|
||||
FOU_ENCAP_DIRECT,
|
||||
FOU_ENCAP_GUE,
|
||||
};
|
||||
|
||||
enum {
|
||||
FOU_ATTR_UNSPEC,
|
||||
FOU_ATTR_PORT, /* u16 */
|
||||
FOU_ATTR_AF, /* u8 */
|
||||
FOU_ATTR_IPPROTO, /* u8 */
|
||||
FOU_ATTR_TYPE, /* u8 */
|
||||
FOU_ATTR_REMCSUM_NOPARTIAL, /* flag */
|
||||
FOU_ATTR_LOCAL_V4, /* u32 */
|
||||
FOU_ATTR_LOCAL_V6, /* in6_addr */
|
||||
FOU_ATTR_PEER_V4, /* u32 */
|
||||
FOU_ATTR_PEER_V6, /* in6_addr */
|
||||
FOU_ATTR_PEER_PORT, /* u16 */
|
||||
FOU_ATTR_IFINDEX, /* s32 */
|
||||
FOU_ATTR_PORT,
|
||||
FOU_ATTR_AF,
|
||||
FOU_ATTR_IPPROTO,
|
||||
FOU_ATTR_TYPE,
|
||||
FOU_ATTR_REMCSUM_NOPARTIAL,
|
||||
FOU_ATTR_LOCAL_V4,
|
||||
FOU_ATTR_LOCAL_V6,
|
||||
FOU_ATTR_PEER_V4,
|
||||
FOU_ATTR_PEER_V6,
|
||||
FOU_ATTR_PEER_PORT,
|
||||
FOU_ATTR_IFINDEX,
|
||||
|
||||
__FOU_ATTR_MAX,
|
||||
__FOU_ATTR_MAX
|
||||
};
|
||||
|
||||
#define FOU_ATTR_MAX (__FOU_ATTR_MAX - 1)
|
||||
|
||||
enum {
|
||||
|
@ -34,15 +39,8 @@ enum {
|
|||
FOU_CMD_DEL,
|
||||
FOU_CMD_GET,
|
||||
|
||||
__FOU_CMD_MAX,
|
||||
__FOU_CMD_MAX
|
||||
};
|
||||
|
||||
enum {
|
||||
FOU_ENCAP_UNSPEC,
|
||||
FOU_ENCAP_DIRECT,
|
||||
FOU_ENCAP_GUE,
|
||||
};
|
||||
|
||||
#define FOU_CMD_MAX (__FOU_CMD_MAX - 1)
|
||||
|
||||
#endif /* _UAPI_LINUX_FOU_H */
|
||||
|
|
|
@ -26,6 +26,7 @@ obj-$(CONFIG_IP_MROUTE) += ipmr.o
|
|||
obj-$(CONFIG_IP_MROUTE_COMMON) += ipmr_base.o
|
||||
obj-$(CONFIG_NET_IPIP) += ipip.o
|
||||
gre-y := gre_demux.o
|
||||
fou-y := fou_core.o fou_nl.o
|
||||
obj-$(CONFIG_NET_FOU) += fou.o
|
||||
obj-$(CONFIG_NET_IPGRE_DEMUX) += gre.o
|
||||
obj-$(CONFIG_NET_IPGRE) += ip_gre.o
|
||||
|
|
|
@ -19,6 +19,8 @@
|
|||
#include <uapi/linux/fou.h>
|
||||
#include <uapi/linux/genetlink.h>
|
||||
|
||||
#include "fou_nl.h"
|
||||
|
||||
struct fou {
|
||||
struct socket *sock;
|
||||
u8 protocol;
|
||||
|
@ -640,20 +642,6 @@ static int fou_destroy(struct net *net, struct fou_cfg *cfg)
|
|||
|
||||
static struct genl_family fou_nl_family;
|
||||
|
||||
static const struct nla_policy fou_nl_policy[FOU_ATTR_MAX + 1] = {
|
||||
[FOU_ATTR_PORT] = { .type = NLA_U16, },
|
||||
[FOU_ATTR_AF] = { .type = NLA_U8, },
|
||||
[FOU_ATTR_IPPROTO] = { .type = NLA_U8, },
|
||||
[FOU_ATTR_TYPE] = { .type = NLA_U8, },
|
||||
[FOU_ATTR_REMCSUM_NOPARTIAL] = { .type = NLA_FLAG, },
|
||||
[FOU_ATTR_LOCAL_V4] = { .type = NLA_U32, },
|
||||
[FOU_ATTR_PEER_V4] = { .type = NLA_U32, },
|
||||
[FOU_ATTR_LOCAL_V6] = { .len = sizeof(struct in6_addr), },
|
||||
[FOU_ATTR_PEER_V6] = { .len = sizeof(struct in6_addr), },
|
||||
[FOU_ATTR_PEER_PORT] = { .type = NLA_U16, },
|
||||
[FOU_ATTR_IFINDEX] = { .type = NLA_S32, },
|
||||
};
|
||||
|
||||
static int parse_nl_config(struct genl_info *info,
|
||||
struct fou_cfg *cfg)
|
||||
{
|
||||
|
@ -745,7 +733,7 @@ static int parse_nl_config(struct genl_info *info,
|
|||
return 0;
|
||||
}
|
||||
|
||||
static int fou_nl_cmd_add_port(struct sk_buff *skb, struct genl_info *info)
|
||||
int fou_nl_add_doit(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
struct net *net = genl_info_net(info);
|
||||
struct fou_cfg cfg;
|
||||
|
@ -758,7 +746,7 @@ static int fou_nl_cmd_add_port(struct sk_buff *skb, struct genl_info *info)
|
|||
return fou_create(net, &cfg, NULL);
|
||||
}
|
||||
|
||||
static int fou_nl_cmd_rm_port(struct sk_buff *skb, struct genl_info *info)
|
||||
int fou_nl_del_doit(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
struct net *net = genl_info_net(info);
|
||||
struct fou_cfg cfg;
|
||||
|
@ -827,7 +815,7 @@ nla_put_failure:
|
|||
return -EMSGSIZE;
|
||||
}
|
||||
|
||||
static int fou_nl_cmd_get_port(struct sk_buff *skb, struct genl_info *info)
|
||||
int fou_nl_get_doit(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
struct net *net = genl_info_net(info);
|
||||
struct fou_net *fn = net_generic(net, fou_net_id);
|
||||
|
@ -874,7 +862,7 @@ out_free:
|
|||
return ret;
|
||||
}
|
||||
|
||||
static int fou_nl_dump(struct sk_buff *skb, struct netlink_callback *cb)
|
||||
int fou_nl_get_dumpit(struct sk_buff *skb, struct netlink_callback *cb)
|
||||
{
|
||||
struct net *net = sock_net(skb->sk);
|
||||
struct fou_net *fn = net_generic(net, fou_net_id);
|
||||
|
@ -897,27 +885,6 @@ static int fou_nl_dump(struct sk_buff *skb, struct netlink_callback *cb)
|
|||
return skb->len;
|
||||
}
|
||||
|
||||
static const struct genl_small_ops fou_nl_ops[] = {
|
||||
{
|
||||
.cmd = FOU_CMD_ADD,
|
||||
.validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
|
||||
.doit = fou_nl_cmd_add_port,
|
||||
.flags = GENL_ADMIN_PERM,
|
||||
},
|
||||
{
|
||||
.cmd = FOU_CMD_DEL,
|
||||
.validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
|
||||
.doit = fou_nl_cmd_rm_port,
|
||||
.flags = GENL_ADMIN_PERM,
|
||||
},
|
||||
{
|
||||
.cmd = FOU_CMD_GET,
|
||||
.validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
|
||||
.doit = fou_nl_cmd_get_port,
|
||||
.dumpit = fou_nl_dump,
|
||||
},
|
||||
};
|
||||
|
||||
static struct genl_family fou_nl_family __ro_after_init = {
|
||||
.hdrsize = 0,
|
||||
.name = FOU_GENL_NAME,
|
|
@ -0,0 +1,48 @@
|
|||
// SPDX-License-Identifier: BSD-3-Clause
|
||||
/* Do not edit directly, auto-generated from: */
|
||||
/* Documentation/netlink/specs/fou.yaml */
|
||||
/* YNL-GEN kernel source */
|
||||
|
||||
#include <net/netlink.h>
|
||||
#include <net/genetlink.h>
|
||||
|
||||
#include "fou_nl.h"
|
||||
|
||||
#include <linux/fou.h>
|
||||
|
||||
/* Global operation policy for fou */
|
||||
const struct nla_policy fou_nl_policy[FOU_ATTR_IFINDEX + 1] = {
|
||||
[FOU_ATTR_PORT] = { .type = NLA_U16, },
|
||||
[FOU_ATTR_AF] = { .type = NLA_U8, },
|
||||
[FOU_ATTR_IPPROTO] = { .type = NLA_U8, },
|
||||
[FOU_ATTR_TYPE] = { .type = NLA_U8, },
|
||||
[FOU_ATTR_REMCSUM_NOPARTIAL] = { .type = NLA_FLAG, },
|
||||
[FOU_ATTR_LOCAL_V4] = { .type = NLA_U32, },
|
||||
[FOU_ATTR_LOCAL_V6] = { .len = 16, },
|
||||
[FOU_ATTR_PEER_V4] = { .type = NLA_U32, },
|
||||
[FOU_ATTR_PEER_V6] = { .len = 16, },
|
||||
[FOU_ATTR_PEER_PORT] = { .type = NLA_U16, },
|
||||
[FOU_ATTR_IFINDEX] = { .type = NLA_S32, },
|
||||
};
|
||||
|
||||
/* Ops table for fou */
|
||||
const struct genl_small_ops fou_nl_ops[3] = {
|
||||
{
|
||||
.cmd = FOU_CMD_ADD,
|
||||
.validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
|
||||
.doit = fou_nl_add_doit,
|
||||
.flags = GENL_ADMIN_PERM,
|
||||
},
|
||||
{
|
||||
.cmd = FOU_CMD_DEL,
|
||||
.validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
|
||||
.doit = fou_nl_del_doit,
|
||||
.flags = GENL_ADMIN_PERM,
|
||||
},
|
||||
{
|
||||
.cmd = FOU_CMD_GET,
|
||||
.validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
|
||||
.doit = fou_nl_get_doit,
|
||||
.dumpit = fou_nl_get_dumpit,
|
||||
},
|
||||
};
|
|
@ -0,0 +1,25 @@
|
|||
/* SPDX-License-Identifier: BSD-3-Clause */
|
||||
/* Do not edit directly, auto-generated from: */
|
||||
/* Documentation/netlink/specs/fou.yaml */
|
||||
/* YNL-GEN kernel header */
|
||||
|
||||
#ifndef _LINUX_FOU_GEN_H
|
||||
#define _LINUX_FOU_GEN_H
|
||||
|
||||
#include <net/netlink.h>
|
||||
#include <net/genetlink.h>
|
||||
|
||||
#include <linux/fou.h>
|
||||
|
||||
/* Global operation policy for fou */
|
||||
extern const struct nla_policy fou_nl_policy[FOU_ATTR_IFINDEX + 1];
|
||||
|
||||
/* Ops table for fou */
|
||||
extern const struct genl_small_ops fou_nl_ops[3];
|
||||
|
||||
int fou_nl_add_doit(struct sk_buff *skb, struct genl_info *info);
|
||||
int fou_nl_del_doit(struct sk_buff *skb, struct genl_info *info);
|
||||
int fou_nl_get_doit(struct sk_buff *skb, struct genl_info *info);
|
||||
int fou_nl_get_dumpit(struct sk_buff *skb, struct netlink_callback *cb);
|
||||
|
||||
#endif /* _LINUX_FOU_GEN_H */
|
|
@ -0,0 +1,47 @@
|
|||
#!/usr/bin/env python
|
||||
# SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
import argparse
|
||||
import json
|
||||
import pprint
|
||||
import time
|
||||
|
||||
from ynl import YnlFamily
|
||||
|
||||
|
||||
def main():
|
||||
parser = argparse.ArgumentParser(description='YNL CLI sample')
|
||||
parser.add_argument('--spec', dest='spec', type=str, required=True)
|
||||
parser.add_argument('--schema', dest='schema', type=str)
|
||||
parser.add_argument('--json', dest='json_text', type=str)
|
||||
parser.add_argument('--do', dest='do', type=str)
|
||||
parser.add_argument('--dump', dest='dump', type=str)
|
||||
parser.add_argument('--sleep', dest='sleep', type=int)
|
||||
parser.add_argument('--subscribe', dest='ntf', type=str)
|
||||
args = parser.parse_args()
|
||||
|
||||
attrs = {}
|
||||
if args.json_text:
|
||||
attrs = json.loads(args.json_text)
|
||||
|
||||
ynl = YnlFamily(args.spec, args.schema)
|
||||
|
||||
if args.ntf:
|
||||
ynl.ntf_subscribe(args.ntf)
|
||||
|
||||
if args.sleep:
|
||||
time.sleep(args.sleep)
|
||||
|
||||
if args.do or args.dump:
|
||||
method = getattr(ynl, args.do if args.do else args.dump)
|
||||
|
||||
reply = method(attrs, dump=bool(args.dump))
|
||||
pprint.PrettyPrinter().pprint(reply)
|
||||
|
||||
if args.ntf:
|
||||
ynl.check_ntf()
|
||||
pprint.PrettyPrinter().pprint(ynl.async_msg_queue)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
|
@ -0,0 +1,534 @@
|
|||
# SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
import functools
|
||||
import jsonschema
|
||||
import os
|
||||
import random
|
||||
import socket
|
||||
import struct
|
||||
import yaml
|
||||
|
||||
#
|
||||
# Generic Netlink code which should really be in some library, but I can't quickly find one.
|
||||
#
|
||||
|
||||
|
||||
class Netlink:
|
||||
# Netlink socket
|
||||
SOL_NETLINK = 270
|
||||
|
||||
NETLINK_ADD_MEMBERSHIP = 1
|
||||
NETLINK_CAP_ACK = 10
|
||||
NETLINK_EXT_ACK = 11
|
||||
|
||||
# Netlink message
|
||||
NLMSG_ERROR = 2
|
||||
NLMSG_DONE = 3
|
||||
|
||||
NLM_F_REQUEST = 1
|
||||
NLM_F_ACK = 4
|
||||
NLM_F_ROOT = 0x100
|
||||
NLM_F_MATCH = 0x200
|
||||
NLM_F_APPEND = 0x800
|
||||
|
||||
NLM_F_CAPPED = 0x100
|
||||
NLM_F_ACK_TLVS = 0x200
|
||||
|
||||
NLM_F_DUMP = NLM_F_ROOT | NLM_F_MATCH
|
||||
|
||||
NLA_F_NESTED = 0x8000
|
||||
NLA_F_NET_BYTEORDER = 0x4000
|
||||
|
||||
NLA_TYPE_MASK = NLA_F_NESTED | NLA_F_NET_BYTEORDER
|
||||
|
||||
# Genetlink defines
|
||||
NETLINK_GENERIC = 16
|
||||
|
||||
GENL_ID_CTRL = 0x10
|
||||
|
||||
# nlctrl
|
||||
CTRL_CMD_GETFAMILY = 3
|
||||
|
||||
CTRL_ATTR_FAMILY_ID = 1
|
||||
CTRL_ATTR_FAMILY_NAME = 2
|
||||
CTRL_ATTR_MAXATTR = 5
|
||||
CTRL_ATTR_MCAST_GROUPS = 7
|
||||
|
||||
CTRL_ATTR_MCAST_GRP_NAME = 1
|
||||
CTRL_ATTR_MCAST_GRP_ID = 2
|
||||
|
||||
# Extack types
|
||||
NLMSGERR_ATTR_MSG = 1
|
||||
NLMSGERR_ATTR_OFFS = 2
|
||||
NLMSGERR_ATTR_COOKIE = 3
|
||||
NLMSGERR_ATTR_POLICY = 4
|
||||
NLMSGERR_ATTR_MISS_TYPE = 5
|
||||
NLMSGERR_ATTR_MISS_NEST = 6
|
||||
|
||||
|
||||
class NlAttr:
|
||||
def __init__(self, raw, offset):
|
||||
self._len, self._type = struct.unpack("HH", raw[offset:offset + 4])
|
||||
self.type = self._type & ~Netlink.NLA_TYPE_MASK
|
||||
self.payload_len = self._len
|
||||
self.full_len = (self.payload_len + 3) & ~3
|
||||
self.raw = raw[offset + 4:offset + self.payload_len]
|
||||
|
||||
def as_u16(self):
|
||||
return struct.unpack("H", self.raw)[0]
|
||||
|
||||
def as_u32(self):
|
||||
return struct.unpack("I", self.raw)[0]
|
||||
|
||||
def as_u64(self):
|
||||
return struct.unpack("Q", self.raw)[0]
|
||||
|
||||
def as_strz(self):
|
||||
return self.raw.decode('ascii')[:-1]
|
||||
|
||||
def as_bin(self):
|
||||
return self.raw
|
||||
|
||||
def __repr__(self):
|
||||
return f"[type:{self.type} len:{self._len}] {self.raw}"
|
||||
|
||||
|
||||
class NlAttrs:
|
||||
def __init__(self, msg):
|
||||
self.attrs = []
|
||||
|
||||
offset = 0
|
||||
while offset < len(msg):
|
||||
attr = NlAttr(msg, offset)
|
||||
offset += attr.full_len
|
||||
self.attrs.append(attr)
|
||||
|
||||
def __iter__(self):
|
||||
yield from self.attrs
|
||||
|
||||
def __repr__(self):
|
||||
msg = ''
|
||||
for a in self.attrs:
|
||||
if msg:
|
||||
msg += '\n'
|
||||
msg += repr(a)
|
||||
return msg
|
||||
|
||||
|
||||
class NlMsg:
|
||||
def __init__(self, msg, offset, attr_space=None):
|
||||
self.hdr = msg[offset:offset + 16]
|
||||
|
||||
self.nl_len, self.nl_type, self.nl_flags, self.nl_seq, self.nl_portid = \
|
||||
struct.unpack("IHHII", self.hdr)
|
||||
|
||||
self.raw = msg[offset + 16:offset + self.nl_len]
|
||||
|
||||
self.error = 0
|
||||
self.done = 0
|
||||
|
||||
extack_off = None
|
||||
if self.nl_type == Netlink.NLMSG_ERROR:
|
||||
self.error = struct.unpack("i", self.raw[0:4])[0]
|
||||
self.done = 1
|
||||
extack_off = 20
|
||||
elif self.nl_type == Netlink.NLMSG_DONE:
|
||||
self.done = 1
|
||||
extack_off = 4
|
||||
|
||||
self.extack = None
|
||||
if self.nl_flags & Netlink.NLM_F_ACK_TLVS and extack_off:
|
||||
self.extack = dict()
|
||||
extack_attrs = NlAttrs(self.raw[extack_off:])
|
||||
for extack in extack_attrs:
|
||||
if extack.type == Netlink.NLMSGERR_ATTR_MSG:
|
||||
self.extack['msg'] = extack.as_strz()
|
||||
elif extack.type == Netlink.NLMSGERR_ATTR_MISS_TYPE:
|
||||
self.extack['miss-type'] = extack.as_u32()
|
||||
elif extack.type == Netlink.NLMSGERR_ATTR_MISS_NEST:
|
||||
self.extack['miss-nest'] = extack.as_u32()
|
||||
elif extack.type == Netlink.NLMSGERR_ATTR_OFFS:
|
||||
self.extack['bad-attr-offs'] = extack.as_u32()
|
||||
else:
|
||||
if 'unknown' not in self.extack:
|
||||
self.extack['unknown'] = []
|
||||
self.extack['unknown'].append(extack)
|
||||
|
||||
if attr_space:
|
||||
# We don't have the ability to parse nests yet, so only do global
|
||||
if 'miss-type' in self.extack and 'miss-nest' not in self.extack:
|
||||
miss_type = self.extack['miss-type']
|
||||
if len(attr_space.attr_list) > miss_type:
|
||||
spec = attr_space.attr_list[miss_type]
|
||||
desc = spec['name']
|
||||
if 'doc' in spec:
|
||||
desc += f" ({spec['doc']})"
|
||||
self.extack['miss-type'] = desc
|
||||
|
||||
def __repr__(self):
|
||||
msg = f"nl_len = {self.nl_len} ({len(self.raw)}) nl_flags = 0x{self.nl_flags:x} nl_type = {self.nl_type}\n"
|
||||
if self.error:
|
||||
msg += '\terror: ' + str(self.error)
|
||||
if self.extack:
|
||||
msg += '\textack: ' + repr(self.extack)
|
||||
return msg
|
||||
|
||||
|
||||
class NlMsgs:
|
||||
def __init__(self, data, attr_space=None):
|
||||
self.msgs = []
|
||||
|
||||
offset = 0
|
||||
while offset < len(data):
|
||||
msg = NlMsg(data, offset, attr_space=attr_space)
|
||||
offset += msg.nl_len
|
||||
self.msgs.append(msg)
|
||||
|
||||
def __iter__(self):
|
||||
yield from self.msgs
|
||||
|
||||
|
||||
genl_family_name_to_id = None
|
||||
|
||||
|
||||
def _genl_msg(nl_type, nl_flags, genl_cmd, genl_version, seq=None):
|
||||
# we prepend length in _genl_msg_finalize()
|
||||
if seq is None:
|
||||
seq = random.randint(1, 1024)
|
||||
nlmsg = struct.pack("HHII", nl_type, nl_flags, seq, 0)
|
||||
genlmsg = struct.pack("bbH", genl_cmd, genl_version, 0)
|
||||
return nlmsg + genlmsg
|
||||
|
||||
|
||||
def _genl_msg_finalize(msg):
|
||||
return struct.pack("I", len(msg) + 4) + msg
|
||||
|
||||
|
||||
def _genl_load_families():
|
||||
with socket.socket(socket.AF_NETLINK, socket.SOCK_RAW, Netlink.NETLINK_GENERIC) as sock:
|
||||
sock.setsockopt(Netlink.SOL_NETLINK, Netlink.NETLINK_CAP_ACK, 1)
|
||||
|
||||
msg = _genl_msg(Netlink.GENL_ID_CTRL,
|
||||
Netlink.NLM_F_REQUEST | Netlink.NLM_F_ACK | Netlink.NLM_F_DUMP,
|
||||
Netlink.CTRL_CMD_GETFAMILY, 1)
|
||||
msg = _genl_msg_finalize(msg)
|
||||
|
||||
sock.send(msg, 0)
|
||||
|
||||
global genl_family_name_to_id
|
||||
genl_family_name_to_id = dict()
|
||||
|
||||
while True:
|
||||
reply = sock.recv(128 * 1024)
|
||||
nms = NlMsgs(reply)
|
||||
for nl_msg in nms:
|
||||
if nl_msg.error:
|
||||
print("Netlink error:", nl_msg.error)
|
||||
return
|
||||
if nl_msg.done:
|
||||
return
|
||||
|
||||
gm = GenlMsg(nl_msg)
|
||||
fam = dict()
|
||||
for attr in gm.raw_attrs:
|
||||
if attr.type == Netlink.CTRL_ATTR_FAMILY_ID:
|
||||
fam['id'] = attr.as_u16()
|
||||
elif attr.type == Netlink.CTRL_ATTR_FAMILY_NAME:
|
||||
fam['name'] = attr.as_strz()
|
||||
elif attr.type == Netlink.CTRL_ATTR_MAXATTR:
|
||||
fam['maxattr'] = attr.as_u32()
|
||||
elif attr.type == Netlink.CTRL_ATTR_MCAST_GROUPS:
|
||||
fam['mcast'] = dict()
|
||||
for entry in NlAttrs(attr.raw):
|
||||
mcast_name = None
|
||||
mcast_id = None
|
||||
for entry_attr in NlAttrs(entry.raw):
|
||||
if entry_attr.type == Netlink.CTRL_ATTR_MCAST_GRP_NAME:
|
||||
mcast_name = entry_attr.as_strz()
|
||||
elif entry_attr.type == Netlink.CTRL_ATTR_MCAST_GRP_ID:
|
||||
mcast_id = entry_attr.as_u32()
|
||||
if mcast_name and mcast_id is not None:
|
||||
fam['mcast'][mcast_name] = mcast_id
|
||||
if 'name' in fam and 'id' in fam:
|
||||
genl_family_name_to_id[fam['name']] = fam
|
||||
|
||||
|
||||
class GenlMsg:
|
||||
def __init__(self, nl_msg):
|
||||
self.nl = nl_msg
|
||||
|
||||
self.hdr = nl_msg.raw[0:4]
|
||||
self.raw = nl_msg.raw[4:]
|
||||
|
||||
self.genl_cmd, self.genl_version, _ = struct.unpack("bbH", self.hdr)
|
||||
|
||||
self.raw_attrs = NlAttrs(self.raw)
|
||||
|
||||
def __repr__(self):
|
||||
msg = repr(self.nl)
|
||||
msg += f"\tgenl_cmd = {self.genl_cmd} genl_ver = {self.genl_version}\n"
|
||||
for a in self.raw_attrs:
|
||||
msg += '\t\t' + repr(a) + '\n'
|
||||
return msg
|
||||
|
||||
|
||||
class GenlFamily:
|
||||
def __init__(self, family_name):
|
||||
self.family_name = family_name
|
||||
|
||||
global genl_family_name_to_id
|
||||
if genl_family_name_to_id is None:
|
||||
_genl_load_families()
|
||||
|
||||
self.genl_family = genl_family_name_to_id[family_name]
|
||||
self.family_id = genl_family_name_to_id[family_name]['id']
|
||||
|
||||
|
||||
#
|
||||
# YNL implementation details.
|
||||
#
|
||||
|
||||
|
||||
class YnlAttrSpace:
|
||||
def __init__(self, family, yaml):
|
||||
self.yaml = yaml
|
||||
|
||||
self.attrs = dict()
|
||||
self.name = self.yaml['name']
|
||||
self.subspace_of = self.yaml['subset-of'] if 'subspace-of' in self.yaml else None
|
||||
|
||||
val = 0
|
||||
max_val = 0
|
||||
for elem in self.yaml['attributes']:
|
||||
if 'value' in elem:
|
||||
val = elem['value']
|
||||
else:
|
||||
elem['value'] = val
|
||||
if val > max_val:
|
||||
max_val = val
|
||||
val += 1
|
||||
|
||||
self.attrs[elem['name']] = elem
|
||||
|
||||
self.attr_list = [None] * (max_val + 1)
|
||||
for elem in self.yaml['attributes']:
|
||||
self.attr_list[elem['value']] = elem
|
||||
|
||||
def __getitem__(self, key):
|
||||
return self.attrs[key]
|
||||
|
||||
def __contains__(self, key):
|
||||
return key in self.yaml
|
||||
|
||||
def __iter__(self):
|
||||
yield from self.attrs
|
||||
|
||||
def items(self):
|
||||
return self.attrs.items()
|
||||
|
||||
|
||||
class YnlFamily:
|
||||
def __init__(self, def_path, schema=None):
|
||||
self.include_raw = False
|
||||
|
||||
with open(def_path, "r") as stream:
|
||||
self.yaml = yaml.safe_load(stream)
|
||||
|
||||
if schema:
|
||||
with open(schema, "r") as stream:
|
||||
schema = yaml.safe_load(stream)
|
||||
|
||||
jsonschema.validate(self.yaml, schema)
|
||||
|
||||
self.sock = socket.socket(socket.AF_NETLINK, socket.SOCK_RAW, Netlink.NETLINK_GENERIC)
|
||||
self.sock.setsockopt(Netlink.SOL_NETLINK, Netlink.NETLINK_CAP_ACK, 1)
|
||||
self.sock.setsockopt(Netlink.SOL_NETLINK, Netlink.NETLINK_EXT_ACK, 1)
|
||||
|
||||
self._ops = dict()
|
||||
self._spaces = dict()
|
||||
self._types = dict()
|
||||
|
||||
for elem in self.yaml['attribute-sets']:
|
||||
self._spaces[elem['name']] = YnlAttrSpace(self, elem)
|
||||
|
||||
for elem in self.yaml['definitions']:
|
||||
self._types[elem['name']] = elem
|
||||
|
||||
async_separation = 'async-prefix' in self.yaml['operations']
|
||||
self.async_msg_ids = set()
|
||||
self.async_msg_queue = []
|
||||
val = 0
|
||||
max_val = 0
|
||||
for elem in self.yaml['operations']['list']:
|
||||
if not (async_separation and ('notify' in elem or 'event' in elem)):
|
||||
if 'value' in elem:
|
||||
val = elem['value']
|
||||
else:
|
||||
elem['value'] = val
|
||||
val += 1
|
||||
max_val = max(val, max_val)
|
||||
|
||||
if 'notify' in elem or 'event' in elem:
|
||||
self.async_msg_ids.add(elem['value'])
|
||||
|
||||
self._ops[elem['name']] = elem
|
||||
|
||||
op_name = elem['name'].replace('-', '_')
|
||||
|
||||
bound_f = functools.partial(self._op, elem['name'])
|
||||
setattr(self, op_name, bound_f)
|
||||
|
||||
self._op_array = [None] * max_val
|
||||
for _, op in self._ops.items():
|
||||
self._op_array[op['value']] = op
|
||||
if 'notify' in op:
|
||||
op['attribute-set'] = self._ops[op['notify']]['attribute-set']
|
||||
|
||||
self.family = GenlFamily(self.yaml['name'])
|
||||
|
||||
def ntf_subscribe(self, mcast_name):
|
||||
if mcast_name not in self.family.genl_family['mcast']:
|
||||
raise Exception(f'Multicast group "{mcast_name}" not present in the family')
|
||||
|
||||
self.sock.bind((0, 0))
|
||||
self.sock.setsockopt(Netlink.SOL_NETLINK, Netlink.NETLINK_ADD_MEMBERSHIP,
|
||||
self.family.genl_family['mcast'][mcast_name])
|
||||
|
||||
def _add_attr(self, space, name, value):
|
||||
attr = self._spaces[space][name]
|
||||
nl_type = attr['value']
|
||||
if attr["type"] == 'nest':
|
||||
nl_type |= Netlink.NLA_F_NESTED
|
||||
attr_payload = b''
|
||||
for subname, subvalue in value.items():
|
||||
attr_payload += self._add_attr(attr['nested-attributes'], subname, subvalue)
|
||||
elif attr["type"] == 'u32':
|
||||
attr_payload = struct.pack("I", int(value))
|
||||
elif attr["type"] == 'string':
|
||||
attr_payload = str(value).encode('ascii') + b'\x00'
|
||||
elif attr["type"] == 'binary':
|
||||
attr_payload = value
|
||||
else:
|
||||
raise Exception(f'Unknown type at {space} {name} {value} {attr["type"]}')
|
||||
|
||||
pad = b'\x00' * ((4 - len(attr_payload) % 4) % 4)
|
||||
return struct.pack('HH', len(attr_payload) + 4, nl_type) + attr_payload + pad
|
||||
|
||||
def _decode_enum(self, rsp, attr_spec):
|
||||
raw = rsp[attr_spec['name']]
|
||||
enum = self._types[attr_spec['enum']]
|
||||
i = attr_spec.get('value-start', 0)
|
||||
if 'enum-as-flags' in attr_spec and attr_spec['enum-as-flags']:
|
||||
value = set()
|
||||
while raw:
|
||||
if raw & 1:
|
||||
value.add(enum['entries'][i])
|
||||
raw >>= 1
|
||||
i += 1
|
||||
else:
|
||||
value = enum['entries'][raw - i]
|
||||
rsp[attr_spec['name']] = value
|
||||
|
||||
def _decode(self, attrs, space):
|
||||
attr_space = self._spaces[space]
|
||||
rsp = dict()
|
||||
for attr in attrs:
|
||||
attr_spec = attr_space.attr_list[attr.type]
|
||||
if attr_spec["type"] == 'nest':
|
||||
subdict = self._decode(NlAttrs(attr.raw), attr_spec['nested-attributes'])
|
||||
rsp[attr_spec['name']] = subdict
|
||||
elif attr_spec['type'] == 'u32':
|
||||
rsp[attr_spec['name']] = attr.as_u32()
|
||||
elif attr_spec['type'] == 'u64':
|
||||
rsp[attr_spec['name']] = attr.as_u64()
|
||||
elif attr_spec["type"] == 'string':
|
||||
rsp[attr_spec['name']] = attr.as_strz()
|
||||
elif attr_spec["type"] == 'binary':
|
||||
rsp[attr_spec['name']] = attr.as_bin()
|
||||
else:
|
||||
raise Exception(f'Unknown {attr.type} {attr_spec["name"]} {attr_spec["type"]}')
|
||||
|
||||
if 'enum' in attr_spec:
|
||||
self._decode_enum(rsp, attr_spec)
|
||||
return rsp
|
||||
|
||||
def handle_ntf(self, nl_msg, genl_msg):
|
||||
msg = dict()
|
||||
if self.include_raw:
|
||||
msg['nlmsg'] = nl_msg
|
||||
msg['genlmsg'] = genl_msg
|
||||
op = self._op_array[genl_msg.genl_cmd]
|
||||
msg['name'] = op['name']
|
||||
msg['msg'] = self._decode(genl_msg.raw_attrs, op['attribute-set'])
|
||||
self.async_msg_queue.append(msg)
|
||||
|
||||
def check_ntf(self):
|
||||
while True:
|
||||
try:
|
||||
reply = self.sock.recv(128 * 1024, socket.MSG_DONTWAIT)
|
||||
except BlockingIOError:
|
||||
return
|
||||
|
||||
nms = NlMsgs(reply)
|
||||
for nl_msg in nms:
|
||||
if nl_msg.error:
|
||||
print("Netlink error in ntf!?", os.strerror(-nl_msg.error))
|
||||
print(nl_msg)
|
||||
continue
|
||||
if nl_msg.done:
|
||||
print("Netlink done while checking for ntf!?")
|
||||
continue
|
||||
|
||||
gm = GenlMsg(nl_msg)
|
||||
if gm.genl_cmd not in self.async_msg_ids:
|
||||
print("Unexpected msg id done while checking for ntf", gm)
|
||||
continue
|
||||
|
||||
self.handle_ntf(nl_msg, gm)
|
||||
|
||||
def _op(self, method, vals, dump=False):
|
||||
op = self._ops[method]
|
||||
|
||||
nl_flags = Netlink.NLM_F_REQUEST | Netlink.NLM_F_ACK
|
||||
if dump:
|
||||
nl_flags |= Netlink.NLM_F_DUMP
|
||||
|
||||
req_seq = random.randint(1024, 65535)
|
||||
msg = _genl_msg(self.family.family_id, nl_flags, op['value'], 1, req_seq)
|
||||
for name, value in vals.items():
|
||||
msg += self._add_attr(op['attribute-set'], name, value)
|
||||
msg = _genl_msg_finalize(msg)
|
||||
|
||||
self.sock.send(msg, 0)
|
||||
|
||||
done = False
|
||||
rsp = []
|
||||
while not done:
|
||||
reply = self.sock.recv(128 * 1024)
|
||||
nms = NlMsgs(reply, attr_space=self._spaces[op['attribute-set']])
|
||||
for nl_msg in nms:
|
||||
if nl_msg.error:
|
||||
print("Netlink error:", os.strerror(-nl_msg.error))
|
||||
print(nl_msg)
|
||||
return
|
||||
if nl_msg.done:
|
||||
done = True
|
||||
break
|
||||
|
||||
gm = GenlMsg(nl_msg)
|
||||
# Check if this is a reply to our request
|
||||
if nl_msg.nl_seq != req_seq or gm.genl_cmd != op['value']:
|
||||
if gm.genl_cmd in self.async_msg_ids:
|
||||
self.handle_ntf(nl_msg, gm)
|
||||
continue
|
||||
else:
|
||||
print('Unexpected message: ' + repr(gm))
|
||||
continue
|
||||
|
||||
rsp.append(self._decode(gm.raw_attrs, op['attribute-set']))
|
||||
|
||||
if not rsp:
|
||||
return None
|
||||
if not dump and len(rsp) == 1:
|
||||
return rsp[0]
|
||||
return rsp
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,30 @@
|
|||
#!/bin/bash
|
||||
# SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
TOOL=$(dirname $(realpath $0))/ynl-gen-c.py
|
||||
|
||||
force=
|
||||
|
||||
while [ ! -z "$1" ]; do
|
||||
case "$1" in
|
||||
-f ) force=yes; shift ;;
|
||||
* ) echo "Unrecognized option '$1'"; exit 1 ;;
|
||||
esac
|
||||
done
|
||||
|
||||
KDIR=$(dirname $(dirname $(dirname $(dirname $(realpath $0)))))
|
||||
|
||||
files=$(git grep --files-with-matches '^/\* YNL-GEN \(kernel\|uapi\)')
|
||||
for f in $files; do
|
||||
# params: 0 1 2 3
|
||||
# $YAML YNL-GEN kernel $mode
|
||||
params=( $(git grep -B1 -h '/\* YNL-GEN' $f | sed 's@/\*\(.*\)\*/@\1@') )
|
||||
|
||||
if [ $f -nt ${params[0]} -a -z "$force" ]; then
|
||||
echo -e "\tSKIP $f"
|
||||
continue
|
||||
fi
|
||||
|
||||
echo -e "\tGEN ${params[2]}\t$f"
|
||||
$TOOL --mode ${params[2]} --${params[3]} --spec $KDIR/${params[0]} -o $f
|
||||
done
|
Loading…
Reference in New Issue