📄 slip.c
字号:
/*
* slip.c This module implements the SLIP protocol for kernel-based
* devices like TTY. It interfaces between a raw TTY, and the
* kernel's INET protocol layers (via DDI).
*
* Version: @(#)slip.c 0.7.6 05/25/93
*
* Authors: Laurence Culhane, <loz@holmes.demon.co.uk>
* Fred N. van Kempen, <waltje@uwalt.nl.mugnet.org>
*
* Fixes:
* Alan Cox : Sanity checks and avoid tx overruns.
* Has a new sl->mtu field.
* Alan Cox : Found cause of overrun. ifconfig sl0 mtu upwards.
* Driver now spots this and grows/shrinks its buffers(hack!).
* Memory leak if you run out of memory setting up a slip driver fixed.
* Matt Dillon : Printable slip (borrowed from NET2E)
* Pauline Middelink : Slip driver fixes.
* Alan Cox : Honours the old SL_COMPRESSED flag
* Alan Cox : KISS AX.25 and AXUI IP support
* Michael Riepe : Automatic CSLIP recognition added
* Charles Hedrick : CSLIP header length problem fix.
* Alan Cox : Corrected non-IP cases of the above.
*/
#include <asm/segment.h>
#include <asm/system.h>
#include <linux/config.h>
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/sched.h>
#include <linux/string.h>
#include <linux/mm.h>
#include <linux/socket.h>
#include <linux/sockios.h>
#include <linux/interrupt.h>
#include <linux/tty.h>
#include <linux/errno.h>
#include <linux/stat.h>
#include <linux/in.h>
#include "inet.h"
#include "dev.h"
#ifdef CONFIG_AX25
#include "ax25.h"
#endif
#include "eth.h"
#include "ip.h"
#include "route.h"
#include "protocol.h"
#include "tcp.h"
#include "skbuff.h"
#include "sock.h"
#include "arp.h"
#include "slip.h"
#include "slhc.h"
#define SLIP_VERSION "0.7.5"
/* Define some IP layer stuff. Not all systems have it. */
#ifdef SL_DUMP
# define IP_VERSION 4 /* version# of our IP software */
# define IPF_F_OFFSET 0x1fff /* Offset field */
# define IPF_DF 0x4000 /* Don't fragment flag */
# define IPF_MF 0x2000 /* More Fragments flag */
# define IP_OF_COPIED 0x80 /* Copied-on-fragmentation flag */
# define IP_OF_CLASS 0x60 /* Option class */
# define IP_OF_NUMBER 0x1f /* Option number */
#endif
static struct slip sl_ctrl[SL_NRUNIT];
static struct tty_ldisc sl_ldisc;
static int already = 0;
/* Dump the contents of an IP datagram. */
static void
ip_dump(unsigned char *ptr, int len)
{
#ifdef SL_DUMP
struct iphdr *ip;
struct tcphdr *th;
int dlen, doff;
if (inet_debug != DBG_SLIP) return;
ip = (struct iphdr *) ptr;
th = (struct tcphdr *) (ptr + ip->ihl * 4);
printk("\r%s -> %s seq %lx ack %lx len %d\n",
in_ntoa(ip->saddr), in_ntoa(ip->daddr),
ntohl(th->seq), ntohl(th->ack_seq), ntohs(ip->tot_len));
return;
printk("\r*****\n");
printk("%p %d\n", ptr, len);
ip = (struct iphdr *) ptr;
dlen = ntohs(ip->tot_len);
doff = ((ntohs(ip->frag_off) & IPF_F_OFFSET) << 3);
printk("SLIP: %s->", in_ntoa(ip->saddr));
printk("%s\n", in_ntoa(ip->daddr));
printk(" len %u ihl %u ver %u ttl %u prot %u",
dlen, ip->ihl, ip->version, ip->ttl, ip->protocol);
if (ip->tos != 0) printk(" tos %u", ip->tos);
if (doff != 0 || (ntohs(ip->frag_off) & IPF_MF))
printk(" id %u offs %u", ntohs(ip->id), doff);
if (ntohs(ip->frag_off) & IPF_DF) printk(" DF");
if (ntohs(ip->frag_off) & IPF_MF) printk(" MF");
printk("\n*****\n");
#endif
}
#if 0
void clh_dump(unsigned char *cp, int len)
{
if (len > 60)
len = 60;
printk("%d:", len);
while (len > 0) {
printk(" %x", *cp++);
len--;
}
printk("\n\n");
}
#endif
/* Initialize a SLIP control block for use. */
static void
sl_initialize(struct slip *sl, struct device *dev)
{
sl->inuse = 0;
sl->sending = 0;
sl->escape = 0;
sl->flags = 0;
#ifdef SL_ADAPTIVE
sl->mode = SL_MODE_ADAPTIVE; /* automatic CSLIP recognition */
#else
#ifdef SL_COMPRESSED
sl->mode = SL_MODE_CSLIP | SL_MODE_ADAPTIVE; /* Default */
#else
sl->mode = SL_MODE_SLIP; /* Default for non compressors */
#endif
#endif
sl->line = dev->base_addr;
sl->tty = NULL;
sl->dev = dev;
sl->slcomp = NULL;
/* Clear all pointers. */
sl->rbuff = NULL;
sl->xbuff = NULL;
sl->cbuff = NULL;
sl->rhead = NULL;
sl->rend = NULL;
dev->rmem_end = (unsigned long) NULL;
dev->rmem_start = (unsigned long) NULL;
dev->mem_end = (unsigned long) NULL;
dev->mem_start = (unsigned long) NULL;
}
/* Find a SLIP channel from its `tty' link. */
static struct slip *
sl_find(struct tty_struct *tty)
{
struct slip *sl;
int i;
if (tty == NULL) return(NULL);
for (i = 0; i < SL_NRUNIT; i++) {
sl = &sl_ctrl[i];
if (sl->tty == tty) return(sl);
}
return(NULL);
}
/* Find a free SLIP channel, and link in this `tty' line. */
static inline struct slip *
sl_alloc(void)
{
unsigned long flags;
struct slip *sl;
int i;
save_flags (flags);
cli();
for (i = 0; i < SL_NRUNIT; i++) {
sl = &sl_ctrl[i];
if (sl->inuse == 0) {
sl->inuse = 1;
sl->tty = NULL;
restore_flags(flags);
return(sl);
}
}
restore_flags(flags);
return(NULL);
}
/* Free a SLIP channel. */
static inline void
sl_free(struct slip *sl)
{
unsigned long flags;
if (sl->inuse) {
save_flags(flags);
cli();
sl->inuse = 0;
sl->tty = NULL;
restore_flags(flags);
}
}
/* MTU has been changed by the IP layer. Unfortunately we are not told about this, but
we spot it ourselves and fix things up. We could be in an upcall from the tty
driver, or in an ip packet queue. */
static void sl_changedmtu(struct slip *sl)
{
struct device *dev=sl->dev;
unsigned char *tb,*rb,*cb,*tf,*rf,*cf;
int l;
int omtu=sl->mtu;
sl->mtu=dev->mtu;
l=(dev->mtu *2);
/*
* allow for arrival of larger UDP packets, even if we say not to
* also fixes a bug in which SunOS sends 512-byte packets even with
* an MSS of 128
*/
if (l < (576 * 2))
l = 576 * 2;
DPRINTF((DBG_SLIP,"SLIP: mtu changed!\n"));
tb= (unsigned char *) kmalloc(l + 4, GFP_ATOMIC);
rb= (unsigned char *) kmalloc(l + 4, GFP_ATOMIC);
cb= (unsigned char *) kmalloc(l + 4, GFP_ATOMIC);
if(tb==NULL || rb==NULL || cb==NULL)
{
printk("Unable to grow slip buffers. MTU change cancelled.\n");
sl->mtu=omtu;
dev->mtu=omtu;
if(tb!=NULL)
kfree(tb);
if(rb!=NULL)
kfree(rb);
if(cb!=NULL)
kfree(cb);
return;
}
cli();
tf=(unsigned char *)sl->dev->mem_start;
sl->dev->mem_start=(unsigned long)tb;
sl->dev->mem_end=(unsigned long) (sl->dev->mem_start + l);
rf=(unsigned char *)sl->dev->rmem_start;
sl->dev->rmem_start=(unsigned long)rb;
sl->dev->rmem_end=(unsigned long) (sl->dev->rmem_start + l);
sl->xbuff = (unsigned char *) sl->dev->mem_start;
sl->rbuff = (unsigned char *) sl->dev->rmem_start;
sl->rend = (unsigned char *) sl->dev->rmem_end;
sl->rhead = sl->rbuff;
cf=sl->cbuff;
sl->cbuff=cb;
sl->escape=0;
sl->sending=0;
sl->rcount=0;
sti();
if(rf!=NULL)
kfree(rf);
if(tf!=NULL)
kfree(tf);
if(cf!=NULL)
kfree(cf);
}
/* Stuff one byte into a SLIP receiver buffer. */
static inline void
sl_enqueue(struct slip *sl, unsigned char c)
{
unsigned long flags;
save_flags(flags);
cli();
if (sl->rhead < sl->rend) {
*sl->rhead = c;
sl->rhead++;
sl->rcount++;
} else sl->roverrun++;
restore_flags(flags);
}
/* Release 'i' bytes from a SLIP receiver buffer. */
static inline void
sl_dequeue(struct slip *sl, int i)
{
unsigned long flags;
save_flags(flags);
cli();
if (sl->rhead > sl->rbuff) {
sl->rhead -= i;
sl->rcount -= i;
}
restore_flags(flags);
}
/* Set the "sending" flag. This must be atomic, hence the ASM. */
static inline void
sl_lock(struct slip *sl)
{
unsigned long flags;
save_flags(flags);
cli();
sl->sending = 1;
sl->dev->tbusy = 1;
restore_flags(flags);
}
/* Clear the "sending" flag. This must be atomic, hence the ASM. */
static inline void
sl_unlock(struct slip *sl)
{
unsigned long flags;
save_flags(flags);
cli();
sl->sending = 0;
sl->dev->tbusy = 0;
restore_flags(flags);
}
/* Send one completely decapsulated IP datagram to the IP layer. */
static void
sl_bump(struct slip *sl)
{
int done;
unsigned char c;
unsigned long flags;
int count;
count = sl->rcount;
if (sl->mode & (SL_MODE_ADAPTIVE | SL_MODE_CSLIP)) {
if ((c = sl->rbuff[0]) & SL_TYPE_COMPRESSED_TCP) {
#if 1
/* ignore compressed packets when CSLIP is off */
if (!(sl->mode & SL_MODE_CSLIP)) {
printk("SLIP: compressed packet ignored\n");
return;
}
#endif
/* make sure we've reserved enough space for uncompress to use */
save_flags(flags);
cli();
if ((sl->rhead + 80) < sl->rend) {
sl->rhead += 80;
sl->rcount += 80;
done = 1;
} else {
sl->roverrun++;
done = 0;
}
restore_flags(flags);
if (! done) /* not enough space available */
return;
count = slhc_uncompress(sl->slcomp, sl->rbuff, count);
if (count <= 0) {
sl->errors++;
return;
}
} else if (c >= SL_TYPE_UNCOMPRESSED_TCP) {
if (!(sl->mode & SL_MODE_CSLIP)) {
/* turn on header compression */
sl->mode |= SL_MODE_CSLIP;
printk("SLIP: header compression turned on\n");
}
sl->rbuff[0] &= 0x4f;
if (slhc_remember(sl->slcomp, sl->rbuff, count) <= 0) {
sl->errors++;
return;
}
}
}
DPRINTF((DBG_SLIP, "<< \"%s\" recv:\r\n", sl->dev->name));
ip_dump(sl->rbuff, sl->rcount);
/* Bump the datagram to the upper layers... */
do {
DPRINTF((DBG_SLIP, "SLIP: packet is %d at 0x%X\n",
sl->rcount, sl->rbuff));
/* clh_dump(sl->rbuff, count); */
done = dev_rint(sl->rbuff, count, 0, sl->dev);
if (done == 0 || done == 1) break;
} while(1);
sl->rpacket++;
}
/* TTY finished sending a datagram, so clean up. */
static void
sl_next(struct slip *sl)
{
DPRINTF((DBG_SLIP, "SLIP: sl_next(0x%X) called!\n", sl));
sl_unlock(sl);
dev_tint(sl->dev);
}
/* Encapsulate one IP datagram and stuff into a TTY queue. */
static void
sl_encaps(struct slip *sl, unsigned char *icp, int len)
{
unsigned char *bp, *p;
int count;
DPRINTF((DBG_SLIP, "SLIP: sl_encaps(0x%X, %d) called\n", icp, len));
DPRINTF((DBG_SLIP, ">> \"%s\" sent:\r\n", sl->dev->name));
ip_dump(icp, len);
if(sl->mtu != sl->dev->mtu) /* Someone has been ifconfigging */
sl_changedmtu(sl);
if(len>sl->mtu) /* Sigh, shouldn't occur BUT ... */
{
len=sl->mtu;
printk("slip: truncating oversized transmit packet!\n");
}
p = icp;
if(sl->mode & SL_MODE_CSLIP)
len = slhc_compress(sl->slcomp, p, len, sl->cbuff, &p, 1);
#ifdef OLD
/*
* Send an initial END character to flush out any
* data that may have accumulated in the receiver
* due to line noise.
*/
bp = sl->xbuff;
*bp++ = END;
count = 1;
/*
* For each byte in the packet, send the appropriate
* character sequence, according to the SLIP protocol.
*/
while(len-- > 0) {
c = *p++;
switch(c) {
case END:
*bp++ = ESC;
*bp++ = ESC_END;
count += 2;
break;
case ESC:
*bp++ = ESC;
*bp++ = ESC_ESC;
count += 2;
break;
default:
*bp++ = c;
count++;
}
}
*bp++ = END;
count++;
#else
if(sl->mode & SL_MODE_SLIP6)
count=slip_esc6(p, (unsigned char *)sl->xbuff,len);
else
count=slip_esc(p, (unsigned char *)sl->xbuff,len);
#endif
sl->spacket++;
bp = sl->xbuff;
/* Tell TTY to send it on its way. */
DPRINTF((DBG_SLIP, "SLIP: kicking TTY for %d bytes at 0x%X\n", count, bp));
if (tty_write_data(sl->tty, (char *) bp, count,
(void (*)(void *))sl_next, (void *) sl) == 0) {
DPRINTF((DBG_SLIP, "SLIP: TTY already done with %d bytes!\n", count));
sl_next(sl);
}
}
/*static void sl_hex_dump(unsigned char *x,int l)
{
int n=0;
printk("sl_xmit: (%d bytes)\n",l);
while(l)
{
printk("%2X ",(int)*x++);
l--;
n++;
if(n%32==0)
printk("\n");
}
if(n%32)
printk("\n");
}*/
/* Encapsulate an IP datagram and kick it into a TTY queue. */
static int
sl_xmit(struct sk_buff *skb, struct device *dev)
{
struct tty_struct *tty;
struct slip *sl;
int size;
/* Find the correct SLIP channel to use. */
sl = &sl_ctrl[dev->base_addr];
tty = sl->tty;
DPRINTF((DBG_SLIP, "SLIP: sl_xmit(\"%s\") skb=0x%X busy=%d\n",
dev->name, skb, sl->sending));
/*
* If we are busy already- too bad. We ought to be able
* to queue things at this point, to allow for a little
* frame buffer. Oh well...
*/
if (sl->sending) {
DPRINTF((DBG_SLIP, "SLIP: sl_xmit: BUSY\r\n"));
sl->sbusy++;
return(1);
}
/* We were not, so we are now... :-) */
if (skb != NULL) {
#ifdef CONFIG_AX25
if(sl->mode & SL_MODE_AX25)
{
if(!skb->arp && dev->rebuild_header(skb->data,dev))
{
skb->dev=dev;
arp_queue(skb);
return 0;
}
skb->arp=1;
}
#endif
sl_lock(sl);
size = skb->len;
if (!(sl->mode & SL_MODE_AX25)) {
if (size < sizeof(struct iphdr)) {
printk("Runt IP frame fed to slip!\n");
} else {
size = ((struct iphdr *)(skb->data))->tot_len;
size = ntohs(size);
}
}
/* sl_hex_dump(skb->data,skb->len);*/
sl_encaps(sl, skb->data, size);
if (skb->free)
kfree_skb(skb, FREE_WRITE);
}
return(0);
}
/* Return the frame type ID. This is normally IP but maybe be AX.25. */
static unsigned short
sl_type_trans (struct sk_buff *skb, struct device *dev)
{
#ifdef CONFIG_AX25
struct slip *sl=&sl_ctrl[dev->base_addr];
if(sl->mode&SL_MODE_AX25)
return(NET16(ETH_P_AX25));
#endif
return(NET16(ETH_P_IP));
}
/* Fill in the MAC-level header. Not used by SLIP. */
static int
sl_header(unsigned char *buff, struct device *dev, unsigned short type,
unsigned long daddr, unsigned long saddr, unsigned len)
{
#ifdef CONFIG_AX25
struct slip *sl=&sl_ctrl[dev->base_addr];
if((sl->mode&SL_MODE_AX25) && type!=NET16(ETH_P_AX25))
return ax25_encapsulate_ip(buff,dev,type,daddr,saddr,len);
#endif
return(0);
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -