platform/surface: Add Surface Aggregator user-space interface
Add a misc-device providing user-space access to the Surface Aggregator EC, mainly intended for debugging, testing, and reverse-engineering. This interface gives user-space applications the ability to send requests to the EC and receive the corresponding responses. The device-file is managed by a pseudo platform-device and corresponding driver to avoid dependence on the dedicated bus, allowing it to be loaded in a minimal configuration. A python library and scripts to access this device can be found at [1]. [1]: https://github.com/linux-surface/surface-aggregator-module/tree/master/scripts/ssam Signed-off-by: Maximilian Luz <luzmaximilian@gmail.com> Link: https://lore.kernel.org/r/20201221183959.1186143-9-luzmaximilian@gmail.com Signed-off-by: Hans de Goede <hdegoede@redhat.com>
This commit is contained in:
parent
8d7792823d
commit
178f6ab77e
87
Documentation/driver-api/surface_aggregator/clients/cdev.rst
Normal file
87
Documentation/driver-api/surface_aggregator/clients/cdev.rst
Normal file
@ -0,0 +1,87 @@
|
||||
.. SPDX-License-Identifier: GPL-2.0+
|
||||
|
||||
.. |u8| replace:: :c:type:`u8 <u8>`
|
||||
.. |u16| replace:: :c:type:`u16 <u16>`
|
||||
.. |ssam_cdev_request| replace:: :c:type:`struct ssam_cdev_request <ssam_cdev_request>`
|
||||
.. |ssam_cdev_request_flags| replace:: :c:type:`enum ssam_cdev_request_flags <ssam_cdev_request_flags>`
|
||||
|
||||
==============================
|
||||
User-Space EC Interface (cdev)
|
||||
==============================
|
||||
|
||||
The ``surface_aggregator_cdev`` module provides a misc-device for the SSAM
|
||||
controller to allow for a (more or less) direct connection from user-space to
|
||||
the SAM EC. It is intended to be used for development and debugging, and
|
||||
therefore should not be used or relied upon in any other way. Note that this
|
||||
module is not loaded automatically, but instead must be loaded manually.
|
||||
|
||||
The provided interface is accessible through the ``/dev/surface/aggregator``
|
||||
device-file. All functionality of this interface is provided via IOCTLs.
|
||||
These IOCTLs and their respective input/output parameter structs are defined in
|
||||
``include/uapi/linux/surface_aggregator/cdev.h``.
|
||||
|
||||
A small python library and scripts for accessing this interface can be found
|
||||
at https://github.com/linux-surface/surface-aggregator-module/tree/master/scripts/ssam.
|
||||
|
||||
|
||||
Controller IOCTLs
|
||||
=================
|
||||
|
||||
The following IOCTLs are provided:
|
||||
|
||||
.. flat-table:: Controller IOCTLs
|
||||
:widths: 1 1 1 1 4
|
||||
:header-rows: 1
|
||||
|
||||
* - Type
|
||||
- Number
|
||||
- Direction
|
||||
- Name
|
||||
- Description
|
||||
|
||||
* - ``0xA5``
|
||||
- ``1``
|
||||
- ``WR``
|
||||
- ``REQUEST``
|
||||
- Perform synchronous SAM request.
|
||||
|
||||
|
||||
``REQUEST``
|
||||
-----------
|
||||
|
||||
Defined as ``_IOWR(0xA5, 1, struct ssam_cdev_request)``.
|
||||
|
||||
Executes a synchronous SAM request. The request specification is passed in
|
||||
as argument of type |ssam_cdev_request|, which is then written to/modified
|
||||
by the IOCTL to return status and result of the request.
|
||||
|
||||
Request payload data must be allocated separately and is passed in via the
|
||||
``payload.data`` and ``payload.length`` members. If a response is required,
|
||||
the response buffer must be allocated by the caller and passed in via the
|
||||
``response.data`` member. The ``response.length`` member must be set to the
|
||||
capacity of this buffer, or if no response is required, zero. Upon
|
||||
completion of the request, the call will write the response to the response
|
||||
buffer (if its capacity allows it) and overwrite the length field with the
|
||||
actual size of the response, in bytes.
|
||||
|
||||
Additionally, if the request has a response, this must be indicated via the
|
||||
request flags, as is done with in-kernel requests. Request flags can be set
|
||||
via the ``flags`` member and the values correspond to the values found in
|
||||
|ssam_cdev_request_flags|.
|
||||
|
||||
Finally, the status of the request itself is returned in the ``status``
|
||||
member (a negative errno value indicating failure). Note that failure
|
||||
indication of the IOCTL is separated from failure indication of the request:
|
||||
The IOCTL returns a negative status code if anything failed during setup of
|
||||
the request (``-EFAULT``) or if the provided argument or any of its fields
|
||||
are invalid (``-EINVAL``). In this case, the status value of the request
|
||||
argument may be set, providing more detail on what went wrong (e.g.
|
||||
``-ENOMEM`` for out-of-memory), but this value may also be zero. The IOCTL
|
||||
will return with a zero status code in case the request has been set up,
|
||||
submitted, and completed (i.e. handed back to user-space) successfully from
|
||||
inside the IOCTL, but the request ``status`` member may still be negative in
|
||||
case the actual execution of the request failed after it has been submitted.
|
||||
|
||||
A full definition of the argument struct is provided below:
|
||||
|
||||
.. kernel-doc:: include/uapi/linux/surface_aggregator/cdev.h
|
@ -7,4 +7,14 @@ Client Driver Documentation
|
||||
This is the documentation for client drivers themselves. Refer to
|
||||
:doc:`../client` for documentation on how to write client drivers.
|
||||
|
||||
.. Place documentation for individual client drivers here.
|
||||
.. toctree::
|
||||
:maxdepth: 1
|
||||
|
||||
cdev
|
||||
|
||||
.. only:: subproject and html
|
||||
|
||||
Indices
|
||||
=======
|
||||
|
||||
* :ref:`genindex`
|
||||
|
@ -324,6 +324,8 @@ Code Seq# Include File Comments
|
||||
0xA3 90-9F linux/dtlk.h
|
||||
0xA4 00-1F uapi/linux/tee.h Generic TEE subsystem
|
||||
0xA4 00-1F uapi/asm/sgx.h <mailto:linux-sgx@vger.kernel.org>
|
||||
0xA5 01 linux/surface_aggregator/cdev.h Microsoft Surface Platform System Aggregator
|
||||
<mailto:luzmaximilian@gmail.com>
|
||||
0xAA 00-3F linux/uapi/linux/userfaultfd.h
|
||||
0xAB 00-1F linux/nbd.h
|
||||
0xAC 00-1F linux/raw.h
|
||||
|
@ -11820,7 +11820,9 @@ W: https://github.com/linux-surface/surface-aggregator-module
|
||||
C: irc://chat.freenode.net/##linux-surface
|
||||
F: Documentation/driver-api/surface_aggregator/
|
||||
F: drivers/platform/surface/aggregator/
|
||||
F: drivers/platform/surface/surface_aggregator_cdev.c
|
||||
F: include/linux/surface_aggregator/
|
||||
F: include/uapi/linux/surface_aggregator/
|
||||
|
||||
MICROTEK X6 SCANNER
|
||||
M: Oliver Neukum <oliver@neukum.org>
|
||||
|
@ -41,6 +41,23 @@ config SURFACE_3_POWER_OPREGION
|
||||
This driver provides support for ACPI operation
|
||||
region of the Surface 3 battery platform driver.
|
||||
|
||||
config SURFACE_AGGREGATOR_CDEV
|
||||
tristate "Surface System Aggregator Module User-Space Interface"
|
||||
depends on SURFACE_AGGREGATOR
|
||||
help
|
||||
Provides a misc-device interface to the Surface System Aggregator
|
||||
Module (SSAM) controller.
|
||||
|
||||
This option provides a module (called surface_aggregator_cdev), that,
|
||||
when loaded, will add a client device (and its respective driver) to
|
||||
the SSAM controller. Said client device manages a misc-device
|
||||
interface (/dev/surface/aggregator), which can be used by user-space
|
||||
tools to directly communicate with the SSAM EC by sending requests and
|
||||
receiving the corresponding responses.
|
||||
|
||||
The provided interface is intended for debugging and development only,
|
||||
and should not be used otherwise.
|
||||
|
||||
config SURFACE_GPE
|
||||
tristate "Surface GPE/Lid Support Driver"
|
||||
depends on DMI
|
||||
|
@ -8,5 +8,6 @@ obj-$(CONFIG_SURFACE3_WMI) += surface3-wmi.o
|
||||
obj-$(CONFIG_SURFACE_3_BUTTON) += surface3_button.o
|
||||
obj-$(CONFIG_SURFACE_3_POWER_OPREGION) += surface3_power.o
|
||||
obj-$(CONFIG_SURFACE_AGGREGATOR) += aggregator/
|
||||
obj-$(CONFIG_SURFACE_AGGREGATOR_CDEV) += surface_aggregator_cdev.o
|
||||
obj-$(CONFIG_SURFACE_GPE) += surface_gpe.o
|
||||
obj-$(CONFIG_SURFACE_PRO3_BUTTON) += surfacepro3_button.o
|
||||
|
303
drivers/platform/surface/surface_aggregator_cdev.c
Normal file
303
drivers/platform/surface/surface_aggregator_cdev.c
Normal file
@ -0,0 +1,303 @@
|
||||
// SPDX-License-Identifier: GPL-2.0+
|
||||
/*
|
||||
* Provides user-space access to the SSAM EC via the /dev/surface/aggregator
|
||||
* misc device. Intended for debugging and development.
|
||||
*
|
||||
* Copyright (C) 2020 Maximilian Luz <luzmaximilian@gmail.com>
|
||||
*/
|
||||
|
||||
#include <linux/fs.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/kref.h>
|
||||
#include <linux/miscdevice.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/rwsem.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/uaccess.h>
|
||||
|
||||
#include <linux/surface_aggregator/cdev.h>
|
||||
#include <linux/surface_aggregator/controller.h>
|
||||
|
||||
#define SSAM_CDEV_DEVICE_NAME "surface_aggregator_cdev"
|
||||
|
||||
struct ssam_cdev {
|
||||
struct kref kref;
|
||||
struct rw_semaphore lock;
|
||||
struct ssam_controller *ctrl;
|
||||
struct miscdevice mdev;
|
||||
};
|
||||
|
||||
static void __ssam_cdev_release(struct kref *kref)
|
||||
{
|
||||
kfree(container_of(kref, struct ssam_cdev, kref));
|
||||
}
|
||||
|
||||
static struct ssam_cdev *ssam_cdev_get(struct ssam_cdev *cdev)
|
||||
{
|
||||
if (cdev)
|
||||
kref_get(&cdev->kref);
|
||||
|
||||
return cdev;
|
||||
}
|
||||
|
||||
static void ssam_cdev_put(struct ssam_cdev *cdev)
|
||||
{
|
||||
if (cdev)
|
||||
kref_put(&cdev->kref, __ssam_cdev_release);
|
||||
}
|
||||
|
||||
static int ssam_cdev_device_open(struct inode *inode, struct file *filp)
|
||||
{
|
||||
struct miscdevice *mdev = filp->private_data;
|
||||
struct ssam_cdev *cdev = container_of(mdev, struct ssam_cdev, mdev);
|
||||
|
||||
filp->private_data = ssam_cdev_get(cdev);
|
||||
return stream_open(inode, filp);
|
||||
}
|
||||
|
||||
static int ssam_cdev_device_release(struct inode *inode, struct file *filp)
|
||||
{
|
||||
ssam_cdev_put(filp->private_data);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static long ssam_cdev_request(struct ssam_cdev *cdev, unsigned long arg)
|
||||
{
|
||||
struct ssam_cdev_request __user *r;
|
||||
struct ssam_cdev_request rqst;
|
||||
struct ssam_request spec;
|
||||
struct ssam_response rsp;
|
||||
const void __user *plddata;
|
||||
void __user *rspdata;
|
||||
int status = 0, ret = 0, tmp;
|
||||
|
||||
r = (struct ssam_cdev_request __user *)arg;
|
||||
ret = copy_struct_from_user(&rqst, sizeof(rqst), r, sizeof(*r));
|
||||
if (ret)
|
||||
goto out;
|
||||
|
||||
plddata = u64_to_user_ptr(rqst.payload.data);
|
||||
rspdata = u64_to_user_ptr(rqst.response.data);
|
||||
|
||||
/* Setup basic request fields. */
|
||||
spec.target_category = rqst.target_category;
|
||||
spec.target_id = rqst.target_id;
|
||||
spec.command_id = rqst.command_id;
|
||||
spec.instance_id = rqst.instance_id;
|
||||
spec.flags = 0;
|
||||
spec.length = rqst.payload.length;
|
||||
spec.payload = NULL;
|
||||
|
||||
if (rqst.flags & SSAM_CDEV_REQUEST_HAS_RESPONSE)
|
||||
spec.flags |= SSAM_REQUEST_HAS_RESPONSE;
|
||||
|
||||
if (rqst.flags & SSAM_CDEV_REQUEST_UNSEQUENCED)
|
||||
spec.flags |= SSAM_REQUEST_UNSEQUENCED;
|
||||
|
||||
rsp.capacity = rqst.response.length;
|
||||
rsp.length = 0;
|
||||
rsp.pointer = NULL;
|
||||
|
||||
/* Get request payload from user-space. */
|
||||
if (spec.length) {
|
||||
if (!plddata) {
|
||||
ret = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
spec.payload = kzalloc(spec.length, GFP_KERNEL);
|
||||
if (!spec.payload) {
|
||||
ret = -ENOMEM;
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (copy_from_user((void *)spec.payload, plddata, spec.length)) {
|
||||
ret = -EFAULT;
|
||||
goto out;
|
||||
}
|
||||
}
|
||||
|
||||
/* Allocate response buffer. */
|
||||
if (rsp.capacity) {
|
||||
if (!rspdata) {
|
||||
ret = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
rsp.pointer = kzalloc(rsp.capacity, GFP_KERNEL);
|
||||
if (!rsp.pointer) {
|
||||
ret = -ENOMEM;
|
||||
goto out;
|
||||
}
|
||||
}
|
||||
|
||||
/* Perform request. */
|
||||
status = ssam_request_sync(cdev->ctrl, &spec, &rsp);
|
||||
if (status)
|
||||
goto out;
|
||||
|
||||
/* Copy response to user-space. */
|
||||
if (rsp.length && copy_to_user(rspdata, rsp.pointer, rsp.length))
|
||||
ret = -EFAULT;
|
||||
|
||||
out:
|
||||
/* Always try to set response-length and status. */
|
||||
tmp = put_user(rsp.length, &r->response.length);
|
||||
if (tmp)
|
||||
ret = tmp;
|
||||
|
||||
tmp = put_user(status, &r->status);
|
||||
if (tmp)
|
||||
ret = tmp;
|
||||
|
||||
/* Cleanup. */
|
||||
kfree(spec.payload);
|
||||
kfree(rsp.pointer);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static long __ssam_cdev_device_ioctl(struct ssam_cdev *cdev, unsigned int cmd,
|
||||
unsigned long arg)
|
||||
{
|
||||
switch (cmd) {
|
||||
case SSAM_CDEV_REQUEST:
|
||||
return ssam_cdev_request(cdev, arg);
|
||||
|
||||
default:
|
||||
return -ENOTTY;
|
||||
}
|
||||
}
|
||||
|
||||
static long ssam_cdev_device_ioctl(struct file *file, unsigned int cmd,
|
||||
unsigned long arg)
|
||||
{
|
||||
struct ssam_cdev *cdev = file->private_data;
|
||||
long status;
|
||||
|
||||
/* Ensure that controller is valid for as long as we need it. */
|
||||
if (down_read_killable(&cdev->lock))
|
||||
return -ERESTARTSYS;
|
||||
|
||||
if (!cdev->ctrl) {
|
||||
up_read(&cdev->lock);
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
status = __ssam_cdev_device_ioctl(cdev, cmd, arg);
|
||||
|
||||
up_read(&cdev->lock);
|
||||
return status;
|
||||
}
|
||||
|
||||
static const struct file_operations ssam_controller_fops = {
|
||||
.owner = THIS_MODULE,
|
||||
.open = ssam_cdev_device_open,
|
||||
.release = ssam_cdev_device_release,
|
||||
.unlocked_ioctl = ssam_cdev_device_ioctl,
|
||||
.compat_ioctl = ssam_cdev_device_ioctl,
|
||||
.llseek = noop_llseek,
|
||||
};
|
||||
|
||||
static int ssam_dbg_device_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct ssam_controller *ctrl;
|
||||
struct ssam_cdev *cdev;
|
||||
int status;
|
||||
|
||||
ctrl = ssam_client_bind(&pdev->dev);
|
||||
if (IS_ERR(ctrl))
|
||||
return PTR_ERR(ctrl) == -ENODEV ? -EPROBE_DEFER : PTR_ERR(ctrl);
|
||||
|
||||
cdev = kzalloc(sizeof(*cdev), GFP_KERNEL);
|
||||
if (!cdev)
|
||||
return -ENOMEM;
|
||||
|
||||
kref_init(&cdev->kref);
|
||||
init_rwsem(&cdev->lock);
|
||||
cdev->ctrl = ctrl;
|
||||
|
||||
cdev->mdev.parent = &pdev->dev;
|
||||
cdev->mdev.minor = MISC_DYNAMIC_MINOR;
|
||||
cdev->mdev.name = "surface_aggregator";
|
||||
cdev->mdev.nodename = "surface/aggregator";
|
||||
cdev->mdev.fops = &ssam_controller_fops;
|
||||
|
||||
status = misc_register(&cdev->mdev);
|
||||
if (status) {
|
||||
kfree(cdev);
|
||||
return status;
|
||||
}
|
||||
|
||||
platform_set_drvdata(pdev, cdev);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ssam_dbg_device_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct ssam_cdev *cdev = platform_get_drvdata(pdev);
|
||||
|
||||
misc_deregister(&cdev->mdev);
|
||||
|
||||
/*
|
||||
* The controller is only guaranteed to be valid for as long as the
|
||||
* driver is bound. Remove controller so that any lingering open files
|
||||
* cannot access it any more after we're gone.
|
||||
*/
|
||||
down_write(&cdev->lock);
|
||||
cdev->ctrl = NULL;
|
||||
up_write(&cdev->lock);
|
||||
|
||||
ssam_cdev_put(cdev);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct platform_device *ssam_cdev_device;
|
||||
|
||||
static struct platform_driver ssam_cdev_driver = {
|
||||
.probe = ssam_dbg_device_probe,
|
||||
.remove = ssam_dbg_device_remove,
|
||||
.driver = {
|
||||
.name = SSAM_CDEV_DEVICE_NAME,
|
||||
.probe_type = PROBE_PREFER_ASYNCHRONOUS,
|
||||
},
|
||||
};
|
||||
|
||||
static int __init ssam_debug_init(void)
|
||||
{
|
||||
int status;
|
||||
|
||||
ssam_cdev_device = platform_device_alloc(SSAM_CDEV_DEVICE_NAME,
|
||||
PLATFORM_DEVID_NONE);
|
||||
if (!ssam_cdev_device)
|
||||
return -ENOMEM;
|
||||
|
||||
status = platform_device_add(ssam_cdev_device);
|
||||
if (status)
|
||||
goto err_device;
|
||||
|
||||
status = platform_driver_register(&ssam_cdev_driver);
|
||||
if (status)
|
||||
goto err_driver;
|
||||
|
||||
return 0;
|
||||
|
||||
err_driver:
|
||||
platform_device_del(ssam_cdev_device);
|
||||
err_device:
|
||||
platform_device_put(ssam_cdev_device);
|
||||
return status;
|
||||
}
|
||||
module_init(ssam_debug_init);
|
||||
|
||||
static void __exit ssam_debug_exit(void)
|
||||
{
|
||||
platform_driver_unregister(&ssam_cdev_driver);
|
||||
platform_device_unregister(ssam_cdev_device);
|
||||
}
|
||||
module_exit(ssam_debug_exit);
|
||||
|
||||
MODULE_AUTHOR("Maximilian Luz <luzmaximilian@gmail.com>");
|
||||
MODULE_DESCRIPTION("User-space interface for Surface System Aggregator Module");
|
||||
MODULE_LICENSE("GPL");
|
78
include/uapi/linux/surface_aggregator/cdev.h
Normal file
78
include/uapi/linux/surface_aggregator/cdev.h
Normal file
@ -0,0 +1,78 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0+ WITH Linux-syscall-note */
|
||||
/*
|
||||
* Surface System Aggregator Module (SSAM) user-space EC interface.
|
||||
*
|
||||
* Definitions, structs, and IOCTLs for the /dev/surface/aggregator misc
|
||||
* device. This device provides direct user-space access to the SSAM EC.
|
||||
* Intended for debugging and development.
|
||||
*
|
||||
* Copyright (C) 2020 Maximilian Luz <luzmaximilian@gmail.com>
|
||||
*/
|
||||
|
||||
#ifndef _UAPI_LINUX_SURFACE_AGGREGATOR_CDEV_H
|
||||
#define _UAPI_LINUX_SURFACE_AGGREGATOR_CDEV_H
|
||||
|
||||
#include <linux/ioctl.h>
|
||||
#include <linux/types.h>
|
||||
|
||||
/**
|
||||
* enum ssam_cdev_request_flags - Request flags for SSAM cdev request IOCTL.
|
||||
*
|
||||
* @SSAM_CDEV_REQUEST_HAS_RESPONSE:
|
||||
* Specifies that the request expects a response. If not set, the request
|
||||
* will be directly completed after its underlying packet has been
|
||||
* transmitted. If set, the request transport system waits for a response
|
||||
* of the request.
|
||||
*
|
||||
* @SSAM_CDEV_REQUEST_UNSEQUENCED:
|
||||
* Specifies that the request should be transmitted via an unsequenced
|
||||
* packet. If set, the request must not have a response, meaning that this
|
||||
* flag and the %SSAM_CDEV_REQUEST_HAS_RESPONSE flag are mutually
|
||||
* exclusive.
|
||||
*/
|
||||
enum ssam_cdev_request_flags {
|
||||
SSAM_CDEV_REQUEST_HAS_RESPONSE = 0x01,
|
||||
SSAM_CDEV_REQUEST_UNSEQUENCED = 0x02,
|
||||
};
|
||||
|
||||
/**
|
||||
* struct ssam_cdev_request - Controller request IOCTL argument.
|
||||
* @target_category: Target category of the SAM request.
|
||||
* @target_id: Target ID of the SAM request.
|
||||
* @command_id: Command ID of the SAM request.
|
||||
* @instance_id: Instance ID of the SAM request.
|
||||
* @flags: Request flags (see &enum ssam_cdev_request_flags).
|
||||
* @status: Request status (output).
|
||||
* @payload: Request payload (input data).
|
||||
* @payload.data: Pointer to request payload data.
|
||||
* @payload.length: Length of request payload data (in bytes).
|
||||
* @response: Request response (output data).
|
||||
* @response.data: Pointer to response buffer.
|
||||
* @response.length: On input: Capacity of response buffer (in bytes).
|
||||
* On output: Length of request response (number of bytes
|
||||
* in the buffer that are actually used).
|
||||
*/
|
||||
struct ssam_cdev_request {
|
||||
__u8 target_category;
|
||||
__u8 target_id;
|
||||
__u8 command_id;
|
||||
__u8 instance_id;
|
||||
__u16 flags;
|
||||
__s16 status;
|
||||
|
||||
struct {
|
||||
__u64 data;
|
||||
__u16 length;
|
||||
__u8 __pad[6];
|
||||
} payload;
|
||||
|
||||
struct {
|
||||
__u64 data;
|
||||
__u16 length;
|
||||
__u8 __pad[6];
|
||||
} response;
|
||||
} __attribute__((__packed__));
|
||||
|
||||
#define SSAM_CDEV_REQUEST _IOWR(0xA5, 1, struct ssam_cdev_request)
|
||||
|
||||
#endif /* _UAPI_LINUX_SURFACE_AGGREGATOR_CDEV_H */
|
Loading…
x
Reference in New Issue
Block a user