The following macros, defined in "rmnet_map.h", assume a socket
buffer is provided as an argument without any real indication this
is the case.
    RMNET_MAP_GET_MUX_ID()
    RMNET_MAP_GET_CD_BIT()
    RMNET_MAP_GET_PAD()
    RMNET_MAP_GET_CMD_START()
    RMNET_MAP_GET_LENGTH()
What they hide is pretty trivial accessing of fields in a structure,
and it's much clearer to see this if we do these accesses directly.
So rather than using these accessor macros, assign a local
variable of the map header pointer type to the socket buffer data
pointer, and derereference that pointer variable.
In "rmnet_map_data.c", use sizeof(object) rather than sizeof(type)
in one spot.  Also, there's no need to byte swap 0; it's all zeros
irrespective of endianness.
Signed-off-by: Alex Elder <elder@linaro.org>
Reviewed-by: Bjorn Andersson <bjorn.andersson@linaro.org>
Reviewed-by: Alexander Duyck <alexanderduyck@fb.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
		
	
			
		
			
				
	
	
		
			103 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			103 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0-only
 | |
| /* Copyright (c) 2013-2018, The Linux Foundation. All rights reserved.
 | |
|  */
 | |
| 
 | |
| #include <linux/netdevice.h>
 | |
| #include "rmnet_config.h"
 | |
| #include "rmnet_map.h"
 | |
| #include "rmnet_private.h"
 | |
| #include "rmnet_vnd.h"
 | |
| 
 | |
| static u8 rmnet_map_do_flow_control(struct sk_buff *skb,
 | |
| 				    struct rmnet_port *port,
 | |
| 				    int enable)
 | |
| {
 | |
| 	struct rmnet_map_header *map_header = (void *)skb->data;
 | |
| 	struct rmnet_endpoint *ep;
 | |
| 	struct net_device *vnd;
 | |
| 	u8 mux_id;
 | |
| 	int r;
 | |
| 
 | |
| 	mux_id = map_header->mux_id;
 | |
| 
 | |
| 	if (mux_id >= RMNET_MAX_LOGICAL_EP) {
 | |
| 		kfree_skb(skb);
 | |
| 		return RX_HANDLER_CONSUMED;
 | |
| 	}
 | |
| 
 | |
| 	ep = rmnet_get_endpoint(port, mux_id);
 | |
| 	if (!ep) {
 | |
| 		kfree_skb(skb);
 | |
| 		return RX_HANDLER_CONSUMED;
 | |
| 	}
 | |
| 
 | |
| 	vnd = ep->egress_dev;
 | |
| 
 | |
| 	/* Ignore the ip family and pass the sequence number for both v4 and v6
 | |
| 	 * sequence. User space does not support creating dedicated flows for
 | |
| 	 * the 2 protocols
 | |
| 	 */
 | |
| 	r = rmnet_vnd_do_flow_control(vnd, enable);
 | |
| 	if (r) {
 | |
| 		kfree_skb(skb);
 | |
| 		return RMNET_MAP_COMMAND_UNSUPPORTED;
 | |
| 	} else {
 | |
| 		return RMNET_MAP_COMMAND_ACK;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void rmnet_map_send_ack(struct sk_buff *skb,
 | |
| 			       unsigned char type,
 | |
| 			       struct rmnet_port *port)
 | |
| {
 | |
| 	struct rmnet_map_header *map_header = (void *)skb->data;
 | |
| 	struct rmnet_map_control_command *cmd;
 | |
| 	struct net_device *dev = skb->dev;
 | |
| 
 | |
| 	if (port->data_format & RMNET_FLAGS_INGRESS_MAP_CKSUMV4)
 | |
| 		skb_trim(skb,
 | |
| 			 skb->len - sizeof(struct rmnet_map_dl_csum_trailer));
 | |
| 
 | |
| 	skb->protocol = htons(ETH_P_MAP);
 | |
| 
 | |
| 	/* Command data immediately follows the MAP header */
 | |
| 	cmd = (struct rmnet_map_control_command *)(map_header + 1);
 | |
| 	cmd->cmd_type = type & 0x03;
 | |
| 
 | |
| 	netif_tx_lock(dev);
 | |
| 	dev->netdev_ops->ndo_start_xmit(skb, dev);
 | |
| 	netif_tx_unlock(dev);
 | |
| }
 | |
| 
 | |
| /* Process MAP command frame and send N/ACK message as appropriate. Message cmd
 | |
|  * name is decoded here and appropriate handler is called.
 | |
|  */
 | |
| void rmnet_map_command(struct sk_buff *skb, struct rmnet_port *port)
 | |
| {
 | |
| 	struct rmnet_map_header *map_header = (void *)skb->data;
 | |
| 	struct rmnet_map_control_command *cmd;
 | |
| 	unsigned char command_name;
 | |
| 	unsigned char rc = 0;
 | |
| 
 | |
| 	/* Command data immediately follows the MAP header */
 | |
| 	cmd = (struct rmnet_map_control_command *)(map_header + 1);
 | |
| 	command_name = cmd->command_name;
 | |
| 
 | |
| 	switch (command_name) {
 | |
| 	case RMNET_MAP_COMMAND_FLOW_ENABLE:
 | |
| 		rc = rmnet_map_do_flow_control(skb, port, 1);
 | |
| 		break;
 | |
| 
 | |
| 	case RMNET_MAP_COMMAND_FLOW_DISABLE:
 | |
| 		rc = rmnet_map_do_flow_control(skb, port, 0);
 | |
| 		break;
 | |
| 
 | |
| 	default:
 | |
| 		rc = RMNET_MAP_COMMAND_UNSUPPORTED;
 | |
| 		kfree_skb(skb);
 | |
| 		break;
 | |
| 	}
 | |
| 	if (rc == RMNET_MAP_COMMAND_ACK)
 | |
| 		rmnet_map_send_ack(skb, rc, port);
 | |
| }
 |