aboutsummaryrefslogtreecommitdiff
path: root/plugins/qcom-soc/mm-broadband-modem-qmi-qcom-soc.c
diff options
context:
space:
mode:
authorAleksander Morgado <aleksander@aleksander.es>2020-10-22 12:48:35 +0200
committerAleksander Morgado <aleksander@aleksander.es>2020-11-20 09:24:51 +0000
commit8fc60754dd1df91560a3123454df1ad0ecf0ac92 (patch)
tree4a53db0baee7b907eef264df6936dc302467ffc9 /plugins/qcom-soc/mm-broadband-modem-qmi-qcom-soc.c
parentcab4b54ad106caadb7f70025348f0aab5522bde4 (diff)
qcom-soc: new plugin for Qualcomm SoCs
This plugin implements support for old Qualcomm SoCs like the MSM8916 or the MSM8974, where: * control ports are available via RPMSG channels exported as devices e.g. with rpmsgexport: https://github.com/andersson/rpmsgexport * network ports are exposed by the bam-dmux kernel driver: https://github.com/msm8916-mainline/linux/commits/bam-dmux Adding support for newer Qualcomm SoCs (e.g. QRTR+IPA) could be done in a similar way on this very same plugin. This plugin is the first and only one that implements support for a modem device that is "built in" the system, as opposed to external modems that may be available via USB or PCI. The ID_MM_PHYSDEV_UID based udev tags provided by the plugin provide the logic to bind all the SoC ports together in the same modem object, and therefore ID_MM_PHYSDEV_UID should not be used by users to override the ones set by the plugin. All "rpmsg[0-9]*" ports that are considered part of the modem are flagged as candidate, ignoring the parent "rpmsg_ctrl[0-9]*" ports on purpose. This setup therefore assumes that the channels have been exported already as devices (e.g. using rpmsgexport). libqmi 1.27.2 is required to support the "WDS Bind Data Port" message.
Diffstat (limited to 'plugins/qcom-soc/mm-broadband-modem-qmi-qcom-soc.c')
-rw-r--r--plugins/qcom-soc/mm-broadband-modem-qmi-qcom-soc.c133
1 files changed, 133 insertions, 0 deletions
diff --git a/plugins/qcom-soc/mm-broadband-modem-qmi-qcom-soc.c b/plugins/qcom-soc/mm-broadband-modem-qmi-qcom-soc.c
new file mode 100644
index 00000000..58a6e5e3
--- /dev/null
+++ b/plugins/qcom-soc/mm-broadband-modem-qmi-qcom-soc.c
@@ -0,0 +1,133 @@
+/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
+/*
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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:
+ *
+ * Copyright (C) 2020 Aleksander Morgado <aleksander@aleksander.es>
+ */
+
+#include <config.h>
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <string.h>
+#include <unistd.h>
+#include <ctype.h>
+
+#include "ModemManager.h"
+#include "mm-log.h"
+#include "mm-broadband-modem-qmi-qcom-soc.h"
+
+G_DEFINE_TYPE (MMBroadbandModemQmiQcomSoc, mm_broadband_modem_qmi_qcom_soc, MM_TYPE_BROADBAND_MODEM_QMI)
+
+/*****************************************************************************/
+
+static const QmiSioPort sio_port_per_port_number[] = {
+ QMI_SIO_PORT_A2_MUX_RMNET0,
+ QMI_SIO_PORT_A2_MUX_RMNET1,
+ QMI_SIO_PORT_A2_MUX_RMNET2,
+ QMI_SIO_PORT_A2_MUX_RMNET3,
+ QMI_SIO_PORT_A2_MUX_RMNET4,
+ QMI_SIO_PORT_A2_MUX_RMNET5,
+ QMI_SIO_PORT_A2_MUX_RMNET6,
+ QMI_SIO_PORT_A2_MUX_RMNET7
+};
+
+static MMPortQmi *
+peek_port_qmi_for_data (MMBroadbandModemQmi *self,
+ MMPort *data,
+ QmiSioPort *out_sio_port,
+ GError **error)
+{
+ GList *rpmsg_qmi_ports;
+ MMPortQmi *found = NULL;
+ MMKernelDevice *net_port;
+ const gchar *net_port_driver;
+ gint net_port_number;
+
+ g_assert (MM_IS_BROADBAND_MODEM_QMI (self));
+ g_assert (mm_port_get_subsys (data) == MM_PORT_SUBSYS_NET);
+
+ net_port = mm_port_peek_kernel_device (data);
+ net_port_driver = mm_kernel_device_get_driver (net_port);
+ if (g_strcmp0 (net_port_driver, "bam-dmux") != 0) {
+ g_set_error (error,
+ MM_CORE_ERROR,
+ MM_CORE_ERROR_FAILED,
+ "Unsupported QMI kernel driver for 'net/%s': %s",
+ mm_port_get_device (data),
+ net_port_driver);
+ return NULL;
+ }
+
+ /* The dev_port notified by the bam-dmux driver indicates which SIO port we should be using */
+ net_port_number = mm_kernel_device_get_attribute_as_int (net_port, "dev_port");
+ if (net_port_number < 0 || net_port_number >= (gint) G_N_ELEMENTS (sio_port_per_port_number)) {
+ g_set_error (error,
+ MM_CORE_ERROR,
+ MM_CORE_ERROR_NOT_FOUND,
+ "Couldn't find SIO port number for 'net/%s'",
+ mm_port_get_device (data));
+ return NULL;
+ }
+
+ /* Find one QMI port, we don't care which one */
+ rpmsg_qmi_ports = mm_base_modem_find_ports (MM_BASE_MODEM (self),
+ MM_PORT_SUBSYS_RPMSG,
+ MM_PORT_TYPE_QMI,
+ NULL);
+ if (!rpmsg_qmi_ports) {
+ g_set_error (error,
+ MM_CORE_ERROR,
+ MM_CORE_ERROR_NOT_FOUND,
+ "Couldn't find any QMI port for 'net/%s'",
+ mm_port_get_device (data));
+ return NULL;
+ }
+
+ /* Set outputs */
+ *out_sio_port = sio_port_per_port_number[net_port_number];
+ found = MM_PORT_QMI (rpmsg_qmi_ports->data);
+
+ g_list_free_full (rpmsg_qmi_ports, g_object_unref);
+
+ return found;
+}
+
+/*****************************************************************************/
+
+MMBroadbandModemQmiQcomSoc *
+mm_broadband_modem_qmi_qcom_soc_new (const gchar *device,
+ const gchar **drivers,
+ const gchar *plugin,
+ guint16 vendor_id,
+ guint16 product_id)
+{
+ return g_object_new (MM_TYPE_BROADBAND_MODEM_QMI_QCOM_SOC,
+ MM_BASE_MODEM_DEVICE, device,
+ MM_BASE_MODEM_DRIVERS, drivers,
+ MM_BASE_MODEM_PLUGIN, plugin,
+ MM_BASE_MODEM_VENDOR_ID, vendor_id,
+ MM_BASE_MODEM_PRODUCT_ID, product_id,
+ NULL);
+}
+
+static void
+mm_broadband_modem_qmi_qcom_soc_init (MMBroadbandModemQmiQcomSoc *self)
+{
+}
+
+static void
+mm_broadband_modem_qmi_qcom_soc_class_init (MMBroadbandModemQmiQcomSocClass *klass)
+{
+ MMBroadbandModemQmiClass *broadband_modem_qmi_class = MM_BROADBAND_MODEM_QMI_CLASS (klass);
+
+ broadband_modem_qmi_class->peek_port_qmi_for_data = peek_port_qmi_for_data;
+}