source: mainline/uspace/srv/net/netif/lo/lo.c@ c7a8442

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since c7a8442 was c7a8442, checked in by Jakub Jermar <jakub@…>, 15 years ago

Move net_modules.[ch] to the standard library. Note that this functionality is
not directly related to networking so the next step regarding these two files
would be to somehow merge its functionality with what we already have in lib c.

  • Property mode set to 100644
File size: 7.0 KB
Line 
1/*
2 * Copyright (c) 2009 Lukas Mejdrech
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29/** @addtogroup lo
30 * @{
31 */
32
33/** @file
34 * Loopback network interface implementation.
35 */
36
37#include <async.h>
38#include <errno.h>
39#include <err.h>
40#include <stdio.h>
41#include <str.h>
42
43#include <ipc/ipc.h>
44#include <ipc/services.h>
45
46#include <net_messages.h>
47#include <net/modules.h>
48#include <adt/measured_strings.h>
49#include <packet/packet_client.h>
50#include <net_device.h>
51#include <nil_interface.h>
52#include <nil_messages.h>
53#include <netif_interface.h>
54#include <netif_local.h>
55
56/** Default hardware address.
57 */
58#define DEFAULT_ADDR "\0\0\0\0\0\0"
59
60/** Default address length.
61 */
62#define DEFAULT_ADDR_LEN (sizeof(DEFAULT_ADDR) / sizeof(char))
63
64/** Loopback module name.
65 */
66#define NAME "lo"
67
68/** Network interface global data.
69 */
70netif_globals_t netif_globals;
71
72/** Changes the loopback state.
73 * @param[in] device The device structure.
74 * @param[in] state The new device state.
75 * @returns The new state if changed.
76 * @returns EOK otherwise.
77 */
78int change_state_message(netif_device_t * device, device_state_t state);
79
80/** Creates and returns the loopback network interface structure.
81 * @param[in] device_id The new devce identifier.
82 * @param[out] device The device structure.
83 * @returns EOK on success.
84 * @returns EXDEV if one loopback network interface already exists.
85 * @returns ENOMEM if there is not enough memory left.
86 */
87int create(device_id_t device_id, netif_device_t * * device);
88
89int netif_specific_message(ipc_callid_t callid, ipc_call_t * call, ipc_call_t * answer, int * answer_count){
90 return ENOTSUP;
91}
92
93int netif_get_addr_message(device_id_t device_id, measured_string_ref address){
94 if(! address){
95 return EBADMEM;
96 }
97 address->value = str_dup(DEFAULT_ADDR);
98 address->length = DEFAULT_ADDR_LEN;
99 return EOK;
100}
101
102int netif_get_device_stats(device_id_t device_id, device_stats_ref stats){
103 ERROR_DECLARE;
104
105 netif_device_t * device;
106
107 if(! stats){
108 return EBADMEM;
109 }
110 ERROR_PROPAGATE(find_device(device_id, &device));
111 memcpy(stats, (device_stats_ref) device->specific, sizeof(device_stats_t));
112 return EOK;
113}
114
115int change_state_message(netif_device_t * device, device_state_t state)
116{
117 if (device->state != state) {
118 device->state = state;
119
120 printf("%s: State changed to %s\n", NAME,
121 (state == NETIF_ACTIVE) ? "active" : "stopped");
122
123 return state;
124 }
125
126 return EOK;
127}
128
129int create(device_id_t device_id, netif_device_t * * device){
130 int index;
131
132 if(netif_device_map_count(&netif_globals.device_map) > 0){
133 return EXDEV;
134 }else{
135 *device = (netif_device_t *) malloc(sizeof(netif_device_t));
136 if(!(*device)){
137 return ENOMEM;
138 }
139 (** device).specific = malloc(sizeof(device_stats_t));
140 if(! (** device).specific){
141 free(*device);
142 return ENOMEM;
143 }
144 null_device_stats((device_stats_ref)(** device).specific);
145 (** device).device_id = device_id;
146 (** device).nil_phone = -1;
147 (** device).state = NETIF_STOPPED;
148 index = netif_device_map_add(&netif_globals.device_map, (** device).device_id, * device);
149 if(index < 0){
150 free(*device);
151 free((** device).specific);
152 *device = NULL;
153 return index;
154 }
155 }
156 return EOK;
157}
158
159int netif_initialize(void){
160 ipcarg_t phonehash;
161
162 return REGISTER_ME(SERVICE_LO, &phonehash);
163}
164
165int netif_probe_message(device_id_t device_id, int irq, uintptr_t io){
166 ERROR_DECLARE;
167
168 netif_device_t * device;
169
170 // create a new device
171 ERROR_PROPAGATE(create(device_id, &device));
172 // print the settings
173 printf("%s: Device created (id: %d)\n", NAME, device->device_id);
174 return EOK;
175}
176
177int netif_send_message(device_id_t device_id, packet_t packet, services_t sender){
178 ERROR_DECLARE;
179
180 netif_device_t * device;
181 size_t length;
182 packet_t next;
183 int phone;
184
185 ERROR_PROPAGATE(find_device(device_id, &device));
186 if(device->state != NETIF_ACTIVE){
187 netif_pq_release(packet_get_id(packet));
188 return EFORWARD;
189 }
190 next = packet;
191 do{
192 ++ ((device_stats_ref) device->specific)->send_packets;
193 ++ ((device_stats_ref) device->specific)->receive_packets;
194 length = packet_get_data_length(next);
195 ((device_stats_ref) device->specific)->send_bytes += length;
196 ((device_stats_ref) device->specific)->receive_bytes += length;
197 next = pq_next(next);
198 }while(next);
199 phone = device->nil_phone;
200 fibril_rwlock_write_unlock(&netif_globals.lock);
201 nil_received_msg(phone, device_id, packet, sender);
202 fibril_rwlock_write_lock(&netif_globals.lock);
203 return EOK;
204}
205
206int netif_start_message(netif_device_t * device){
207 return change_state_message(device, NETIF_ACTIVE);
208}
209
210int netif_stop_message(netif_device_t * device){
211 return change_state_message(device, NETIF_STOPPED);
212}
213
214/** Default thread for new connections.
215 *
216 * @param[in] iid The initial message identifier.
217 * @param[in] icall The initial message call structure.
218 *
219 */
220static void netif_client_connection(ipc_callid_t iid, ipc_call_t *icall)
221{
222 /*
223 * Accept the connection
224 * - Answer the first IPC_M_CONNECT_ME_TO call.
225 */
226 ipc_answer_0(iid, EOK);
227
228 while(true) {
229 ipc_call_t answer;
230 int answer_count;
231
232 /* Clear the answer structure */
233 refresh_answer(&answer, &answer_count);
234
235 /* Fetch the next message */
236 ipc_call_t call;
237 ipc_callid_t callid = async_get_call(&call);
238
239 /* Process the message */
240 int res = netif_module_message(NAME, callid, &call, &answer,
241 &answer_count);
242
243 /* End if said to either by the message or the processing result */
244 if ((IPC_GET_METHOD(call) == IPC_M_PHONE_HUNGUP) || (res == EHANGUP))
245 return;
246
247 /* Answer the message */
248 answer_call(callid, res, &answer, answer_count);
249 }
250}
251
252int main(int argc, char *argv[])
253{
254 ERROR_DECLARE;
255
256 /* Start the module */
257 if (ERROR_OCCURRED(netif_module_start(netif_client_connection)))
258 return ERROR_CODE;
259
260 return EOK;
261}
262
263/** @}
264 */
Note: See TracBrowser for help on using the repository browser.