|
@@ -11,6 +11,7 @@
|
|
|
#include <linux/in.h>
|
|
|
#include <linux/stddef.h>
|
|
|
#include <linux/ip.h>
|
|
|
+#include <linux/ipv6.h>
|
|
|
#include <linux/fs.h>
|
|
|
#include <linux/module.h>
|
|
|
#include <linux/msm_ipa.h>
|
|
@@ -34,6 +35,8 @@
|
|
|
#define NETDEV_NAME "rndis"
|
|
|
#define IPV4_HDR_NAME "rndis_eth_ipv4"
|
|
|
#define IPV6_HDR_NAME "rndis_eth_ipv6"
|
|
|
+#define RNDIS_HDR_NAME "rndis"
|
|
|
+#define ULSO_MAX_SIZE 64000
|
|
|
#define IPA_TO_USB_CLIENT IPA_CLIENT_USB_CONS
|
|
|
#define INACTIVITY_MSEC_DELAY 100
|
|
|
#define DEFAULT_OUTSTANDING_HIGH 64
|
|
@@ -110,6 +113,13 @@ static void *ipa_rndis_logbuf;
|
|
|
#define RNDIS_IPA_LOG_ENTRY() RNDIS_IPA_DEBUG("begin\n")
|
|
|
#define RNDIS_IPA_LOG_EXIT() RNDIS_IPA_DEBUG("end\n")
|
|
|
|
|
|
+#define IPV4_IS_TCP(iph) ((iph)->protocol == IPPROTO_TCP)
|
|
|
+#define IPV4_IS_UDP(iph) ((iph)->protocol == IPPROTO_UDP)
|
|
|
+#define IPV6_IS_TCP(iph) (((struct ipv6hdr *)iph)->nexthdr == IPPROTO_TCP)
|
|
|
+#define IPV6_IS_UDP(iph) (((struct ipv6hdr *)iph)->nexthdr == IPPROTO_UDP)
|
|
|
+#define IPV4_DELTA 40
|
|
|
+#define IPV6_DELTA 60
|
|
|
+
|
|
|
/**
|
|
|
* enum rndis_ipa_state - specify the current driver internal state
|
|
|
* which is guarded by a state machine.
|
|
@@ -195,6 +205,8 @@ enum rndis_ipa_operation {
|
|
|
* @pm_hdl: handle for IPA PM framework
|
|
|
* @is_vlan_mode: should driver work in vlan mode?
|
|
|
* @netif_rx_function: holds the correct network stack API, needed for NAPI
|
|
|
+ * @is_ulso_mode: indicator for ulso support
|
|
|
+ * @rndis_hdr_hdl: hdr handle of rndis header
|
|
|
*/
|
|
|
struct rndis_ipa_dev {
|
|
|
struct net_device *net;
|
|
@@ -225,6 +237,8 @@ struct rndis_ipa_dev {
|
|
|
u32 pm_hdl;
|
|
|
bool is_vlan_mode;
|
|
|
int (*netif_rx_function)(struct sk_buff *skb);
|
|
|
+ bool is_ulso_mode;
|
|
|
+ u32 rndis_hdr_hdl;
|
|
|
};
|
|
|
|
|
|
/**
|
|
@@ -243,6 +257,34 @@ struct rndis_pkt_hdr {
|
|
|
__le32 zeroes[7];
|
|
|
} __packed__;
|
|
|
|
|
|
+/**
|
|
|
+ * qmap_hdr -
|
|
|
+ * @next_hdr: 1 - there is a qmap extension header, 0 - opposite
|
|
|
+ * @cd: 0 - data, 1 - command
|
|
|
+ * @packet_len: length excluding qmap header
|
|
|
+ * @ext_next_hdr: always zero
|
|
|
+ * @hdr_type: type of extension header
|
|
|
+ * @additional_hdr_size: distance between end of qmap header to start of ip
|
|
|
+ * header
|
|
|
+ * @zero_checksum: 0 - compute checksum, 1 - zero checksum
|
|
|
+ * @ip_id_cfg: 0 - running ip id per segment, 1 - constant ip id
|
|
|
+ * @segment_size: maximum segment size for the segmentation operation
|
|
|
+ */
|
|
|
+struct qmap_hdr {
|
|
|
+ u16 pad: 6;
|
|
|
+ u16 next_hdr: 1;
|
|
|
+ u16 cd: 1;
|
|
|
+ u16 mux_id: 8;
|
|
|
+ u16 packet_len_with_pad: 16;
|
|
|
+ u16 ext_next_hdr: 1;
|
|
|
+ u16 hdr_type: 7;
|
|
|
+ u16 additional_hdr_size: 5;
|
|
|
+ u16 reserved: 1;
|
|
|
+ u16 zero_checksum: 1;
|
|
|
+ u16 ip_id_cfg: 1;
|
|
|
+ u16 segment_size: 16;
|
|
|
+} __packed;
|
|
|
+
|
|
|
static int rndis_ipa_open(struct net_device *net);
|
|
|
static void rndis_ipa_packet_receive_notify
|
|
|
(void *private, enum ipa_dp_evt_type evt, unsigned long data);
|
|
@@ -260,15 +302,17 @@ static int rndis_ipa_stop(struct net_device *net);
|
|
|
static void rndis_ipa_enable_data_path(struct rndis_ipa_dev *rndis_ipa_ctx);
|
|
|
static struct sk_buff *rndis_encapsulate_skb(struct sk_buff *skb,
|
|
|
struct rndis_ipa_dev *rndis_ipa_ctx);
|
|
|
+static struct sk_buff* qmap_encapsulate_skb(struct sk_buff *skb);
|
|
|
static void rndis_ipa_xmit_error(struct sk_buff *skb);
|
|
|
static void rndis_ipa_xmit_error_aftercare_wq(struct work_struct *work);
|
|
|
static void rndis_ipa_prepare_header_insertion
|
|
|
(int eth_type,
|
|
|
const char *hdr_name, struct ipa_hdr_add *add_hdr,
|
|
|
const void *dst_mac, const void *src_mac, bool is_vlan_mode);
|
|
|
-static int rndis_ipa_hdrs_cfg
|
|
|
- (struct rndis_ipa_dev *rndis_ipa_ctx,
|
|
|
+static int rndis_ipa_hdrs_cfg(struct rndis_ipa_dev *rndis_ipa_ctx,
|
|
|
const void *dst_mac, const void *src_mac);
|
|
|
+static int rndis_ipa_hdrs_hpc_cfg(struct rndis_ipa_dev *rndis_ipa_ctx);
|
|
|
+static int rndis_ipa_hdrs_hpc_destroy(struct rndis_ipa_dev *rndis_ipa_ctx);
|
|
|
static int rndis_ipa_hdrs_destroy(struct rndis_ipa_dev *rndis_ipa_ctx);
|
|
|
static struct net_device_stats *rndis_ipa_get_stats(struct net_device *net);
|
|
|
static int rndis_ipa_register_properties(char *netdev_name, bool is_vlan_mode);
|
|
@@ -330,7 +374,7 @@ static const struct file_operations rndis_ipa_aggr_ops = {
|
|
|
static struct ipa_ep_cfg ipa_to_usb_ep_cfg = {
|
|
|
.mode = {
|
|
|
.mode = IPA_BASIC,
|
|
|
- .dst = IPA_CLIENT_APPS_LAN_CONS,
|
|
|
+ .dst = IPA_CLIENT_APPS_LAN_CONS,
|
|
|
},
|
|
|
.hdr = {
|
|
|
.hdr_len = ETH_HLEN + sizeof(struct rndis_pkt_hdr),
|
|
@@ -338,14 +382,14 @@ static struct ipa_ep_cfg ipa_to_usb_ep_cfg = {
|
|
|
.hdr_ofst_metadata = 0,
|
|
|
.hdr_additional_const_len = ETH_HLEN,
|
|
|
.hdr_ofst_pkt_size_valid = true,
|
|
|
- .hdr_ofst_pkt_size = 3 * sizeof(u32),
|
|
|
+ .hdr_ofst_pkt_size = offsetof(struct rndis_pkt_hdr, data_len),
|
|
|
.hdr_a5_mux = false,
|
|
|
.hdr_remove_additional = false,
|
|
|
.hdr_metadata_reg_valid = false,
|
|
|
},
|
|
|
.hdr_ext = {
|
|
|
.hdr_pad_to_alignment = 0,
|
|
|
- .hdr_total_len_or_pad_offset = 1 * sizeof(u32),
|
|
|
+ .hdr_total_len_or_pad_offset = offsetof(struct rndis_pkt_hdr, msg_len),
|
|
|
.hdr_payload_len_inc_padding = false,
|
|
|
.hdr_total_len_or_pad = IPA_HDR_TOTAL_LEN,
|
|
|
.hdr_total_len_or_pad_valid = true,
|
|
@@ -484,6 +528,35 @@ static struct rndis_pkt_hdr rndis_template_hdr = {
|
|
|
.zeroes = {0},
|
|
|
};
|
|
|
|
|
|
+/**
|
|
|
+ * qmap_template_hdr - QMAP template structure for RNDIS_IPA SW insertion
|
|
|
+ * @pad: Set to 0
|
|
|
+ * @next_hdr: extension header exists - 1
|
|
|
+ * @cd: Data packet - 0
|
|
|
+ * @mux_id: Always 0
|
|
|
+ * @packet_len_with_pad: Set dynamically
|
|
|
+ * @ext_next_hdr: Always 0
|
|
|
+ * @hdr_type: Set to ULSO - 0x3
|
|
|
+ * @additional_hdr_size: - Set to VLAN tag size
|
|
|
+ * @zero_checksum: Always compute checksum - 0
|
|
|
+ * @ip_id_cfg: Always run up ip id per segment - 0
|
|
|
+ * @segment_size: Set dynamically
|
|
|
+ */
|
|
|
+static struct qmap_hdr qmap_template_hdr = {
|
|
|
+ .pad = 0,
|
|
|
+ .next_hdr = 1,
|
|
|
+ .cd = 0,
|
|
|
+ .mux_id = 0,
|
|
|
+ .packet_len_with_pad = 0,
|
|
|
+ .ext_next_hdr = 0,
|
|
|
+ .hdr_type = 0x3,
|
|
|
+ .additional_hdr_size = 0,
|
|
|
+ .reserved = 0,
|
|
|
+ .zero_checksum = 0,
|
|
|
+ .ip_id_cfg = 0,
|
|
|
+ .segment_size = 0,
|
|
|
+};
|
|
|
+
|
|
|
static void rndis_ipa_msg_free_cb(void *buff, u32 len, u32 type)
|
|
|
{
|
|
|
kfree(buff);
|
|
@@ -610,18 +683,36 @@ int rndis_ipa_init(struct ipa_usb_init_params *params)
|
|
|
RNDIS_IPA_ERROR_RL("couldn't acquire vlan mode, is ipa ready?\n");
|
|
|
goto fail_get_vlan_mode;
|
|
|
}
|
|
|
-
|
|
|
RNDIS_IPA_DEBUG("is_vlan_mode %d\n", rndis_ipa_ctx->is_vlan_mode);
|
|
|
|
|
|
- result = rndis_ipa_hdrs_cfg
|
|
|
- (rndis_ipa_ctx,
|
|
|
- params->host_ethaddr,
|
|
|
+ rndis_ipa_ctx->is_ulso_mode = ipa3_is_ulso_supported();
|
|
|
+ RNDIS_IPA_DEBUG("is_ulso_mode=%d\n", rndis_ipa_ctx->is_ulso_mode);
|
|
|
+
|
|
|
+ result = rndis_ipa_hdrs_cfg(rndis_ipa_ctx, params->host_ethaddr,
|
|
|
params->device_ethaddr);
|
|
|
if (result) {
|
|
|
RNDIS_IPA_ERROR("fail on ipa hdrs set\n");
|
|
|
goto fail_hdrs_cfg;
|
|
|
}
|
|
|
- RNDIS_IPA_DEBUG("IPA header-insertion configed for Ethernet+RNDIS\n");
|
|
|
+ RNDIS_IPA_DEBUG("IPA header-insertion configured for Ethernet\n");
|
|
|
+
|
|
|
+ if (rndis_ipa_ctx->is_ulso_mode) {
|
|
|
+ result = rndis_ipa_hdrs_hpc_cfg(rndis_ipa_ctx);
|
|
|
+ if (result) {
|
|
|
+ RNDIS_IPA_ERROR("fail on ipa hdrs hpc set\n");
|
|
|
+ goto fail_add_hdrs_hpc;
|
|
|
+ }
|
|
|
+ RNDIS_IPA_DEBUG("IPA header-insertion configured for RNDIS\n");
|
|
|
+
|
|
|
+ rndis_ipa_ctx->net->hw_features = NETIF_F_RXCSUM;
|
|
|
+ rndis_ipa_ctx->net->hw_features |=
|
|
|
+ NETIF_F_IP_CSUM | NETIF_F_IPV6_CSUM;
|
|
|
+ rndis_ipa_ctx->net->hw_features |= NETIF_F_SG;
|
|
|
+ rndis_ipa_ctx->net->hw_features |= NETIF_F_GRO_HW;
|
|
|
+ rndis_ipa_ctx->net->hw_features |= NETIF_F_GSO_UDP_L4;
|
|
|
+ rndis_ipa_ctx->net->hw_features |= NETIF_F_ALL_TSO;
|
|
|
+ rndis_ipa_ctx->net->gso_max_size = ULSO_MAX_SIZE;
|
|
|
+ }
|
|
|
|
|
|
result = rndis_ipa_register_properties(net->name,
|
|
|
rndis_ipa_ctx->is_vlan_mode);
|
|
@@ -651,6 +742,10 @@ int rndis_ipa_init(struct ipa_usb_init_params *params)
|
|
|
RNDIS_IPA_DEBUG("LAN RX NAPI enabled = False");
|
|
|
}
|
|
|
|
|
|
+ if (rndis_ipa_ctx->is_vlan_mode)
|
|
|
+ qmap_template_hdr.additional_hdr_size =
|
|
|
+ VLAN_ETH_HLEN - ETH_HLEN;
|
|
|
+
|
|
|
rndis_ipa = rndis_ipa_ctx;
|
|
|
params->ipa_rx_notify = rndis_ipa_packet_receive_notify;
|
|
|
params->ipa_tx_notify = rndis_ipa_tx_complete_notify;
|
|
@@ -667,6 +762,9 @@ int rndis_ipa_init(struct ipa_usb_init_params *params)
|
|
|
fail_register_netdev:
|
|
|
rndis_ipa_deregister_properties(net->name);
|
|
|
fail_register_tx:
|
|
|
+ if (rndis_ipa_ctx->is_ulso_mode)
|
|
|
+ rndis_ipa_hdrs_hpc_destroy(rndis_ipa_ctx);
|
|
|
+fail_add_hdrs_hpc:
|
|
|
rndis_ipa_hdrs_destroy(rndis_ipa_ctx);
|
|
|
fail_hdrs_cfg:
|
|
|
fail_get_vlan_mode:
|
|
@@ -938,6 +1036,7 @@ static netdev_tx_t rndis_ipa_start_xmit(struct sk_buff *skb,
|
|
|
int ret;
|
|
|
netdev_tx_t status = NETDEV_TX_BUSY;
|
|
|
struct rndis_ipa_dev *rndis_ipa_ctx = netdev_priv(net);
|
|
|
+ unsigned int skb_len = skb->len;
|
|
|
|
|
|
netif_trans_update(net);
|
|
|
|
|
@@ -984,7 +1083,41 @@ static netdev_tx_t rndis_ipa_start_xmit(struct sk_buff *skb,
|
|
|
goto out;
|
|
|
}
|
|
|
|
|
|
- skb = rndis_encapsulate_skb(skb, rndis_ipa_ctx);
|
|
|
+ if (rndis_ipa_ctx->is_ulso_mode &&
|
|
|
+ (net->features & (NETIF_F_ALL_TSO | NETIF_F_GSO_UDP_L4))){
|
|
|
+ struct iphdr *iph = NULL;
|
|
|
+ /*
|
|
|
+ * gso_size must be set here because tx feature must be on
|
|
|
+ * meanning that in case of a small packet its checksum will
|
|
|
+ * not be computed and we must compute it using the hardware
|
|
|
+ * and thus marking it as gso packet, and the way to do it is to
|
|
|
+ * set gso_size to non 0 value. It is only used internally by
|
|
|
+ * the ipa driver so, there is no significance which non-0 value
|
|
|
+ * is set.
|
|
|
+ */
|
|
|
+ if (ntohs(skb->protocol) == ETH_P_IP) {
|
|
|
+ iph = ip_hdr(skb);
|
|
|
+ if (IPV4_IS_TCP(iph) || IPV4_IS_UDP(iph)) {
|
|
|
+ skb = qmap_encapsulate_skb(skb);
|
|
|
+ skb_shinfo(skb)->gso_size =
|
|
|
+ net->mtu - IPV4_DELTA;
|
|
|
+ }
|
|
|
+ } else if (ntohs(skb->protocol) == ETH_P_IPV6) {
|
|
|
+ iph = ip_hdr(skb);
|
|
|
+ if (IPV6_IS_TCP(iph) || IPV6_IS_UDP(iph)) {
|
|
|
+ skb = qmap_encapsulate_skb(skb);
|
|
|
+ skb_shinfo(skb)->gso_size =
|
|
|
+ net->mtu - IPV6_DELTA;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ skb = rndis_encapsulate_skb(skb, rndis_ipa_ctx);
|
|
|
+ }
|
|
|
+ /* This indicates no encapsulation was done - ulso mode with bad skb*/
|
|
|
+ if (unlikely(skb_len == skb->len)) {
|
|
|
+ skb_shinfo(skb)->gso_size = 0;
|
|
|
+ skb = rndis_encapsulate_skb(skb, rndis_ipa_ctx);
|
|
|
+ }
|
|
|
trace_rndis_tx_dp(skb->protocol);
|
|
|
ret = ipa_tx_dp(IPA_TO_USB_CLIENT, skb, NULL);
|
|
|
if (ret) {
|
|
@@ -1404,6 +1537,14 @@ void rndis_ipa_cleanup(void *private)
|
|
|
}
|
|
|
RNDIS_IPA_DEBUG("deregister Tx/Rx properties was successful\n");
|
|
|
|
|
|
+ if (rndis_ipa_ctx->is_ulso_mode) {
|
|
|
+ ret = rndis_ipa_hdrs_hpc_destroy(rndis_ipa_ctx);
|
|
|
+ if (ret)
|
|
|
+ RNDIS_IPA_ERROR("rndis_ipa_hdrs_hpc_destroy failed\n");
|
|
|
+ else
|
|
|
+ RNDIS_IPA_DEBUG("rndis_ipa_hdrs_hpc_destroy success\n");
|
|
|
+ }
|
|
|
+
|
|
|
ret = rndis_ipa_hdrs_destroy(rndis_ipa_ctx);
|
|
|
if (ret)
|
|
|
RNDIS_IPA_ERROR(
|
|
@@ -1443,6 +1584,8 @@ static void rndis_ipa_enable_data_path(struct rndis_ipa_dev *rndis_ipa_ctx)
|
|
|
RNDIS_IPA_DEBUG("device_ready_notify() not supplied\n");
|
|
|
}
|
|
|
|
|
|
+ qmap_template_hdr.segment_size = htons(rndis_ipa_ctx->net->mtu -
|
|
|
+ sizeof(qmap_template_hdr));
|
|
|
netif_start_queue(rndis_ipa_ctx->net);
|
|
|
RNDIS_IPA_DEBUG("netif_start_queue() was called\n");
|
|
|
}
|
|
@@ -1570,6 +1713,95 @@ static void rndis_ipa_prepare_header_insertion(
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+/**
|
|
|
+ * rndis_ipa_hdrs_hpc_cfg() - configure hpc header insertion in IPA core
|
|
|
+ * @rndis_ipa_ctx: main driver context
|
|
|
+ *
|
|
|
+ * This function adds headers that are used by the hpc header insertion
|
|
|
+ * mechanism.
|
|
|
+ *
|
|
|
+ * Returns negative errno, or zero on success
|
|
|
+ */
|
|
|
+static int rndis_ipa_hdrs_hpc_cfg(struct rndis_ipa_dev *rndis_ipa_ctx)
|
|
|
+{
|
|
|
+ struct ipa_ioc_add_hdr *hdrs;
|
|
|
+ struct ipa_hdr_add *rndis_hdr;
|
|
|
+ struct ipa_pkt_init_ex_hdr_ofst_set lookup;
|
|
|
+ int result = 0;
|
|
|
+
|
|
|
+ RNDIS_IPA_LOG_ENTRY();
|
|
|
+
|
|
|
+ hdrs = kzalloc(sizeof(*hdrs) + sizeof(*rndis_hdr), GFP_KERNEL);
|
|
|
+ if (!hdrs) {
|
|
|
+ result = -ENOMEM;
|
|
|
+ goto fail_mem;
|
|
|
+ }
|
|
|
+ rndis_hdr = &hdrs->hdr[0];
|
|
|
+ strlcpy(rndis_hdr->name, RNDIS_HDR_NAME, sizeof(rndis_hdr->name));
|
|
|
+ memcpy(rndis_hdr->hdr, &rndis_template_hdr, sizeof(rndis_template_hdr));
|
|
|
+ rndis_hdr->hdr_len = sizeof(rndis_template_hdr);
|
|
|
+ rndis_hdr->hdr_hdl = -1;
|
|
|
+ rndis_hdr->is_partial = false;
|
|
|
+ rndis_hdr->status = -1;
|
|
|
+ hdrs->num_hdrs = 1;
|
|
|
+ hdrs->commit = 1;
|
|
|
+
|
|
|
+ result = ipa3_add_hdr_hpc(hdrs);
|
|
|
+ if (result) {
|
|
|
+ RNDIS_IPA_ERROR("Fail on Header-Insertion(%d)\n", result);
|
|
|
+ goto fail_add_hdr;
|
|
|
+ }
|
|
|
+ if (rndis_hdr->status) {
|
|
|
+ RNDIS_IPA_ERROR("Fail on Header-Insertion rndis(%d)\n",
|
|
|
+ rndis_hdr->status);
|
|
|
+ result = rndis_hdr->status;
|
|
|
+ goto fail_add_hdr;
|
|
|
+ }
|
|
|
+
|
|
|
+ rndis_ipa_ctx->rndis_hdr_hdl = rndis_hdr->hdr_hdl;
|
|
|
+ lookup.ep = IPA_TO_USB_CLIENT;
|
|
|
+ strlcpy(lookup.name, RNDIS_HDR_NAME, sizeof(lookup.name));
|
|
|
+ if (ipa_set_pkt_init_ex_hdr_ofst(&lookup, true))
|
|
|
+ goto fail_add_hdr;
|
|
|
+
|
|
|
+ RNDIS_IPA_LOG_EXIT();
|
|
|
+
|
|
|
+fail_add_hdr:
|
|
|
+ kfree(hdrs);
|
|
|
+fail_mem:
|
|
|
+ return result;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * rndis_ipa_hdrs_hpc_destroy() - remove the IPA headers hpc configuration done
|
|
|
+ * for the driver data path bridging.
|
|
|
+ * @rndis_ipa_ctx: the driver context
|
|
|
+ *
|
|
|
+ * Revert the work done on rndis_ipa_hdrs_hpc_cfg()
|
|
|
+ */
|
|
|
+static int rndis_ipa_hdrs_hpc_destroy(struct rndis_ipa_dev *rndis_ipa_ctx)
|
|
|
+{
|
|
|
+ struct ipa_ioc_del_hdr *del_hdr;
|
|
|
+ struct ipa_hdr_del *rndis_hdr;
|
|
|
+ int result;
|
|
|
+
|
|
|
+ del_hdr = kzalloc(sizeof(*del_hdr) + sizeof(*rndis_hdr), GFP_KERNEL);
|
|
|
+ if (!del_hdr)
|
|
|
+ return -ENOMEM;
|
|
|
+
|
|
|
+ del_hdr->commit = 1;
|
|
|
+ del_hdr->num_hdls = 1;
|
|
|
+ rndis_hdr = &del_hdr->hdl[0];
|
|
|
+ rndis_hdr->hdl = rndis_ipa_ctx->rndis_hdr_hdl;
|
|
|
+
|
|
|
+ result = ipa3_del_hdr(del_hdr);
|
|
|
+ if (result || rndis_hdr->status)
|
|
|
+ RNDIS_IPA_ERROR("ipa3_del_hdr failed\n");
|
|
|
+ kfree(del_hdr);
|
|
|
+
|
|
|
+ return result;
|
|
|
+}
|
|
|
+
|
|
|
/**
|
|
|
* rndis_ipa_hdrs_cfg() - configure header insertion block in IPA core
|
|
|
* to allow HW bridging
|
|
@@ -1596,9 +1828,8 @@ static int rndis_ipa_hdrs_cfg(
|
|
|
|
|
|
RNDIS_IPA_LOG_ENTRY();
|
|
|
|
|
|
- hdrs = kzalloc
|
|
|
- (sizeof(*hdrs) + sizeof(*ipv4_hdr) + sizeof(*ipv6_hdr),
|
|
|
- GFP_KERNEL);
|
|
|
+ hdrs = kzalloc(sizeof(*hdrs) + sizeof(*ipv4_hdr) + sizeof(*ipv6_hdr),
|
|
|
+ GFP_KERNEL);
|
|
|
if (!hdrs) {
|
|
|
result = -ENOMEM;
|
|
|
goto fail_mem;
|
|
@@ -1606,15 +1837,13 @@ static int rndis_ipa_hdrs_cfg(
|
|
|
|
|
|
ipv4_hdr = &hdrs->hdr[0];
|
|
|
ipv6_hdr = &hdrs->hdr[1];
|
|
|
- rndis_ipa_prepare_header_insertion
|
|
|
- (ETH_P_IP, IPV4_HDR_NAME,
|
|
|
+ rndis_ipa_prepare_header_insertion(ETH_P_IP, IPV4_HDR_NAME,
|
|
|
ipv4_hdr, dst_mac, src_mac, rndis_ipa_ctx->is_vlan_mode);
|
|
|
- rndis_ipa_prepare_header_insertion
|
|
|
- (ETH_P_IPV6, IPV6_HDR_NAME,
|
|
|
+ rndis_ipa_prepare_header_insertion(ETH_P_IPV6, IPV6_HDR_NAME,
|
|
|
ipv6_hdr, dst_mac, src_mac, rndis_ipa_ctx->is_vlan_mode);
|
|
|
|
|
|
- hdrs->commit = 1;
|
|
|
hdrs->num_hdrs = 2;
|
|
|
+ hdrs->commit = 1;
|
|
|
result = ipa3_add_hdr(hdrs);
|
|
|
if (result) {
|
|
|
RNDIS_IPA_ERROR("Fail on Header-Insertion(%d)\n", result);
|
|
@@ -1665,7 +1894,6 @@ static int rndis_ipa_hdrs_destroy(struct rndis_ipa_dev *rndis_ipa_ctx)
|
|
|
|
|
|
del_hdr->commit = 1;
|
|
|
del_hdr->num_hdls = 2;
|
|
|
-
|
|
|
ipv4 = &del_hdr->hdl[0];
|
|
|
ipv4->hdl = rndis_ipa_ctx->eth_ipv4_hdr_hdl;
|
|
|
ipv6 = &del_hdr->hdl[1];
|
|
@@ -1840,7 +2068,6 @@ static int rndis_ipa_deregister_pm_client(struct rndis_ipa_dev *rndis_ipa_ctx)
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
-
|
|
|
/**
|
|
|
* rndis_encapsulate_skb() - encapsulate the given Ethernet skb with
|
|
|
* an RNDIS header
|
|
@@ -1849,7 +2076,7 @@ static int rndis_ipa_deregister_pm_client(struct rndis_ipa_dev *rndis_ipa_ctx)
|
|
|
*
|
|
|
* Shall use a template header for RNDIS and update it with the given
|
|
|
* skb values.
|
|
|
- * Ethernet is expected to be already encapsulate the packet.
|
|
|
+ * Ethernet 2 header should already be encapsulated in the packet.
|
|
|
*/
|
|
|
static struct sk_buff *rndis_encapsulate_skb(struct sk_buff *skb,
|
|
|
struct rndis_ipa_dev *rndis_ipa_ctx)
|
|
@@ -1861,6 +2088,7 @@ static struct sk_buff *rndis_encapsulate_skb(struct sk_buff *skb,
|
|
|
if (unlikely(skb_headroom(skb) < sizeof(rndis_template_hdr))) {
|
|
|
struct sk_buff *new_skb = skb_copy_expand(skb,
|
|
|
sizeof(rndis_template_hdr), 0, GFP_ATOMIC);
|
|
|
+
|
|
|
if (!new_skb) {
|
|
|
RNDIS_IPA_ERROR_RL("no memory for skb expand\n");
|
|
|
return skb;
|
|
@@ -1888,6 +2116,43 @@ static struct sk_buff *rndis_encapsulate_skb(struct sk_buff *skb,
|
|
|
return skb;
|
|
|
}
|
|
|
|
|
|
+/**
|
|
|
+ * qmap_encapsulate_skb() - encapsulate the given Ethernet skb with
|
|
|
+ * a QMAP header
|
|
|
+ * @skb: packet to be encapsulated with the QMAP header
|
|
|
+ *
|
|
|
+ * Shall use a template header for QMAP and update it with the given
|
|
|
+ * skb values.
|
|
|
+ * Ethernet 2 header should already be encapsulated in the packet.
|
|
|
+ */
|
|
|
+static struct sk_buff* qmap_encapsulate_skb(struct sk_buff *skb)
|
|
|
+{
|
|
|
+ struct qmap_hdr *qh_ptr;
|
|
|
+ struct qmap_hdr qh = qmap_template_hdr;
|
|
|
+
|
|
|
+ /* if there is no room in this skb, allocate a new one */
|
|
|
+ if (unlikely(skb_headroom(skb) < sizeof(qmap_template_hdr))) {
|
|
|
+ struct sk_buff *new_skb = skb_copy_expand(skb,
|
|
|
+ sizeof(qmap_template_hdr), 0, GFP_ATOMIC);
|
|
|
+
|
|
|
+ if (!new_skb) {
|
|
|
+ RNDIS_IPA_ERROR("no memory for skb expand\n");
|
|
|
+ return skb;
|
|
|
+ }
|
|
|
+ RNDIS_IPA_DEBUG("skb expanded. old %pK new %pK\n",
|
|
|
+ skb, new_skb);
|
|
|
+ dev_kfree_skb_any(skb);
|
|
|
+ skb = new_skb;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* make room at the head of the SKB to put the QMAP header */
|
|
|
+ qh_ptr = (struct qmap_hdr *)skb_push(skb, sizeof(qh));
|
|
|
+ qh.packet_len_with_pad = htons(skb->len);
|
|
|
+ memcpy(qh_ptr, &qh, sizeof(*qh_ptr));
|
|
|
+
|
|
|
+ return skb;
|
|
|
+}
|
|
|
+
|
|
|
/**
|
|
|
* rx_filter() - logic that decide if the current skb is to be filtered out
|
|
|
* @skb: skb that may be sent up to the network stack
|
|
@@ -1976,20 +2241,21 @@ static int rndis_ipa_ep_registers_cfg(
|
|
|
}
|
|
|
|
|
|
if (is_vlan_mode) {
|
|
|
- usb_to_ipa_ep_cfg->hdr.hdr_len =
|
|
|
- VLAN_ETH_HLEN + add;
|
|
|
+ usb_to_ipa_ep_cfg->hdr.hdr_len = VLAN_ETH_HLEN + add;
|
|
|
ipa_to_usb_ep_cfg.hdr.hdr_len =
|
|
|
VLAN_ETH_HLEN + sizeof(struct rndis_pkt_hdr);
|
|
|
ipa_to_usb_ep_cfg.hdr.hdr_additional_const_len = VLAN_ETH_HLEN;
|
|
|
+ qmap_template_hdr.additional_hdr_size =
|
|
|
+ VLAN_ETH_HLEN - ETH_HLEN;
|
|
|
} else {
|
|
|
- usb_to_ipa_ep_cfg->hdr.hdr_len =
|
|
|
- ETH_HLEN + add;
|
|
|
+ usb_to_ipa_ep_cfg->hdr.hdr_len = ETH_HLEN + add;
|
|
|
ipa_to_usb_ep_cfg.hdr.hdr_len =
|
|
|
ETH_HLEN + sizeof(struct rndis_pkt_hdr);
|
|
|
ipa_to_usb_ep_cfg.hdr.hdr_additional_const_len = ETH_HLEN;
|
|
|
}
|
|
|
|
|
|
usb_to_ipa_ep_cfg->deaggr.max_packet_len = max_xfer_size_bytes_to_dev;
|
|
|
+
|
|
|
result = ipa3_cfg_ep(usb_to_ipa_hdl, usb_to_ipa_ep_cfg);
|
|
|
if (result) {
|
|
|
pr_err("failed to configure USB to IPA point\n");
|
|
@@ -2006,8 +2272,7 @@ static int rndis_ipa_ep_registers_cfg(
|
|
|
} else {
|
|
|
ipa_to_usb_ep_cfg.aggr.aggr_time_limit =
|
|
|
DEFAULT_AGGR_TIME_LIMIT;
|
|
|
- ipa_to_usb_ep_cfg.aggr.aggr_pkt_limit =
|
|
|
- DEFAULT_AGGR_PKT_LIMIT;
|
|
|
+ ipa_to_usb_ep_cfg.aggr.aggr_pkt_limit = DEFAULT_AGGR_PKT_LIMIT;
|
|
|
}
|
|
|
|
|
|
RNDIS_IPA_DEBUG(
|
|
@@ -2020,6 +2285,9 @@ static int rndis_ipa_ep_registers_cfg(
|
|
|
/* enable hdr_metadata_reg_valid */
|
|
|
usb_to_ipa_ep_cfg->hdr.hdr_metadata_reg_valid = true;
|
|
|
|
|
|
+ if (ipa3_is_ulso_supported())
|
|
|
+ ipa_to_usb_ep_cfg.ulso.is_ulso_pipe = true;
|
|
|
+
|
|
|
/*xlat config in vlan mode */
|
|
|
if (is_vlan_mode) {
|
|
|
usb_to_ipa_ep_cfg->hdr.hdr_ofst_metadata_valid = 1;
|