OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [rc203soc/] [sw/] [uClinux/] [drivers/] [net/] [loopback.c] - Blame information for rev 1777

Go to most recent revision | Details | Compare with Previous | View Log

Line No. Rev Author Line
1 1626 jcastillo
/*
2
 * INET         An implementation of the TCP/IP protocol suite for the LINUX
3
 *              operating system.  INET is implemented using the  BSD Socket
4
 *              interface as the means of communication with the user level.
5
 *
6
 *              Pseudo-driver for the loopback interface.
7
 *
8
 * Version:     @(#)loopback.c  1.0.4b  08/16/93
9
 *
10
 * Authors:     Ross Biro, <bir7@leland.Stanford.Edu>
11
 *              Fred N. van Kempen, <waltje@uWalt.NL.Mugnet.ORG>
12
 *              Donald Becker, <becker@cesdis.gsfc.nasa.gov>
13
 *
14
 *              Alan Cox        :       Fixed oddments for NET3.014
15
 *              Alan Cox        :       Rejig for NET3.029 snap #3
16
 *              Alan Cox        :       Fixed NET3.029 bugs and sped up
17
 *              Larry McVoy     :       Tiny tweak to double performance
18
 *              Alan Cox        :       Backed out LMV's tweak - the linux mm
19
 *                                      can't take it...
20
 *              Michael Griffith:       Don't bother computing the checksums
21
 *                                      on packets received on the loopback
22
 *                                      interface.
23
 *              Alexey Kuznetsov:       Potential hang under some extreme
24
 *                                      cases removed.
25
 *
26
 *              This program is free software; you can redistribute it and/or
27
 *              modify it under the terms of the GNU General Public License
28
 *              as published by the Free Software Foundation; either version
29
 *              2 of the License, or (at your option) any later version.
30
 */
31
#include <linux/config.h>
32
#include <linux/kernel.h>
33
#include <linux/sched.h>
34
#include <linux/interrupt.h>
35
#include <linux/fs.h>
36
#include <linux/types.h>
37
#include <linux/string.h>
38
#include <linux/socket.h>
39
#include <linux/errno.h>
40
#include <linux/fcntl.h>
41
#include <linux/in.h>
42
 
43
#include <asm/system.h>
44
#include <asm/segment.h>
45
#include <asm/io.h>
46
 
47
#include <linux/inet.h>
48
#include <linux/netdevice.h>
49
#include <linux/etherdevice.h>
50
#include <linux/skbuff.h>
51
#include <net/sock.h>
52
#include <linux/if_ether.h>     /* For the statistics structure. */
53
#include <linux/if_arp.h>       /* For ARPHRD_ETHER */
54
 
55
#define LOOPBACK_MTU (PAGE_SIZE*7/8)
56
 
57
/*
58
 * The higher levels take care of making this non-reentrant (it's
59
 * called with bh's disabled).
60
 */
61
static int loopback_xmit(struct sk_buff *skb, struct device *dev)
62
{
63
        struct enet_statistics *stats = (struct enet_statistics *)dev->priv;
64
        int unlock=1;
65
 
66
        if (skb == NULL || dev == NULL)
67
                return(0);
68
 
69
        /*
70
         *      Optimise so buffers with skb->free=1 are not copied but
71
         *      instead are lobbed from tx queue to rx queue
72
         */
73
 
74
        if(skb->free==0)
75
        {
76
                struct sk_buff *skb2=skb;
77
                skb=skb_clone(skb, GFP_ATOMIC);         /* Clone the buffer */
78
                dev_kfree_skb(skb2, FREE_WRITE);
79
                if(skb==NULL)
80
                        return 0;
81
                unlock=0;
82
        }
83
        else if(skb->sk)
84
        {
85
                /*
86
                 *      Packet sent but looped back around. Cease to charge
87
                 *      the socket for the frame.
88
                 */
89
                atomic_sub(skb->truesize, &skb->sk->wmem_alloc);
90
                skb->sk->write_space(skb->sk);
91
        }
92
 
93
        skb->protocol=eth_type_trans(skb,dev);
94
        skb->dev=dev;
95
#ifndef LOOPBACK_MUST_CHECKSUM
96
        skb->ip_summed = CHECKSUM_UNNECESSARY;
97
#endif
98
        netif_rx(skb);
99
        if(unlock)
100
                skb_device_unlock(skb);
101
 
102
        stats->rx_packets++;
103
        stats->tx_packets++;
104
 
105
        return(0);
106
}
107
 
108
static struct enet_statistics *get_stats(struct device *dev)
109
{
110
        return (struct enet_statistics *)dev->priv;
111
}
112
 
113
static int loopback_open(struct device *dev)
114
{
115
        dev->flags|=IFF_LOOPBACK;
116
        return 0;
117
}
118
 
119
/* Initialize the rest of the LOOPBACK device. */
120
int loopback_init(struct device *dev)
121
{
122
        int i;
123
 
124
        dev->mtu                = LOOPBACK_MTU;
125
        dev->tbusy              = 0;
126
        dev->hard_start_xmit    = loopback_xmit;
127
        dev->hard_header        = eth_header;
128
        dev->hard_header_len    = ETH_HLEN;             /* 14                   */
129
        dev->addr_len           = ETH_ALEN;             /* 6                    */
130
        dev->tx_queue_len       = 50000;                /* No limit on loopback */
131
        dev->type               = ARPHRD_LOOPBACK;      /* 0x0001               */
132
        dev->rebuild_header     = eth_rebuild_header;
133
        dev->open               = loopback_open;
134
        dev->flags              = IFF_LOOPBACK|IFF_BROADCAST;
135
        dev->family             = AF_INET;
136
#ifdef CONFIG_INET    
137
        dev->pa_addr            = in_aton("127.0.0.1");
138
        dev->pa_brdaddr         = in_aton("127.255.255.255");
139
        dev->pa_mask            = in_aton("255.0.0.0");
140
        dev->pa_alen            = 4;
141
#endif  
142
        dev->priv = kmalloc(sizeof(struct enet_statistics), GFP_KERNEL);
143
        if (dev->priv == NULL)
144
                        return -ENOMEM;
145
        memset(dev->priv, 0, sizeof(struct enet_statistics));
146
        dev->get_stats = get_stats;
147
 
148
        /*
149
         *      Fill in the generic fields of the device structure.
150
         */
151
 
152
        for (i = 0; i < DEV_NUMBUFFS; i++)
153
                skb_queue_head_init(&dev->buffs[i]);
154
 
155
        return(0);
156
};

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.