mirror of
				https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
				synced 2025-10-26 00:44:03 +00:00 
			
		
		
		
	 854e981cc6
			
		
	
	
		854e981cc6
		
	
	
	
	
		
			
			The C99 initialization, with GCC's bad handling, for 6K wide structs (which
_aren't_ on the stack), is causing GCC to use 12K for these silly procs with 3
vars.  Workaround this.
Note that .name = { '\0' } translates to memset(->name, 0, '->name' size) - I verified
this with GCC's docs and a testprogram.
Signed-off-by: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Cc: Jeff Dike <jdike@addtoit.com>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
		
	
			
		
			
				
	
	
		
			106 lines
		
	
	
		
			2.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			106 lines
		
	
	
		
			2.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #ifndef __UM_SLIP_COMMON_H
 | |
| #define __UM_SLIP_COMMON_H
 | |
| 
 | |
| #define BUF_SIZE 1500
 | |
|  /* two bytes each for a (pathological) max packet of escaped chars +  *
 | |
|   * terminating END char + initial END char                            */
 | |
| #define ENC_BUF_SIZE (2 * BUF_SIZE + 2)
 | |
| 
 | |
| /* SLIP protocol characters. */
 | |
| #define SLIP_END             0300	/* indicates end of frame	*/
 | |
| #define SLIP_ESC             0333	/* indicates byte stuffing	*/
 | |
| #define SLIP_ESC_END         0334	/* ESC ESC_END means END 'data'	*/
 | |
| #define SLIP_ESC_ESC         0335	/* ESC ESC_ESC means ESC 'data'	*/
 | |
| 
 | |
| static inline int slip_unesc(unsigned char c, unsigned char *buf, int *pos,
 | |
|                              int *esc)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	switch(c){
 | |
| 	case SLIP_END:
 | |
| 		*esc = 0;
 | |
| 		ret=*pos;
 | |
| 		*pos=0;
 | |
| 		return(ret);
 | |
| 	case SLIP_ESC:
 | |
| 		*esc = 1;
 | |
| 		return(0);
 | |
| 	case SLIP_ESC_ESC:
 | |
| 		if(*esc){
 | |
| 			*esc = 0;
 | |
| 			c = SLIP_ESC;
 | |
| 		}
 | |
| 		break;
 | |
| 	case SLIP_ESC_END:
 | |
| 		if(*esc){
 | |
| 			*esc = 0;
 | |
| 			c = SLIP_END;
 | |
| 		}
 | |
| 		break;
 | |
| 	}
 | |
| 	buf[(*pos)++] = c;
 | |
| 	return(0);
 | |
| }
 | |
| 
 | |
| static inline int slip_esc(unsigned char *s, unsigned char *d, int len)
 | |
| {
 | |
| 	unsigned char *ptr = d;
 | |
| 	unsigned char c;
 | |
| 
 | |
| 	/*
 | |
| 	 * Send an initial END character to flush out any
 | |
| 	 * data that may have accumulated in the receiver
 | |
| 	 * due to line noise.
 | |
| 	 */
 | |
| 
 | |
| 	*ptr++ = SLIP_END;
 | |
| 
 | |
| 	/*
 | |
| 	 * For each byte in the packet, send the appropriate
 | |
| 	 * character sequence, according to the SLIP protocol.
 | |
| 	 */
 | |
| 
 | |
| 	while (len-- > 0) {
 | |
| 		switch(c = *s++) {
 | |
| 		case SLIP_END:
 | |
| 			*ptr++ = SLIP_ESC;
 | |
| 			*ptr++ = SLIP_ESC_END;
 | |
| 			break;
 | |
| 		case SLIP_ESC:
 | |
| 			*ptr++ = SLIP_ESC;
 | |
| 			*ptr++ = SLIP_ESC_ESC;
 | |
| 			break;
 | |
| 		default:
 | |
| 			*ptr++ = c;
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 	*ptr++ = SLIP_END;
 | |
| 	return (ptr - d);
 | |
| }
 | |
| 
 | |
| struct slip_proto {
 | |
| 	unsigned char ibuf[ENC_BUF_SIZE];
 | |
| 	unsigned char obuf[ENC_BUF_SIZE];
 | |
| 	int more; /* more data: do not read fd until ibuf has been drained */
 | |
| 	int pos;
 | |
| 	int esc;
 | |
| };
 | |
| 
 | |
| static inline void slip_proto_init(struct slip_proto * slip)
 | |
| {
 | |
| 	memset(slip->ibuf, 0, sizeof(slip->ibuf));
 | |
| 	memset(slip->obuf, 0, sizeof(slip->obuf));
 | |
| 	slip->more = 0;
 | |
| 	slip->pos = 0;
 | |
| 	slip->esc = 0;
 | |
| }
 | |
| 
 | |
| extern int slip_proto_read(int fd, void *buf, int len,
 | |
| 			   struct slip_proto *slip);
 | |
| extern int slip_proto_write(int fd, void *buf, int len,
 | |
| 			    struct slip_proto *slip);
 | |
| 
 | |
| #endif
 |