1 /* SPDX-License-Identifier: GPL-2.0+ */
2 /*
3  * SSH message parser.
4  *
5  * Copyright (C) 2019-2020 Maximilian Luz <luzmaximilian@gmail.com>
6  */
7 
8 #ifndef _SURFACE_AGGREGATOR_SSH_PARSER_H
9 #define _SURFACE_AGGREGATOR_SSH_PARSER_H
10 
11 #include <linux/device.h>
12 #include <linux/kfifo.h>
13 #include <linux/slab.h>
14 #include <linux/types.h>
15 
16 #include <linux/surface_aggregator/serial_hub.h>
17 
18 /**
19  * struct sshp_buf - Parser buffer for SSH messages.
20  * @ptr: Pointer to the beginning of the buffer.
21  * @len: Number of bytes used in the buffer.
22  * @cap: Maximum capacity of the buffer.
23  */
24 struct sshp_buf {
25 	u8    *ptr;
26 	size_t len;
27 	size_t cap;
28 };
29 
30 /**
31  * sshp_buf_init() - Initialize a SSH parser buffer.
32  * @buf: The buffer to initialize.
33  * @ptr: The memory backing the buffer.
34  * @cap: The length of the memory backing the buffer, i.e. its capacity.
35  *
36  * Initializes the buffer with the given memory as backing and set its used
37  * length to zero.
38  */
39 static inline void sshp_buf_init(struct sshp_buf *buf, u8 *ptr, size_t cap)
40 {
41 	buf->ptr = ptr;
42 	buf->len = 0;
43 	buf->cap = cap;
44 }
45 
46 /**
47  * sshp_buf_alloc() - Allocate and initialize a SSH parser buffer.
48  * @buf:   The buffer to initialize/allocate to.
49  * @cap:   The desired capacity of the buffer.
50  * @flags: The flags used for allocating the memory.
51  *
52  * Allocates @cap bytes and initializes the provided buffer struct with the
53  * allocated memory.
54  *
55  * Return: Returns zero on success and %-ENOMEM if allocation failed.
56  */
57 static inline int sshp_buf_alloc(struct sshp_buf *buf, size_t cap, gfp_t flags)
58 {
59 	u8 *ptr;
60 
61 	ptr = kzalloc(cap, flags);
62 	if (!ptr)
63 		return -ENOMEM;
64 
65 	sshp_buf_init(buf, ptr, cap);
66 	return 0;
67 }
68 
69 /**
70  * sshp_buf_free() - Free a SSH parser buffer.
71  * @buf: The buffer to free.
72  *
73  * Frees a SSH parser buffer by freeing the memory backing it and then
74  * resetting its pointer to %NULL and length and capacity to zero. Intended to
75  * free a buffer previously allocated with sshp_buf_alloc().
76  */
77 static inline void sshp_buf_free(struct sshp_buf *buf)
78 {
79 	kfree(buf->ptr);
80 	buf->ptr = NULL;
81 	buf->len = 0;
82 	buf->cap = 0;
83 }
84 
85 /**
86  * sshp_buf_drop() - Drop data from the beginning of the buffer.
87  * @buf: The buffer to drop data from.
88  * @n:   The number of bytes to drop.
89  *
90  * Drops the first @n bytes from the buffer. Re-aligns any remaining data to
91  * the beginning of the buffer.
92  */
93 static inline void sshp_buf_drop(struct sshp_buf *buf, size_t n)
94 {
95 	memmove(buf->ptr, buf->ptr + n, buf->len - n);
96 	buf->len -= n;
97 }
98 
99 /**
100  * sshp_buf_read_from_fifo() - Transfer data from a fifo to the buffer.
101  * @buf:  The buffer to write the data into.
102  * @fifo: The fifo to read the data from.
103  *
104  * Transfers the data contained in the fifo to the buffer, removing it from
105  * the fifo. This function will try to transfer as much data as possible,
106  * limited either by the remaining space in the buffer or by the number of
107  * bytes available in the fifo.
108  *
109  * Return: Returns the number of bytes transferred.
110  */
111 static inline size_t sshp_buf_read_from_fifo(struct sshp_buf *buf,
112 					     struct kfifo *fifo)
113 {
114 	size_t n;
115 
116 	n =  kfifo_out(fifo, buf->ptr + buf->len, buf->cap - buf->len);
117 	buf->len += n;
118 
119 	return n;
120 }
121 
122 /**
123  * sshp_buf_span_from() - Initialize a span from the given buffer and offset.
124  * @buf:    The buffer to create the span from.
125  * @offset: The offset in the buffer at which the span should start.
126  * @span:   The span to initialize (output).
127  *
128  * Initializes the provided span to point to the memory at the given offset in
129  * the buffer, with the length of the span being capped by the number of bytes
130  * used in the buffer after the offset (i.e. bytes remaining after the
131  * offset).
132  *
133  * Warning: This function does not validate that @offset is less than or equal
134  * to the number of bytes used in the buffer or the buffer capacity. This must
135  * be guaranteed by the caller.
136  */
137 static inline void sshp_buf_span_from(struct sshp_buf *buf, size_t offset,
138 				      struct ssam_span *span)
139 {
140 	span->ptr = buf->ptr + offset;
141 	span->len = buf->len - offset;
142 }
143 
144 bool sshp_find_syn(const struct ssam_span *src, struct ssam_span *rem);
145 
146 int sshp_parse_frame(const struct device *dev, const struct ssam_span *source,
147 		     struct ssh_frame **frame, struct ssam_span *payload,
148 		     size_t maxlen);
149 
150 int sshp_parse_command(const struct device *dev, const struct ssam_span *source,
151 		       struct ssh_command **command,
152 		       struct ssam_span *command_data);
153 
154 #endif /* _SURFACE_AGGREGATOR_SSH_PARSER_h */
155