Prechádzať zdrojové kódy

ipc: Add support for GPR

Add support for GPR and Audio-pkt drivers for AudioReach.

Change-Id: I4813ec1e3696900ebd426caa3017eae623b3a6c9
Signed-off-by: Taha Azzaoui <[email protected]>
Taha Azzaoui 4 rokov pred
rodič
commit
5986527a1e
4 zmenil súbory, kde vykonal 1186 pridanie a 25 odobranie
  1. 278 0
      include/ipc/gpr-lite.h
  2. 10 25
      ipc/Kbuild
  3. 523 0
      ipc/audio-pkt.c
  4. 375 0
      ipc/gpr-lite.c

+ 278 - 0
include/ipc/gpr-lite.h

@@ -0,0 +1,278 @@
+/* Copyright (c) 2011-2017, 2019-2020 The Linux Foundation. All rights reserved.
+ * Copyright (c) 2018, Linaro Limited
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 and
+ * only version 2 as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ */
+
+#ifndef __QCOM_GPR_H_
+#define __QCOM_GPR_H_
+
+#include <linux/spinlock.h>
+#include <linux/device.h>
+#include <linux/mod_devicetable.h>
+#include <dt-bindings/sound/qcom,gpr.h>
+
+extern struct bus_type gprbus;
+
+enum gpr_subsys_state {
+        GPR_SUBSYS_DOWN,
+        GPR_SUBSYS_UP,
+        GPR_SUBSYS_LOADED,
+};
+
+
+/* Version */
+#define GPR_PKT_VER		0x0
+
+/*
+ * Header format
+ * Bits 3-0  : version
+ * Bits 7-4  : header_size
+ * Bits 31-8 : packet_size
+ */
+
+#define GPR_PKT_INIT_PORT_V ( ( uint32_t ) 0 )
+
+#define GPR_PKT_INIT_RESERVED_V ( ( uint16_t ) 0 )
+
+#define GPR_PKT_INIT_DOMAIN_ID_V ( ( uint8_t ) 0 )
+
+#define GPR_IDS_DOMAIN_ID_MODEM_V    ( 1 ) /**< Modem domain. */
+#define GPR_IDS_DOMAIN_ID_ADSP_V     ( 2 ) /**< aDSP domain. */
+#define GPR_IDS_DOMAIN_ID_APPS_V     ( 3 ) /**< Application domain. */
+
+/* V2.0 header size in number of 32-bit words. */
+#define GPR_PKT_HEADER_WORD_SIZE_V ( ( sizeof( struct gpr_pkt ) + 3 ) >> 2 )
+
+/* V2.0 header size in number of bytes. */
+#define GPR_PKT_HEADER_BYTE_SIZE_V ( GPR_PKT_HEADER_WORD_SIZE_V << 2 )
+
+/* Undefined value for where a valid GUID is expected. */
+#define GPR_UNDEFINED_ID_V ( ( uint32_t ) 0xFFFFFFFF )
+
+#define GPR_PKT_INIT_CLIENT_DATA_V ( ( uint8_t ) 0 )
+
+/***********************************************************/
+/**     GPRv2 Packet Bitfield Macros                     ***/
+/***********************************************************/
+
+/* Bitmask and bit shift of the version field. */
+#define GPR_PKT_VERSION_MASK ( 0x0000000F )
+#define GPR_PKT_VERSION_SHFT ( 0 )
+
+/* Bitmask and bit shift of the header size field. */
+#define GPR_PKT_HEADER_SIZE_MASK ( 0x000000F0 )
+#define GPR_PKT_HEADER_SIZE_SHFT ( 4 )
+
+/* Bitmask and bit shift of the reserved field.
+ * Includes 4 reserved bits from client data field */
+#define GPR_PKT_RESERVED_MASK ( 0xFFF00000 )
+#define GPR_PKT_RESERVED_SHFT ( 20 )
+
+/* Bitmask and bit shift of the packet size field. */
+#define GPR_PKT_PACKET_SIZE_MASK ( 0xFFFFFF00 )
+#define GPR_PKT_PACKET_SIZE_SHFT ( 8 )
+
+/**********************************************************************/
+/**       Packet Helper Macros                                      ***/
+/**********************************************************************/
+
+/* Gets the value of a field, including the specified mask and shift. */
+#define GPR_GET_BITMASK( mask, shift, value ) \
+  ( ( ( value ) & ( mask ) ) >> ( shift ) )
+
+/* Sets a value in a field with a specified mask and shift. */
+#define GPR_SET_BITMASK( mask, shift, value ) \
+  ( ( ( value ) << ( shift ) ) & ( mask ) )
+
+/* Gets the value of a field. */
+#define GPR_GET_FIELD( field, value ) \
+  GPR_GET_BITMASK( ( field##_MASK ), ( field##_SHFT ), ( value ) )
+
+/* Sets a value in a field. */
+#define GPR_SET_FIELD( field, value ) \
+  GPR_SET_BITMASK( ( field##_MASK ), ( field##_SHFT ), ( value ) )
+
+/* Returns an 8-bit aligned pointer to a base address pointer plus an
+   offset, in bytes.*/
+#define GPR_PTR_END_OF( base_ptr, offset ) \
+  ( ( ( uint8_t* ) base_ptr ) + ( offset ) )
+
+/** Given the packet header, returns the packet's current size in bytes.
+    The current packet byte size is the sum of the base packet structure and
+    the used portion of the payload.
+*/
+#define GPR_PKT_GET_PACKET_BYTE_SIZE( header ) \
+  ( GPR_GET_FIELD( GPR_PKT_PACKET_SIZE, header ) )
+
+/* Given the packet header, returns the header's current size in bytes. */
+#define GPR_PKT_GET_HEADER_BYTE_SIZE( header ) \
+  ( GPR_GET_FIELD( GPR_PKT_HEADER_SIZE, header ) << 2 )
+
+/** Given the packet header, returns the payload's current size in bytes.
+    The current payload byte size is the difference between the packet size and
+    the header size.
+*/
+#define GPR_PKT_GET_PAYLOAD_BYTE_SIZE( header ) \
+  ( GPR_PKT_GET_PACKET_BYTE_SIZE( header ) - \
+    GPR_PKT_GET_HEADER_BYTE_SIZE( header ) )
+
+/** Given the packet, returns a pointer to the beginning of the packet's
+    payload. */
+#define GPR_PKT_GET_PAYLOAD( type, packet_ptr ) \
+  ( ( type* ) GPR_PTR_END_OF( packet_ptr, \
+                GPR_PKT_GET_HEADER_BYTE_SIZE( \
+                  ( ( struct gpr_hdr* ) packet_ptr )->header ) ) )
+
+#define GPR_HDR_FIELD(msg_type, hdr_len, ver)\
+	(((msg_type & 0x3) << 8) | ((hdr_len & 0xF) << 4) | (ver & 0xF))
+
+#define GPR_HDR_SIZE sizeof(struct gpr_hdr)
+#define GPR_SEQ_CMD_HDR_FIELD GPR_HDR_FIELD(GPR_MSG_TYPE_SEQ_CMD, \
+					    GPR_HDR_LEN(GPR_HDR_SIZE), \
+					    GPR_PKT_VER)
+
+/**********************************************************************/
+/**       GECKO GUID definitions                                    ***/
+/**********************************************************************/
+/* An empty macro to mark non-GUIDs so that GUID script doesn't mistake
+   the ID for a GUID */
+#define MM_NON_GUID(x)                    x
+
+/* GUID owner is QTI. */
+#define MM_GUID_OWNER_QTI                  0x0
+
+/* GUID owner is not-QTI. */
+#define MM_GUID_OWNER_NON_QTI              0x1
+
+/* Mask and Shift for Owner */
+#define MM_GUID_OWNER_MASK                MM_NON_GUID(0xF0000000)
+#define MM_GUID_OWNER_SHIFT               28
+
+/************************************************/
+/*       GUID types                           ***/
+/************************************************/
+
+/* Reserved */
+#define MM_GUID_TYPE_RESERVED             0x0
+
+/* Control commands */
+#define MM_GUID_TYPE_CONTROL_CMD          0x1
+
+/* Responses of control commands */
+#define MM_GUID_TYPE_CONTROL_CMD_RSP      0x2
+
+/* Control events */
+#define MM_GUID_TYPE_CONTROL_EVENT        0x3
+
+/* Data commands */
+#define MM_GUID_TYPE_DATA_CMD             0x4
+
+/* Data command responses */
+#define MM_GUID_TYPE_DATA_CMD_RSP         0x5
+
+/* Data events */
+#define MM_GUID_TYPE_DATA_EVENT           0x6
+
+/* Module-id */
+#define MM_GUID_TYPE_MODULE_ID            0x7
+
+/* Module's param & event-id */
+#define MM_GUID_TYPE_PARAM_EVENT_ID       0x8
+
+/* Media format-id. */
+#define MM_GUID_TYPE_FORMAT_ID            0x9
+
+/* CAPI */
+#define MM_GUID_TYPE_CAPI                 0xA
+
+/* MAX */
+#define MM_GUID_TYPE_MAX                  0xB
+/** Rest are reserved */
+
+/* Zero is an invalid ID.
+ * To be used like NULL value for pointers */
+#define MM_GUID_INVALID                   0
+
+/* Mask and shift for GUID type */
+#define MM_GUID_TYPE_MASK                 MM_NON_GUID(0x0F000000)
+#define MM_GUID_TYPE_SHIFT                24
+
+struct gpr_hdr {
+        uint32_t header;
+        uint8_t dst_domain_id;
+        uint8_t src_domain_id;
+        uint8_t client_data;
+        uint8_t reserved;
+        uint32_t src_port;
+        uint32_t dst_port;
+        uint32_t token;
+        uint32_t opcode;
+} __packed;
+
+struct gpr_pkt {
+       struct gpr_hdr hdr;
+       uint8_t payload[0];
+};
+
+#define GPR_IBASIC_RSP_RESULT 0x02001005
+
+/* Bits 0 to 15 -- Minor version,  Bits 16 to 31 -- Major version */
+#define GPR_SVC_MAJOR_VERSION(v)	((v >> 16) & 0xFF)
+#define GPR_SVC_MINOR_VERSION(v)	(v & 0xFF)
+
+struct gpr_device {
+	struct device	dev;
+	uint16_t	svc_id;
+	uint16_t	domain_id;
+	uint32_t	version;
+	char name[GPR_NAME_SIZE];
+	spinlock_t	lock;
+	struct list_head node;
+};
+
+#define to_gpr_device(d) container_of(d, struct gpr_device, dev)
+
+struct gpr_driver {
+	int	(*probe)(struct gpr_device *sl);
+	int	(*remove)(struct gpr_device *sl);
+	int	(*callback)(struct gpr_device *a,
+			    void *d);
+	struct device_driver		driver;
+	const struct gpr_device_id	*id_table;
+};
+
+#define to_gpr_driver(d) container_of(d, struct gpr_driver, driver)
+
+/*
+ * use a macro to avoid include chaining to get THIS_MODULE
+ */
+#define gpr_driver_register(drv) __gpr_driver_register(drv, THIS_MODULE)
+
+int __gpr_driver_register(struct gpr_driver *drv, struct module *owner);
+void gpr_driver_unregister(struct gpr_driver *drv);
+
+/**
+ * module_gpr_driver() - Helper macro for registering a gprbus driver
+ * @__gprbus_driver: gprbus_driver struct
+ *
+ * Helper macro for gprbus drivers which do not do anything special in
+ * module init/exit. This eliminates a lot of boilerplate. Each module
+ * may only use this macro once, and calling it replaces module_init()
+ * and module_exit()
+ */
+#define module_gpr_driver(__gpr_driver) \
+	module_driver(__gpr_driver, gpr_driver_register, \
+			gpr_driver_unregister)
+
+int gpr_send_pkt(struct gpr_device *adev, struct gpr_pkt *pkt);
+
+#endif /* __QCOM_GPR_H_ */

+ 10 - 25
ipc/Kbuild

@@ -105,30 +105,17 @@ COMMON_INC :=	-I$(AUDIO_ROOT)/$(COMMON_DIR)
 
 ############ IPC ############
 
-ifdef CONFIG_MSM_QDSP6_APRV2_RPMSG
-APRV_GLINK += apr.o
-APRV_GLINK += apr_v2.o
-APRV_GLINK += apr_tal_rpmsg.o
+ifdef CONFIG_MSM_QDSP6_GPR_RPMSG
+GPRV_GLINK += gpr-lite.o
 endif
 
-ifdef CONFIG_MSM_QDSP6_APRV3_RPMSG
-APRV_GLINK += apr.o
-APRV_GLINK += apr_v3.o
-APRV_GLINK += apr_tal_rpmsg.o
+ifdef CONFIG_AUDIO_PKT
+AUDIO_PKT += audio-pkt.o
 endif
 
-ifdef CONFIG_MSM_QDSP6_APRV2_VM
-APRV_GLINK += apr_vm.o
-APRV_GLINK += apr_v2.o
-endif
-
-ifdef CONFIG_WCD_DSP_GLINK
-WDSP_GLINK += wcd-dsp-glink.o
-endif
-
-LINUX_INC +=	-Iinclude/linux
+LINUX_INC += -Iinclude/linux
 
-INCS +=		$(COMMON_INC) \
+INCS +=	$(COMMON_INC) \
 		$(UAPI_INC)
 
 EXTRA_CFLAGS += $(INCS)
@@ -178,13 +165,11 @@ GCOV_PROFILE := y
 endif
 
 # Module information used by KBuild framework
-obj-$(CONFIG_MSM_QDSP6_APRV2_RPMSG) += apr_dlkm.o
-obj-$(CONFIG_MSM_QDSP6_APRV3_RPMSG) += apr_dlkm.o
-obj-$(CONFIG_MSM_QDSP6_APRV2_VM) += apr_dlkm.o
-apr_dlkm-y := $(APRV_GLINK)
+obj-$(CONFIG_MSM_QDSP6_GPR_RPMSG) += gpr_dlkm.o
+gpr_dlkm-y := $(GPRV_GLINK)
 
-obj-$(CONFIG_WCD_DSP_GLINK) += wglink_dlkm.o
-wglink_dlkm-y := $(WDSP_GLINK)
+obj-$(CONFIG_AUDIO_PKT) += audio_pkt_dlkm.o
+audio_pkt_dlkm-y := $(AUDIO_PKT)
 
 # inject some build related information
 CDEFINES += -DBUILD_TIMESTAMP=\"$(shell date -u +'%Y-%m-%dT%H:%M:%SZ')\"

+ 523 - 0
ipc/audio-pkt.c

@@ -0,0 +1,523 @@
+/* Copyright (c) 2019-2020, The Linux Foundation. All rights reserved.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 and
+ * only version 2 as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ */
+
+#include <linux/platform_device.h>
+#include <linux/of_platform.h>
+#include <linux/ipc_logging.h>
+#include <linux/refcount.h>
+#include <linux/device.h>
+#include <linux/module.h>
+#include <linux/skbuff.h>
+#include <linux/cdev.h>
+#include <linux/slab.h>
+#include <linux/poll.h>
+#include <linux/idr.h>
+#include <linux/of.h>
+#include <linux/fs.h>
+#include <linux/uaccess.h>
+#include <linux/termios.h>
+#include <ipc/gpr-lite.h>
+#include <dsp/msm_audio_ion.h>
+
+/* Define IPC Logging Macros */
+#define AUDIO_PKT_IPC_LOG_PAGE_CNT 2
+static void *audio_pkt_ilctxt;
+
+static int audio_pkt_debug_mask;
+module_param_named(debug_mask, audio_pkt_debug_mask, int, 0664);
+
+#define APM_CMD_SHARED_MEM_MAP_REGIONS		0x0100100C
+#define APM_MEMORY_MAP_BIT_MASK_IS_OFFSET_MODE	0x00000004UL
+enum {
+	AUDIO_PKT_INFO = 1U << 0,
+};
+
+#define AUDIO_PKT_INFO(x, ...)						\
+do {									\
+	if (audio_pkt_debug_mask & AUDIO_PKT_INFO) {			\
+		ipc_log_string(audio_pkt_ilctxt,			\
+			"[%s]: "x, __func__, ##__VA_ARGS__);		\
+	}								\
+} while (0)
+
+#define AUDIO_PKT_ERR(x, ...)						      \
+do {									      \
+	pr_err_ratelimited("[%s]: "x, __func__, ##__VA_ARGS__);		      \
+	ipc_log_string(audio_pkt_ilctxt, "[%s]: "x, __func__, ##__VA_ARGS__); \
+} while (0)
+
+
+#define MODULE_NAME "audio-pkt"
+#define MINOR_NUMBER_COUNT 1
+#define AUDPKT_DRIVER_NAME "aud_pasthru_adsp"
+#define CHANNEL_NAME "to_apps"
+
+/**
+ * struct audio_pkt - driver context, relates rpdev to cdev
+ * @adev:	gpr device node
+ * @dev:	audio pkt device
+ * @cdev:	cdev for the audio pkt device
+ * @lock:	synchronization of @rpdev
+ * @queue_lock:	synchronization of @queue operations
+ * @queue:	incoming message queue
+ * @readq:	wait object for incoming queue
+ * @dev_name:	/dev/@dev_name for audio_pkt device
+ * @ch_name:	audio channel to match to
+ * @audio_pkt_major: Major number of audio pkt driver
+ * @audio_pkt_class: audio pkt class pointer
+ */
+struct audio_pkt_device {
+	struct gpr_device *adev;
+	struct device *dev;
+	struct cdev cdev;
+
+	struct mutex lock;
+
+	spinlock_t queue_lock;
+	struct sk_buff_head queue;
+	wait_queue_head_t readq;
+
+	char dev_name[20];
+	char ch_name[20];
+
+	dev_t audio_pkt_major;
+	struct class *audio_pkt_class;
+};
+
+struct audio_pkt_apm_cmd_shared_mem_map_regions_t {
+	uint16_t mem_pool_id;
+	uint16_t num_regions;
+	uint32_t property_flag;
+
+};
+
+struct audio_pkt_apm_shared_map_region_payload_t {
+	uint32_t shm_addr_lsw;
+	uint32_t shm_addr_msw;
+	uint32_t mem_size_bytes;
+};
+
+struct audio_pkt_apm_mem_map {
+	struct audio_pkt_apm_cmd_shared_mem_map_regions_t mmap_header;
+	struct audio_pkt_apm_shared_map_region_payload_t mmap_payload;
+};
+
+struct audio_gpr_pkt {
+	struct gpr_hdr audpkt_hdr;
+	struct audio_pkt_apm_mem_map audpkt_mem_map;
+};
+
+typedef void (*audio_pkt_clnt_cb_fn)(void *buf, int len, void *priv);
+
+struct audio_pkt_clnt_ch {
+	int client_id;
+	audio_pkt_clnt_cb_fn func;
+};
+
+#define dev_to_audpkt_dev(_dev) container_of(_dev, struct audio_pkt_device, dev)
+#define cdev_to_audpkt_dev(_cdev) container_of(_cdev, struct audio_pkt_device, cdev)
+
+/**
+ * audio_pkt_open() - open() syscall for the audio_pkt device
+ * inode:	Pointer to the inode structure.
+ * file:	Pointer to the file structure.
+ *
+ * This function is used to open the audio pkt device when
+ * userspace client do a open() system call. All input arguments are
+ * validated by the virtual file system before calling this function.
+ */
+int audio_pkt_open(struct inode *inode, struct file *file)
+{
+	struct audio_pkt_device *audpkt_dev = cdev_to_audpkt_dev(inode->i_cdev);
+	struct device *dev = audpkt_dev->dev;
+
+	AUDIO_PKT_ERR("for %s\n", audpkt_dev->ch_name);
+
+	get_device(dev);
+	file->private_data = audpkt_dev;
+
+	return 0;
+}
+
+/**
+ * audio_pkt_release() - release operation on audio_pkt device
+ * inode:	Pointer to the inode structure.
+ * file:	Pointer to the file structure.
+ *
+ * This function is used to release the audio pkt device when
+ * userspace client do a close() system call. All input arguments are
+ * validated by the virtual file system before calling this function.
+ */
+int audio_pkt_release(struct inode *inode, struct file *file)
+{
+	struct audio_pkt_device *audpkt_dev = cdev_to_audpkt_dev(inode->i_cdev);
+	struct device *dev = audpkt_dev->dev;
+	struct sk_buff *skb;
+	unsigned long flags;
+
+	AUDIO_PKT_INFO("for %s \n", audpkt_dev->ch_name);
+	spin_lock_irqsave(&audpkt_dev->queue_lock, flags);
+
+	/* Discard all SKBs */
+	while (!skb_queue_empty(&audpkt_dev->queue)) {
+		skb = skb_dequeue(&audpkt_dev->queue);
+		kfree_skb(skb);
+	}
+	wake_up_interruptible(&audpkt_dev->readq);
+	spin_unlock_irqrestore(&audpkt_dev->queue_lock, flags);
+
+	put_device(dev);
+	file->private_data = NULL;
+
+	return 0;
+}
+
+/**
+ * audio_pkt_read() - read() syscall for the audio_pkt device
+ * file:	Pointer to the file structure.
+ * buf:		Pointer to the userspace buffer.
+ * count:	Number bytes to read from the file.
+ * ppos:	Pointer to the position into the file.
+ *
+ * This function is used to Read the data from audio pkt device when
+ * userspace client do a read() system call. All input arguments are
+ * validated by the virtual file system before calling this function.
+ */
+ssize_t audio_pkt_read(struct file *file, char __user *buf,
+		       size_t count, loff_t *ppos)
+{
+	struct audio_pkt_device *audpkt_dev = file->private_data;
+	unsigned long flags;
+	struct sk_buff *skb;
+	int use;
+	uint32_t *temp;
+
+	if (!audpkt_dev) {
+		AUDIO_PKT_ERR("invalid device handle\n");
+		return -EINVAL;
+	}
+
+	spin_lock_irqsave(&audpkt_dev->queue_lock, flags);
+	/* Wait for data in the queue */
+	if (skb_queue_empty(&audpkt_dev->queue)) {
+		spin_unlock_irqrestore(&audpkt_dev->queue_lock, flags);
+
+		if (file->f_flags & O_NONBLOCK)
+			return -EAGAIN;
+
+		/* Wait until we get data or the endpoint goes away */
+		if (wait_event_interruptible(audpkt_dev->readq,
+					!skb_queue_empty(&audpkt_dev->queue)))
+			return -ERESTARTSYS;
+
+		spin_lock_irqsave(&audpkt_dev->queue_lock, flags);
+	}
+
+	skb = skb_dequeue(&audpkt_dev->queue);
+	spin_unlock_irqrestore(&audpkt_dev->queue_lock, flags);
+	if (!skb)
+		return -EFAULT;
+
+	use = min_t(size_t, count, skb->len);
+	if (copy_to_user(buf, skb->data, use))
+		use = -EFAULT;
+	temp = (uint32_t *) skb->data;
+	kfree_skb(skb);
+
+	return use;
+}
+
+/**
+ * audpkt_update_physical_addr - Update physical address
+ * audpkt_hdr:	Pointer to the file structure.
+ */
+int audpkt_chk_and_update_physical_addr(struct audio_gpr_pkt *gpr_pkt)
+{
+	int ret = 0;
+
+	dma_addr_t paddr;
+	if (gpr_pkt->audpkt_mem_map.mmap_header.property_flag &
+				APM_MEMORY_MAP_BIT_MASK_IS_OFFSET_MODE) {
+		ret = msm_audio_get_phy_addr(
+			(int) gpr_pkt->audpkt_mem_map.mmap_payload.shm_addr_lsw,
+			&paddr);
+		if (ret < 0) {
+			AUDIO_PKT_ERR("%s Get phy. address failed, ret %d\n",
+					__func__, ret);
+			return ret;
+		}
+		AUDIO_PKT_INFO("%s physical address %pK", __func__,
+				(void *) paddr);
+		gpr_pkt->audpkt_mem_map.mmap_payload.shm_addr_lsw = (uint32_t) paddr;
+		gpr_pkt->audpkt_mem_map.mmap_payload.shm_addr_msw = (uint64_t) paddr >> 32;
+	}
+	return ret;
+}
+
+/**
+ * audio_pkt_write() - write() syscall for the audio_pkt device
+ * file:	Pointer to the file structure.
+ * buf:		Pointer to the userspace buffer.
+ * count:	Number bytes to read from the file.
+ * ppos:	Pointer to the position into the file.
+ *
+ * This function is used to write the data to audio pkt device when
+ * userspace client do a write() system call. All input arguments are
+ * validated by the virtual file system before calling this function.
+ */
+ssize_t audio_pkt_write(struct file *file, const char __user *buf,
+			size_t count, loff_t *ppos)
+{
+	struct audio_pkt_device *audpkt_dev = file->private_data;
+	struct gpr_hdr *audpkt_hdr = NULL;
+	void *kbuf;
+	int ret;
+
+	if (!audpkt_dev)  {
+		AUDIO_PKT_ERR("invalid device handle\n");
+		return -EINVAL;
+	}
+
+	kbuf = memdup_user(buf, count);
+	if (IS_ERR(kbuf))
+		return PTR_ERR(kbuf);
+
+	audpkt_hdr = (struct gpr_hdr *) kbuf;
+	if (audpkt_hdr->opcode == APM_CMD_SHARED_MEM_MAP_REGIONS) {
+		ret = audpkt_chk_and_update_physical_addr((struct audio_gpr_pkt *) audpkt_hdr);
+		if (ret < 0) {
+			AUDIO_PKT_ERR("Update Physical Address Failed -%d\n", ret);
+		        return ret;
+		}
+	}
+
+	if (mutex_lock_interruptible(&audpkt_dev->lock)) {
+		ret = -ERESTARTSYS;
+		goto free_kbuf;
+	}
+	ret = gpr_send_pkt(audpkt_dev->adev,(struct gpr_pkt *) kbuf);
+	if (ret < 0) {
+		AUDIO_PKT_ERR("APR Send Packet Failed ret -%d\n", ret);
+		return ret;
+	}
+	mutex_unlock(&audpkt_dev->lock);
+
+free_kbuf:
+	kfree(kbuf);
+	return ret < 0 ? ret : count;
+}
+
+/**
+ * audio_pkt_poll() - poll() syscall for the audio_pkt device
+ * file:	Pointer to the file structure.
+ * wait:	pointer to Poll table.
+ *
+ * This function is used to poll on the audio pkt device when
+ * userspace client do a poll() system call. All input arguments are
+ * validated by the virtual file system before calling this function.
+ */
+static unsigned int audio_pkt_poll(struct file *file, poll_table *wait)
+{
+	struct audio_pkt_device *audpkt_dev = file->private_data;
+	unsigned int mask = 0;
+	unsigned long flags;
+
+	audpkt_dev = file->private_data;
+	if (!audpkt_dev) {
+		AUDIO_PKT_ERR("invalid device handle\n");
+		return POLLERR;
+	}
+
+	poll_wait(file, &audpkt_dev->readq, wait);
+
+	mutex_lock(&audpkt_dev->lock);
+
+	spin_lock_irqsave(&audpkt_dev->queue_lock, flags);
+	if (!skb_queue_empty(&audpkt_dev->queue))
+		mask |= POLLIN | POLLRDNORM;
+
+	spin_unlock_irqrestore(&audpkt_dev->queue_lock, flags);
+
+	mutex_unlock(&audpkt_dev->lock);
+
+	return mask;
+}
+
+static const struct file_operations audio_pkt_fops = {
+	.owner = THIS_MODULE,
+	.open = audio_pkt_open,
+	.release = audio_pkt_release,
+	.read = audio_pkt_read,
+	.write = audio_pkt_write,
+	.poll = audio_pkt_poll,
+};
+
+/**
+ * audio_pkt_srvc_callback() - Callback from gpr driver
+ * adev:	pointer to the gpr device of this audio packet device
+ * data:	APR response data packet
+ *
+ * return:	0 for success, Standard Linux errors
+ */
+static int audio_pkt_srvc_callback(struct gpr_device *adev,
+				void *data)
+{
+	struct audio_pkt_device *audpkt_dev = dev_get_drvdata(&adev->dev);
+	unsigned long flags;
+	struct sk_buff *skb;
+	struct gpr_hdr *hdr = (struct gpr_hdr *)data;
+	uint16_t hdr_size, pkt_size;
+	hdr_size = GPR_PKT_GET_HEADER_BYTE_SIZE(hdr->header);
+	pkt_size = GPR_PKT_GET_PACKET_BYTE_SIZE(hdr->header);
+
+        AUDIO_PKT_INFO("header %d packet %d \n",hdr_size, pkt_size);
+
+	skb = alloc_skb(pkt_size, GFP_ATOMIC);
+	if (!skb)
+		return -ENOMEM;
+
+	skb_put_data(skb, data, pkt_size);
+
+	spin_lock_irqsave(&audpkt_dev->queue_lock, flags);
+	skb_queue_tail(&audpkt_dev->queue, skb);
+	spin_unlock_irqrestore(&audpkt_dev->queue_lock, flags);
+
+	/* wake up any blocking processes, waiting for new data */
+	wake_up_interruptible(&audpkt_dev->readq);
+	return 0;
+}
+
+/**
+ * audio_pkt_probe() - Probe a AUDIO packet device
+ *
+ * adev:	Pointer to gpr device.
+ *
+ * return:	0 on success, standard Linux error codes on error.
+ *
+ * This function is called when the underlying device tree driver registers
+ * a gpr device, mapped to a Audio packet device.
+ */
+static int audio_pkt_probe(struct gpr_device *adev)
+{
+	struct audio_pkt_device *audpkt_dev;
+	struct device *dev = &adev->dev;
+	int ret;
+
+	audpkt_dev = devm_kzalloc(dev, sizeof(*audpkt_dev), GFP_KERNEL);
+	if (!audpkt_dev)
+		return -ENOMEM;
+
+	ret = alloc_chrdev_region(&audpkt_dev->audio_pkt_major, 0,
+				  MINOR_NUMBER_COUNT,AUDPKT_DRIVER_NAME);
+	if (ret < 0) {
+		AUDIO_PKT_ERR("alloc_chrdev_region failed ret:%d\n", ret);
+		goto err_chrdev;
+	}
+
+	audpkt_dev->audio_pkt_class = class_create(THIS_MODULE,
+						   AUDPKT_DRIVER_NAME);
+	if (IS_ERR(audpkt_dev->audio_pkt_class)) {
+		ret = PTR_ERR(audpkt_dev->audio_pkt_class);
+		AUDIO_PKT_ERR("class_create failed ret:%ld\n",
+			      PTR_ERR(audpkt_dev->audio_pkt_class));
+		goto err_class;
+	}
+
+	audpkt_dev->dev = device_create(audpkt_dev->audio_pkt_class, NULL,
+					audpkt_dev->audio_pkt_major, NULL,
+					AUDPKT_DRIVER_NAME);
+	if (IS_ERR(audpkt_dev->dev)) {
+		ret = PTR_ERR(audpkt_dev->dev);
+		AUDIO_PKT_ERR("device_create failed ret:%ld\n",
+			      PTR_ERR(audpkt_dev->dev));
+		goto err_device;
+	}
+	strlcpy(audpkt_dev->dev_name, CHANNEL_NAME, 20);
+	strlcpy(audpkt_dev->ch_name, CHANNEL_NAME, 20);
+	dev_set_name(audpkt_dev->dev, audpkt_dev->dev_name);
+
+	mutex_init(&audpkt_dev->lock);
+
+	spin_lock_init(&audpkt_dev->queue_lock);
+	skb_queue_head_init(&audpkt_dev->queue);
+	init_waitqueue_head(&audpkt_dev->readq);
+
+	audpkt_dev->adev = adev;
+	dev_set_drvdata(dev, audpkt_dev);
+
+	cdev_init(&audpkt_dev->cdev, &audio_pkt_fops);
+	audpkt_dev->cdev.owner = THIS_MODULE;
+
+	ret = cdev_add(&audpkt_dev->cdev, audpkt_dev->audio_pkt_major,
+		       MINOR_NUMBER_COUNT);
+	if (ret) {
+		AUDIO_PKT_ERR("cdev_add failed for %s ret:%d\n",
+			      audpkt_dev->dev_name, ret);
+		goto free_dev;
+	}
+
+	AUDIO_PKT_INFO("Audio Packet Port Driver Initialized\n");
+
+	return of_platform_populate(dev->of_node, NULL, NULL, dev);
+
+free_dev:
+	put_device(dev);
+	device_destroy(audpkt_dev->audio_pkt_class,audpkt_dev->audio_pkt_major);
+err_device:
+	class_destroy(audpkt_dev->audio_pkt_class);
+err_class:
+	unregister_chrdev_region(MAJOR(audpkt_dev->audio_pkt_major),
+				 MINOR_NUMBER_COUNT);
+err_chrdev:
+	kfree(audpkt_dev);
+	return ret;
+
+}
+
+/**
+ * audio_pkt_remove() - Remove a AUDIO packet device
+ *
+ * adev:	Pointer to gpr device.
+ *
+ * return:	0 on success, standard Linux error codes on error.
+ *
+ * This function is called when the underlying device tree driver
+ * removeds a gpr device, mapped to a Audio packet device.
+ */
+static int audio_pkt_remove(struct gpr_device *adev)
+{
+	of_platform_depopulate(&adev->dev);
+	AUDIO_PKT_INFO("Audio Packet Port Driver Removed\n");
+
+	return 0;
+
+}
+static const struct of_device_id audio_pkt_match_table[] = {
+	{ .compatible = "qcom,audio-pkt" },
+	{},
+};
+MODULE_DEVICE_TABLE(of, audio_pkt_match_table);
+
+static struct gpr_driver audio_pkt_driver = {
+	.probe = audio_pkt_probe,
+	.remove = audio_pkt_remove,
+	.callback = audio_pkt_srvc_callback,
+	.driver = {
+		.name = MODULE_NAME,
+		.of_match_table = of_match_ptr(audio_pkt_match_table),
+	 },
+};
+
+module_gpr_driver(audio_pkt_driver);
+MODULE_DESCRIPTION("MSM Audio Packet Driver");
+MODULE_LICENSE("GPL v2");

+ 375 - 0
ipc/gpr-lite.c

@@ -0,0 +1,375 @@
+/* Copyright (c) 2011-2017, 2019-2020 The Linux Foundation. All rights reserved.
+ * Copyright (c) 2018, Linaro Limited
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 and
+ * only version 2 as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ */
+
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/device.h>
+#include <linux/spinlock.h>
+#include <linux/idr.h>
+#include <linux/slab.h>
+#include <linux/of_device.h>
+#include <ipc/gpr-lite.h>
+#include <linux/rpmsg.h>
+#include <linux/of.h>
+
+struct gpr {
+	struct rpmsg_endpoint *ch;
+	struct device *dev;
+	spinlock_t svcs_lock;
+	struct idr svcs_idr;
+	int dest_domain_id;
+};
+
+/**
+ * gpr_send_pkt() - Send a gpr message from gpr device
+ *
+ * @adev: Pointer to previously registered gpr device.
+ * @pkt: Pointer to gpr packet to send
+ *
+ * Return: Will be an negative on packet size on success.
+ */
+int gpr_send_pkt(struct gpr_device *adev, struct gpr_pkt *pkt)
+{
+	struct gpr *gpr = dev_get_drvdata(adev->dev.parent);
+	struct gpr_hdr *hdr;
+	unsigned long flags;
+	uint32_t pkt_size;
+	int ret;
+
+	spin_lock_irqsave(&adev->lock, flags);
+
+	hdr = &pkt->hdr;
+	pkt_size = GPR_PKT_GET_PACKET_BYTE_SIZE(hdr->header);
+
+	dev_dbg(gpr->dev, "SVC_ID %d %s packet size %d\n", adev->svc_id, __func__, pkt_size);
+	ret = rpmsg_trysend(gpr->ch, pkt, pkt_size);
+	spin_unlock_irqrestore(&adev->lock, flags);
+
+	return ret ? ret : pkt_size;
+}
+EXPORT_SYMBOL_GPL(gpr_send_pkt);
+
+static void gpr_dev_release(struct device *dev)
+{
+	struct gpr_device *adev = to_gpr_device(dev);
+
+	kfree(adev);
+}
+
+static int gpr_callback(struct rpmsg_device *rpdev, void *buf,
+				  int len, void *priv, u32 addr)
+{
+	struct gpr *gpr = dev_get_drvdata(&rpdev->dev);
+	uint16_t hdr_size, pkt_size, svc_id;
+	//uint16_t ver;
+	struct gpr_device *svc = NULL;
+	struct gpr_driver *adrv = NULL;
+	struct gpr_hdr *hdr;
+	unsigned long flags;
+	//uint32_t opcode_type;
+
+	if (len <= GPR_HDR_SIZE) {
+		dev_err(gpr->dev, "GPR: Improper gpr pkt received:%p %d\n",
+			buf, len);
+		return -EINVAL;
+	}
+
+	hdr = buf;
+
+	hdr_size = GPR_PKT_GET_HEADER_BYTE_SIZE(hdr->header);
+	if (hdr_size < GPR_HDR_SIZE) {
+		dev_err(gpr->dev, "GPR: Wrong hdr size:%d\n", hdr_size);
+		return -EINVAL;
+	}
+
+	pkt_size = GPR_PKT_GET_PACKET_BYTE_SIZE(hdr->header);
+	dev_dbg(gpr->dev,"Header %x", hdr->header);
+
+	if (pkt_size < GPR_HDR_SIZE || pkt_size != len) {
+		dev_err(gpr->dev, "GPR: Wrong packet size\n");
+		return -EINVAL;
+	}
+
+	dev_dbg(gpr->dev, "%s: dst_port %x hdr_size %d pkt_size %d\n",__func__ , hdr->dst_port, hdr_size, pkt_size);
+
+	svc_id = hdr->dst_port;
+	spin_lock_irqsave(&gpr->svcs_lock, flags);
+	svc = idr_find(&gpr->svcs_idr, svc_id);
+	if (svc && svc->dev.driver) {
+		adrv = to_gpr_driver(svc->dev.driver);
+	} else {
+		/*Does not match any SVC ID hence would be routed to audio passthrough*/
+		svc = idr_find(&gpr->svcs_idr, GPR_SVC_MAX);
+		if (svc && svc->dev.driver)
+			adrv = to_gpr_driver(svc->dev.driver);
+	}
+	spin_unlock_irqrestore(&gpr->svcs_lock, flags);
+
+	if (!adrv) {
+		dev_err(gpr->dev, "GPR: service is not registered\n");
+		return -EINVAL;
+	}
+
+	/*
+	 * NOTE: hdr_size is not same as GPR_HDR_SIZE as remote can include
+	 * optional headers in to gpr_hdr which should be ignored
+	 */
+
+	adrv->callback(svc, buf);
+
+	return 0;
+}
+
+static int gpr_device_match(struct device *dev, struct device_driver *drv)
+{
+	struct gpr_device *adev = to_gpr_device(dev);
+	struct gpr_driver *adrv = to_gpr_driver(drv);
+	const struct gpr_device_id *id = adrv->id_table;
+
+	/* Attempt an OF style match first */
+	if (of_driver_match_device(dev, drv))
+		return 1;
+
+	if (!id)
+		return 0;
+
+	while (id->domain_id != 0 || id->svc_id != 0) {
+		if (id->domain_id == adev->domain_id &&
+		    id->svc_id == adev->svc_id)
+			return 1;
+		id++;
+	}
+
+	return 0;
+}
+
+static int gpr_device_probe(struct device *dev)
+{
+	struct gpr_device *adev = to_gpr_device(dev);
+	struct gpr_driver *adrv = to_gpr_driver(dev->driver);
+
+	return adrv->probe(adev);
+}
+
+static int gpr_device_remove(struct device *dev)
+{
+	struct gpr_device *adev = to_gpr_device(dev);
+	struct gpr_driver *adrv;
+	struct gpr *gpr = dev_get_drvdata(adev->dev.parent);
+
+	if (dev->driver) {
+		adrv = to_gpr_driver(dev->driver);
+		if (adrv->remove)
+			adrv->remove(adev);
+		spin_lock(&gpr->svcs_lock);
+		idr_remove(&gpr->svcs_idr, adev->svc_id);
+		spin_unlock(&gpr->svcs_lock);
+	}
+
+	return 0;
+}
+
+static int gpr_uevent(struct device *dev, struct kobj_uevent_env *env)
+{
+	struct gpr_device *adev = to_gpr_device(dev);
+	int ret;
+
+	ret = of_device_uevent_modalias(dev, env);
+	if (ret != -ENODEV)
+		return ret;
+
+	return add_uevent_var(env, "MODALIAS=gpr:%s", adev->name);
+}
+
+struct bus_type gprbus = {
+	.name		= "gprbus",
+	.match		= gpr_device_match,
+	.probe		= gpr_device_probe,
+	.uevent		= gpr_uevent,
+	.remove		= gpr_device_remove,
+};
+EXPORT_SYMBOL_GPL(gprbus);
+
+static int gpr_add_device(struct device *dev, struct device_node *np,
+			  const struct gpr_device_id *id)
+{
+	struct gpr *gpr = dev_get_drvdata(dev);
+	struct gpr_device *adev = NULL;
+	int ret;
+
+	adev = kzalloc(sizeof(*adev), GFP_KERNEL);
+	if (!adev)
+		return -ENOMEM;
+
+	spin_lock_init(&adev->lock);
+
+	adev->svc_id = id->svc_id;
+	adev->domain_id = id->domain_id;
+	adev->version = id->svc_version;
+	if (np)
+		strscpy(adev->name, np->name, GPR_NAME_SIZE);
+	else
+		strscpy(adev->name, id->name, GPR_NAME_SIZE);
+
+	dev_set_name(&adev->dev, "gprsvc:%s:%x:%x", adev->name,
+		     id->domain_id, id->svc_id);
+
+	adev->dev.bus = &gprbus;
+	adev->dev.parent = dev;
+	adev->dev.of_node = np;
+	adev->dev.release = gpr_dev_release;
+	adev->dev.driver = NULL;
+
+	spin_lock(&gpr->svcs_lock);
+	idr_alloc(&gpr->svcs_idr, adev, id->svc_id,
+		  id->svc_id + 1, GFP_ATOMIC);
+	spin_unlock(&gpr->svcs_lock);
+
+	dev_info(dev, "Adding GPR dev: %s\n", dev_name(&adev->dev));
+
+	ret = device_register(&adev->dev);
+	if (ret) {
+		dev_err(dev, "device_register failed: %d\n", ret);
+		put_device(&adev->dev);
+	}
+
+	return ret;
+}
+
+static void of_register_gpr_devices(struct device *dev)
+{
+	struct gpr *gpr = dev_get_drvdata(dev);
+	struct device_node *node;
+
+	for_each_child_of_node(dev->of_node, node) {
+		struct gpr_device_id id = { {0} };
+
+		if (of_property_read_u32(node, "reg", &id.svc_id))
+			continue;
+
+		id.domain_id = gpr->dest_domain_id;
+
+		if (gpr_add_device(dev, node, &id))
+			dev_err(dev, "Failed to add gpr %d svc\n", id.svc_id);
+	}
+}
+
+static int gpr_probe(struct rpmsg_device *rpdev)
+{
+	struct device *dev = &rpdev->dev;
+	struct gpr *gpr;
+	int ret;
+
+	gpr = devm_kzalloc(dev, sizeof(*gpr), GFP_KERNEL);
+	if (!gpr)
+		return -ENOMEM;
+
+	ret = of_property_read_u32(dev->of_node, "reg", &gpr->dest_domain_id);
+	if (ret) {
+		dev_err(dev, "GPR Domain ID not specified in DT\n");
+		return ret;
+	}
+
+	dev_set_drvdata(dev, gpr);
+	gpr->ch = rpdev->ept;
+	gpr->dev = dev;
+	spin_lock_init(&gpr->svcs_lock);
+	idr_init(&gpr->svcs_idr);
+	of_register_gpr_devices(dev);
+
+	return 0;
+}
+
+static int gpr_remove_device(struct device *dev, void *null)
+{
+	struct gpr_device *adev = to_gpr_device(dev);
+
+	device_unregister(&adev->dev);
+
+	return 0;
+}
+
+static void gpr_remove(struct rpmsg_device *rpdev)
+{
+	device_for_each_child(&rpdev->dev, NULL, gpr_remove_device);
+}
+
+/*
+ * __gpr_driver_register() - Client driver registration with gprbus
+ *
+ * @drv:Client driver to be associated with client-device.
+ * @owner: owning module/driver
+ *
+ * This API will register the client driver with the gprbus
+ * It is called from the driver's module-init function.
+ */
+int __gpr_driver_register(struct gpr_driver *drv, struct module *owner)
+{
+	drv->driver.bus = &gprbus;
+	drv->driver.owner = owner;
+
+	return driver_register(&drv->driver);
+}
+EXPORT_SYMBOL_GPL(__gpr_driver_register);
+
+/*
+ * gpr_driver_unregister() - Undo effect of gpr_driver_register
+ *
+ * @drv: Client driver to be unregistered
+ */
+void gpr_driver_unregister(struct gpr_driver *drv)
+{
+	driver_unregister(&drv->driver);
+}
+EXPORT_SYMBOL_GPL(gpr_driver_unregister);
+
+static const struct of_device_id gpr_of_match[] = {
+	{ .compatible = "qcom,gpr"},
+	{}
+};
+MODULE_DEVICE_TABLE(of, gpr_of_match);
+
+static struct rpmsg_driver gpr_driver = {
+	.probe = gpr_probe,
+	.remove = gpr_remove,
+	.callback = gpr_callback,
+	.drv = {
+		.name = "qcom,gpr",
+		.of_match_table = gpr_of_match,
+	},
+};
+
+static int __init gpr_init(void)
+{
+	int ret;
+
+	ret = bus_register(&gprbus);
+	if (!ret)
+		ret = register_rpmsg_driver(&gpr_driver);
+	else
+		bus_unregister(&gprbus);
+
+	return ret;
+}
+
+static void __exit gpr_exit(void)
+{
+	bus_unregister(&gprbus);
+	unregister_rpmsg_driver(&gpr_driver);
+}
+
+subsys_initcall(gpr_init);
+module_exit(gpr_exit);
+
+MODULE_LICENSE("GPL v2");
+MODULE_DESCRIPTION("QTI GPR Bus");