ViewVC Help
View File | Revision Log | Show Annotations | Revision Graph | Root Listing
root/cebix/BasiliskII/src/Windows/ether_windows.cpp
Revision: 1.6
Committed: 2006-04-29T10:57:56Z (18 years, 6 months ago) by gbeauche
Branch: MAIN
Changes since 1.5: +1321 -1233 lines
Log Message:
slirp now works on windows

File Contents

# Content
1 /*
2 * ether_windows.cpp - Ethernet device driver
3 *
4 * Basilisk II (C) 1997-2005 Christian Bauer
5 *
6 * Windows platform specific code copyright (C) Lauri Pesonen
7 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21 */
22
23 #include <process.h>
24 #include <windowsx.h>
25 #include <ctype.h>
26
27 #include "sysdeps.h"
28 #include "cpu_emulation.h"
29 #include "main.h"
30 #include "macos_util.h"
31 #include "prefs.h"
32 #include "user_strings.h"
33 #include "ether.h"
34 #include "ether_defs.h"
35 #include "b2ether/multiopt.h"
36 #include "b2ether/inc/b2ether_hl.h"
37 #include "ether_windows.h"
38 #include "router/router.h"
39 #include "kernel_windows.h"
40 #include "libslirp.h"
41
42 // Define to let the slirp library determine the right timeout for select()
43 #define USE_SLIRP_TIMEOUT 1
44
45
46 #define DEBUG 0
47 #define MONITOR 0
48
49 #if DEBUG
50 #pragma optimize("",off)
51 #endif
52
53 #include "debug.h"
54
55
56 // Ethernet device types
57 enum {
58 NET_IF_B2ETHER,
59 NET_IF_ROUTER,
60 NET_IF_SLIRP,
61 NET_IF_FAKE,
62 };
63
64 // Options
65 bool ether_use_permanent = true;
66 static int16 ether_multi_mode = ETHER_MULTICAST_MAC;
67
68 // Global variables
69 HANDLE ether_th;
70 unsigned int ether_tid;
71 HANDLE ether_th1;
72 HANDLE ether_th2;
73 static int net_if_type = -1; // Ethernet device type
74 #ifdef SHEEPSHAVER
75 static bool net_open = false; // Flag: initialization succeeded, network device open
76 uint8 ether_addr[6]; // Our Ethernet address
77 #endif
78
79 // These are protected by queue_csection
80 // Controls transfer for read thread to feed thread
81 static CRITICAL_SECTION queue_csection;
82 typedef struct _win_queue_t {
83 uint8 *buf;
84 int sz;
85 } win_queue_t;
86 #define MAX_QUEUE_ITEMS 1024
87 static win_queue_t queue[MAX_QUEUE_ITEMS];
88 static int queue_head = 0;
89 static int queue_inx = 0;
90 static bool wait_request = true;
91
92
93
94 // Read thread protected packet pool
95 static CRITICAL_SECTION fetch_csection;
96 // Some people use pools as large as 64.
97 #define PACKET_POOL_COUNT 10
98 static LPPACKET packets[PACKET_POOL_COUNT];
99 static bool wait_request2 = false;
100
101
102
103 // Write thread packet queue
104 static CRITICAL_SECTION send_csection;
105 static LPPACKET send_queue = 0;
106
107
108 // Write thread free packet pool
109 static CRITICAL_SECTION wpool_csection;
110 static LPPACKET write_packet_pool = 0;
111
112
113
114 // Try to deal with echos. Protected by fetch_csection.
115 // The code should be moved to the driver. No need to lift
116 // the echo packets to the application level.
117 // MAX_ECHO must be a power of two.
118 #define MAX_ECHO (1<<2)
119 static int echo_count = 0;
120 typedef uint8 echo_t[1514];
121 static echo_t pending_packet[MAX_ECHO];
122 static int pending_packet_sz[MAX_ECHO];
123
124
125 // List of attached protocols
126 struct NetProtocol {
127 NetProtocol *next;
128 uint16 type;
129 uint32 handler;
130 };
131
132 static NetProtocol *prot_list = NULL;
133
134
135 static LPADAPTER fd = 0;
136 static bool thread_active = false;
137 static bool thread_active_1 = false;
138 static bool thread_active_2 = false;
139 static bool thread_active_3 = false;
140 static HANDLE int_ack = 0;
141 static HANDLE int_sig = 0;
142 static HANDLE int_sig2 = 0;
143 static HANDLE int_send_now = 0;
144
145 // Prototypes
146 static WINAPI unsigned int slirp_receive_func(void *arg);
147 static WINAPI unsigned int ether_thread_feed_int(void *arg);
148 static WINAPI unsigned int ether_thread_get_packets_nt(void *arg);
149 static WINAPI unsigned int ether_thread_write_packets(void *arg);
150 static void init_queue(void);
151 static void final_queue(void);
152 static bool allocate_read_packets(void);
153 static void free_read_packets(void);
154 static void free_write_packets(void);
155 static int16 ether_do_add_multicast(uint8 *addr);
156 static int16 ether_do_del_multicast(uint8 *addr);
157 static int16 ether_do_write(uint32 arg);
158 static void ether_do_interrupt(void);
159
160
161 /*
162 * Find protocol in list
163 */
164
165 static NetProtocol *find_protocol(uint16 type)
166 {
167 // All 802.2 types are the same
168 if (type <= 1500)
169 type = 0;
170
171 // Search list (we could use hashing here but there are usually only three
172 // handlers installed: 0x0000 for AppleTalk and 0x0800/0x0806 for TCP/IP)
173 NetProtocol *p = prot_list;
174 while (p) {
175 if (p->type == type)
176 return p;
177 p = p->next;
178 }
179 return NULL;
180 }
181
182
183 /*
184 * Initialization
185 */
186
187 bool ether_init(void)
188 {
189 char str[256];
190
191 // Do nothing if no Ethernet device specified
192 const char *name = PrefsFindString("ether");
193 if (name == NULL)
194 return false;
195
196 ether_multi_mode = PrefsFindInt32("ethermulticastmode");
197 ether_use_permanent = PrefsFindBool("etherpermanentaddress");
198
199 // Determine Ethernet device type
200 net_if_type = -1;
201 if (strcmp(name, "router") == 0)
202 net_if_type = NET_IF_ROUTER;
203 else if (strcmp(name, "slirp") == 0)
204 net_if_type = NET_IF_SLIRP;
205 else
206 net_if_type = NET_IF_B2ETHER;
207
208 // Initialize NAT-Router
209 if (net_if_type == NET_IF_ROUTER) {
210 if (!router_init())
211 net_if_type = NET_IF_FAKE;
212 }
213
214 // Initialize slirp library
215 if (net_if_type == NET_IF_SLIRP) {
216 if (slirp_init() < 0) {
217 sprintf(str, GetString(STR_SLIRP_NO_DNS_FOUND_WARN));
218 WarningAlert(str);
219 return false;
220 }
221 }
222
223 // Open ethernet device
224 const char *dev_name;
225 switch (net_if_type) {
226 case NET_IF_B2ETHER:
227 dev_name = PrefsFindString("etherguid");
228 break;
229 }
230 if (net_if_type == NET_IF_B2ETHER) {
231 if (dev_name == NULL) {
232 WarningAlert("No ethernet device GUID specified. Ethernet is not available.");
233 goto open_error;
234 }
235
236 fd = PacketOpenAdapter( dev_name, ether_multi_mode );
237 if (!fd) {
238 sprintf(str, "Could not open ethernet adapter %s.", dev_name);
239 WarningAlert(str);
240 goto open_error;
241 }
242
243 // Get Ethernet address
244 if(!PacketGetMAC(fd,ether_addr,ether_use_permanent)) {
245 sprintf(str, "Could not get hardware address of device %s. Ethernet is not available.", dev_name);
246 WarningAlert(str);
247 goto open_error;
248 }
249 D(bug("Real ethernet address %02x %02x %02x %02x %02x %02x\n", ether_addr[0], ether_addr[1], ether_addr[2], ether_addr[3], ether_addr[4], ether_addr[5]));
250
251 const char *ether_fake_address;
252 ether_fake_address = PrefsFindString("etherfakeaddress");
253 if(ether_fake_address && strlen(ether_fake_address) == 12) {
254 char sm[10];
255 strcpy( sm, "0x00" );
256 for( int i=0; i<6; i++ ) {
257 sm[2] = ether_fake_address[i*2];
258 sm[3] = ether_fake_address[i*2+1];
259 ether_addr[i] = (uint8)strtoul(sm,0,0);
260 }
261 D(bug("Fake ethernet address %02x %02x %02x %02x %02x %02x\n", ether_addr[0], ether_addr[1], ether_addr[2], ether_addr[3], ether_addr[4], ether_addr[5]));
262 }
263 }
264 else if (net_if_type == NET_IF_SLIRP) {
265 ether_addr[0] = 0x52;
266 ether_addr[1] = 0x54;
267 ether_addr[2] = 0x00;
268 ether_addr[3] = 0x12;
269 ether_addr[4] = 0x34;
270 ether_addr[5] = 0x56;
271 D(bug("Ethernet address %02x %02x %02x %02x %02x %02x\n", ether_addr[0], ether_addr[1], ether_addr[2], ether_addr[3], ether_addr[4], ether_addr[5]));
272 }
273 else {
274 memcpy( ether_addr, router_mac_addr, 6 );
275 D(bug("Fake ethernet address (same as router) %02x %02x %02x %02x %02x %02x\n", ether_addr[0], ether_addr[1], ether_addr[2], ether_addr[3], ether_addr[4], ether_addr[5]));
276 }
277
278 // Start packet reception thread
279 int_ack = CreateSemaphore( 0, 0, 1, NULL);
280 if(!int_ack) {
281 WarningAlert("WARNING: Cannot create int_ack semaphore");
282 goto open_error;
283 }
284
285 // nonsignaled
286 int_sig = CreateSemaphore( 0, 0, 1, NULL);
287 if(!int_sig) {
288 WarningAlert("WARNING: Cannot create int_sig semaphore");
289 goto open_error;
290 }
291
292 int_sig2 = CreateSemaphore( 0, 0, 1, NULL);
293 if(!int_sig2) {
294 WarningAlert("WARNING: Cannot create int_sig2 semaphore");
295 goto open_error;
296 }
297
298 int_send_now = CreateSemaphore( 0, 0, 1, NULL);
299 if(!int_send_now) {
300 WarningAlert("WARNING: Cannot create int_send_now semaphore");
301 goto open_error;
302 }
303
304 init_queue();
305
306 if(!allocate_read_packets()) goto open_error;
307
308 // No need to enter wait state if we can avoid it.
309 // These all terminate fast.
310
311 if(pfnInitializeCriticalSectionAndSpinCount) {
312 pfnInitializeCriticalSectionAndSpinCount( &fetch_csection, 5000 );
313 } else {
314 InitializeCriticalSection( &fetch_csection );
315 }
316 if(pfnInitializeCriticalSectionAndSpinCount) {
317 pfnInitializeCriticalSectionAndSpinCount( &queue_csection, 5000 );
318 } else {
319 InitializeCriticalSection( &queue_csection );
320 }
321 if(pfnInitializeCriticalSectionAndSpinCount) {
322 pfnInitializeCriticalSectionAndSpinCount( &send_csection, 5000 );
323 } else {
324 InitializeCriticalSection( &send_csection );
325 }
326 if(pfnInitializeCriticalSectionAndSpinCount) {
327 pfnInitializeCriticalSectionAndSpinCount( &wpool_csection, 5000 );
328 } else {
329 InitializeCriticalSection( &wpool_csection );
330 }
331
332 ether_th = (HANDLE)_beginthreadex( 0, 0, ether_thread_feed_int, 0, 0, &ether_tid );
333 if (!ether_th) {
334 D(bug("Failed to create ethernet thread\n"));
335 goto open_error;
336 }
337 thread_active = true;
338
339 unsigned int dummy;
340 ether_th2 = (HANDLE)_beginthreadex( 0, 0,
341 net_if_type == NET_IF_SLIRP ? slirp_receive_func : ether_thread_get_packets_nt,
342 0, 0, &dummy );
343 ether_th1 = (HANDLE)_beginthreadex( 0, 0, ether_thread_write_packets, 0, 0, &dummy );
344
345 // Everything OK
346 return true;
347
348 open_error:
349 if (thread_active) {
350 TerminateThread(ether_th,0);
351 ether_th = 0;
352 if (int_ack)
353 CloseHandle(int_ack);
354 int_ack = 0;
355 if(int_sig)
356 CloseHandle(int_sig);
357 int_sig = 0;
358 if(int_sig2)
359 CloseHandle(int_sig2);
360 int_sig2 = 0;
361 if(int_send_now)
362 CloseHandle(int_send_now);
363 int_send_now = 0;
364 thread_active = false;
365 }
366 if(net_if_type == NET_IF_B2ETHER) {
367 PacketCloseAdapter(fd);
368 }
369 fd = 0;
370 return false;
371 }
372
373
374 /*
375 * Deinitialization
376 */
377
378 void ether_exit(void)
379 {
380 D(bug("EtherExit\n"));
381
382 // Stop reception thread
383 thread_active = false;
384
385 if(int_ack) ReleaseSemaphore(int_ack,1,NULL);
386 if(int_sig) ReleaseSemaphore(int_sig,1,NULL);
387 if(int_sig2) ReleaseSemaphore(int_sig2,1,NULL);
388 if(int_send_now) ReleaseSemaphore(int_send_now,1,NULL);
389
390 D(bug("CancelIO if needed\n"));
391 if (fd && fd->hFile && pfnCancelIo)
392 pfnCancelIo(fd->hFile);
393
394 // Wait max 2 secs to shut down pending io. After that, kill them.
395 D(bug("Wait delay\n"));
396 for( int i=0; i<10; i++ ) {
397 if(!thread_active_1 && !thread_active_2 && !thread_active_3) break;
398 Sleep(200);
399 }
400
401 if(thread_active_1) {
402 D(bug("Ether killing ether_th1\n"));
403 if(ether_th1) TerminateThread(ether_th1,0);
404 thread_active_1 = false;
405 }
406 if(thread_active_2) {
407 D(bug("Ether killing ether_th2\n"));
408 if(ether_th2) TerminateThread(ether_th2,0);
409 thread_active_2 = false;
410 }
411 if(thread_active_3) {
412 D(bug("Ether killing thread\n"));
413 if(ether_th) TerminateThread(ether_th,0);
414 thread_active_3 = false;
415 }
416
417 ether_th1 = 0;
418 ether_th2 = 0;
419 ether_th = 0;
420
421 D(bug("Closing semaphores\n"));
422 if(int_ack) {
423 CloseHandle(int_ack);
424 int_ack = 0;
425 }
426 if(int_sig) {
427 CloseHandle(int_sig);
428 int_sig = 0;
429 }
430 if(int_sig2) {
431 CloseHandle(int_sig2);
432 int_sig2 = 0;
433 }
434 if(int_send_now) {
435 CloseHandle(int_send_now);
436 int_send_now = 0;
437 }
438
439 // Close ethernet device
440 if(fd) {
441 PacketCloseAdapter(fd);
442 fd = 0;
443 }
444
445 // Remove all protocols
446 D(bug("Removing protocols\n"));
447 NetProtocol *p = prot_list;
448 while (p) {
449 NetProtocol *next = p->next;
450 delete p;
451 p = next;
452 }
453 prot_list = 0;
454
455 D(bug("Deleting sections\n"));
456 DeleteCriticalSection( &fetch_csection );
457 DeleteCriticalSection( &queue_csection );
458 DeleteCriticalSection( &send_csection );
459 DeleteCriticalSection( &wpool_csection );
460
461 D(bug("Freeing read packets\n"));
462 free_read_packets();
463
464 D(bug("Freeing write packets\n"));
465 free_write_packets();
466
467 D(bug("Finalizing queue\n"));
468 final_queue();
469
470 if (net_if_type == NET_IF_ROUTER) {
471 D(bug("Stopping router\n"));
472 router_final();
473 }
474
475 D(bug("EtherExit done\n"));
476 }
477
478
479 /*
480 * Glue around low-level implementation
481 */
482
483 #ifdef SHEEPSHAVER
484 // Error codes
485 enum {
486 eMultiErr = -91,
487 eLenErr = -92,
488 lapProtErr = -94,
489 excessCollsns = -95
490 };
491
492 // Initialize ethernet
493 void EtherInit(void)
494 {
495 net_open = false;
496
497 // Do nothing if the user disabled the network
498 if (PrefsFindBool("nonet"))
499 return;
500
501 net_open = ether_init();
502 }
503
504 // Exit ethernet
505 void EtherExit(void)
506 {
507 ether_exit();
508 net_open = false;
509 }
510
511 // Get ethernet hardware address
512 void AO_get_ethernet_address(uint32 arg)
513 {
514 uint8 *addr = Mac2HostAddr(arg);
515 if (net_open)
516 OTCopy48BitAddress(ether_addr, addr);
517 else {
518 addr[0] = 0x12;
519 addr[1] = 0x34;
520 addr[2] = 0x56;
521 addr[3] = 0x78;
522 addr[4] = 0x9a;
523 addr[5] = 0xbc;
524 }
525 D(bug("AO_get_ethernet_address: got address %02x%02x%02x%02x%02x%02x\n", addr[0], addr[1], addr[2], addr[3], addr[4], addr[5]));
526 }
527
528 // Add multicast address
529 void AO_enable_multicast(uint32 addr)
530 {
531 if (net_open)
532 ether_do_add_multicast(Mac2HostAddr(addr));
533 }
534
535 // Disable multicast address
536 void AO_disable_multicast(uint32 addr)
537 {
538 if (net_open)
539 ether_do_del_multicast(Mac2HostAddr(addr));
540 }
541
542 // Transmit one packet
543 void AO_transmit_packet(uint32 mp)
544 {
545 if (net_open) {
546 switch (ether_do_write(mp)) {
547 case noErr:
548 num_tx_packets++;
549 break;
550 case excessCollsns:
551 num_tx_buffer_full++;
552 break;
553 }
554 }
555 }
556
557 // Copy packet data from message block to linear buffer
558 static inline int ether_arg_to_buffer(uint32 mp, uint8 *p)
559 {
560 return ether_msgb_to_buffer(mp, p);
561 }
562
563 // Ethernet interrupt
564 void EtherIRQ(void)
565 {
566 D(bug("EtherIRQ\n"));
567 num_ether_irq++;
568
569 OTEnterInterrupt();
570 ether_do_interrupt();
571 OTLeaveInterrupt();
572
573 // Acknowledge interrupt to reception thread
574 D(bug(" EtherIRQ done\n"));
575 ReleaseSemaphore(int_ack,1,NULL);
576 }
577 #else
578 // Add multicast address
579 int16 ether_add_multicast(uint32 pb)
580 {
581 return ether_do_add_multicast(Mac2HostAddr(pb + eMultiAddr));
582 }
583
584 // Disable multicast address
585 int16 ether_del_multicast(uint32 pb)
586 {
587 return ether_do_del_multicast(Mac2HostAddr(pb + eMultiAddr));
588 }
589
590 // Transmit one packet
591 int16 ether_write(uint32 wds)
592 {
593 return ether_do_write(wds);
594 }
595
596 // Copy packet data from WDS to linear buffer
597 static inline int ether_arg_to_buffer(uint32 wds, uint8 *p)
598 {
599 return ether_wds_to_buffer(wds, p);
600 }
601
602 // Dispatch packet to protocol handler
603 static void ether_dispatch_packet(uint32 packet, uint32 length)
604 {
605 // Get packet type
606 uint16 type = ReadMacInt16(packet + 12);
607
608 // Look for protocol
609 NetProtocol *prot = find_protocol(type);
610 if (prot == NULL)
611 return;
612
613 // No default handler
614 if (prot->handler == 0)
615 return;
616
617 // Copy header to RHA
618 Mac2Mac_memcpy(ether_data + ed_RHA, packet, 14);
619 D(bug(" header %08lx%04lx %08lx%04lx %04lx\n", ReadMacInt32(ether_data + ed_RHA), ReadMacInt16(ether_data + ed_RHA + 4), ReadMacInt32(ether_data + ed_RHA + 6), ReadMacInt16(ether_data + ed_RHA + 10), ReadMacInt16(ether_data + ed_RHA + 12)));
620
621 // Call protocol handler
622 M68kRegisters r;
623 r.d[0] = type; // Packet type
624 r.d[1] = length - 14; // Remaining packet length (without header, for ReadPacket)
625 r.a[0] = packet + 14; // Pointer to packet (Mac address, for ReadPacket)
626 r.a[3] = ether_data + ed_RHA + 14; // Pointer behind header in RHA
627 r.a[4] = ether_data + ed_ReadPacket; // Pointer to ReadPacket/ReadRest routines
628 D(bug(" calling protocol handler %08lx, type %08lx, length %08lx, data %08lx, rha %08lx, read_packet %08lx\n", prot->handler, r.d[0], r.d[1], r.a[0], r.a[3], r.a[4]));
629 Execute68k(prot->handler, &r);
630 }
631
632 // Ethernet interrupt
633 void EtherInterrupt(void)
634 {
635 D(bug("EtherIRQ\n"));
636 ether_do_interrupt();
637
638 // Acknowledge interrupt to reception thread
639 D(bug(" EtherIRQ done\n"));
640 ReleaseSemaphore(int_ack,1,NULL);
641 }
642 #endif
643
644
645 /*
646 * Reset
647 */
648
649 void ether_reset(void)
650 {
651 D(bug("EtherReset\n"));
652
653 // Remove all protocols
654 NetProtocol *p = prot_list;
655 while (p) {
656 NetProtocol *next = p->next;
657 delete p;
658 p = next;
659 }
660 prot_list = NULL;
661 }
662
663
664 /*
665 * Add multicast address
666 */
667
668 static int16 ether_do_add_multicast(uint8 *addr)
669 {
670 D(bug("ether_add_multicast\n"));
671
672 // We wouldn't need to do this
673 // if(ether_multi_mode != ETHER_MULTICAST_MAC) return noErr;
674
675 switch (net_if_type) {
676 case NET_IF_B2ETHER:
677 if (!PacketAddMulticast( fd, addr)) {
678 D(bug("WARNING: couldn't enable multicast address\n"));
679 return eMultiErr;
680 }
681 default:
682 D(bug("ether_add_multicast: noErr\n"));
683 return noErr;
684 }
685 }
686
687
688 /*
689 * Delete multicast address
690 */
691
692 int16 ether_do_del_multicast(uint8 *addr)
693 {
694 D(bug("ether_del_multicast\n"));
695
696 // We wouldn't need to do this
697 // if(ether_multi_mode != ETHER_MULTICAST_MAC) return noErr;
698
699 switch (net_if_type) {
700 case NET_IF_B2ETHER:
701 if (!PacketDelMulticast( fd, addr)) {
702 D(bug("WARNING: couldn't disable multicast address\n"));
703 return eMultiErr;
704 }
705 default:
706 return noErr;
707 }
708 }
709
710
711 /*
712 * Attach protocol handler
713 */
714
715 int16 ether_attach_ph(uint16 type, uint32 handler)
716 {
717 D(bug("ether_attach_ph type=0x%x, handler=0x%x\n",(int)type,handler));
718
719 // Already attached?
720 NetProtocol *p = find_protocol(type);
721 if (p != NULL) {
722 D(bug("ether_attach_ph: lapProtErr\n"));
723 return lapProtErr;
724 } else {
725 // No, create and attach
726 p = new NetProtocol;
727 p->next = prot_list;
728 p->type = type;
729 p->handler = handler;
730 prot_list = p;
731 D(bug("ether_attach_ph: noErr\n"));
732 return noErr;
733 }
734 }
735
736
737 /*
738 * Detach protocol handler
739 */
740
741 int16 ether_detach_ph(uint16 type)
742 {
743 D(bug("ether_detach_ph type=%08lx\n",(int)type));
744
745 NetProtocol *p = find_protocol(type);
746 if (p != NULL) {
747 NetProtocol *previous = 0;
748 NetProtocol *q = prot_list;
749 while(q) {
750 if (q == p) {
751 if(previous) {
752 previous->next = q->next;
753 } else {
754 prot_list = q->next;
755 }
756 delete p;
757 return noErr;
758 }
759 previous = q;
760 q = q->next;
761 }
762 }
763 return lapProtErr;
764 }
765
766 #if MONITOR
767 static void dump_packet( uint8 *packet, int length )
768 {
769 char buf[1000], sm[10];
770
771 *buf = 0;
772
773 if(length > 256) length = 256;
774
775 for (int i=0; i<length; i++) {
776 sprintf(sm," %02x", (int)packet[i]);
777 strcat( buf, sm );
778 }
779 strcat( buf, "\n" );
780 bug(buf);
781 }
782 #endif
783
784
785 /*
786 * Transmit raw ethernet packet
787 */
788
789 static void insert_send_queue( LPPACKET Packet )
790 {
791 EnterCriticalSection( &send_csection );
792 Packet->next = 0;
793 if(send_queue) {
794 LPPACKET p = send_queue;
795 // The queue is short. It would be larger overhead to double-link it.
796 while(p->next) p = p->next;
797 p->next = Packet;
798 } else {
799 send_queue = Packet;
800 }
801 LeaveCriticalSection( &send_csection );
802 }
803
804 static LPPACKET get_send_head( void )
805 {
806 LPPACKET Packet = 0;
807
808 EnterCriticalSection( &send_csection );
809 if(send_queue) {
810 Packet = send_queue;
811 send_queue = send_queue->next;
812 }
813 LeaveCriticalSection( &send_csection );
814
815 return Packet;
816 }
817
818 static int get_write_packet_pool_sz( void )
819 {
820 LPPACKET t = write_packet_pool;
821 int sz = 0;
822
823 while(t) {
824 t = t->next;
825 sz++;
826 }
827 return(sz);
828 }
829
830 static void free_write_packets( void )
831 {
832 LPPACKET next;
833 int i = 0;
834 while(write_packet_pool) {
835 next = write_packet_pool->next;
836 D(bug("Freeing write packet %ld\n",++i));
837 PacketFreePacket(write_packet_pool);
838 write_packet_pool = next;
839 }
840 }
841
842 void recycle_write_packet( LPPACKET Packet )
843 {
844 EnterCriticalSection( &wpool_csection );
845 Packet->next = write_packet_pool;
846 write_packet_pool = Packet;
847 D(bug("Pool size after recycling = %ld\n",get_write_packet_pool_sz()));
848 LeaveCriticalSection( &wpool_csection );
849 }
850
851 static LPPACKET get_write_packet( UINT len )
852 {
853 LPPACKET Packet = 0;
854
855 EnterCriticalSection( &wpool_csection );
856 if(write_packet_pool) {
857 Packet = write_packet_pool;
858 write_packet_pool = write_packet_pool->next;
859 Packet->OverLapped.Offset = 0;
860 Packet->OverLapped.OffsetHigh = 0;
861 Packet->Length = len;
862 Packet->BytesReceived = 0;
863 Packet->bIoComplete = FALSE;
864 Packet->free = TRUE;
865 Packet->next = 0;
866 // actually an auto-reset event.
867 if(Packet->OverLapped.hEvent) ResetEvent(Packet->OverLapped.hEvent);
868 } else {
869 Packet = PacketAllocatePacket(fd,len);
870 }
871
872 D(bug("Pool size after get wr packet = %ld\n",get_write_packet_pool_sz()));
873
874 LeaveCriticalSection( &wpool_csection );
875
876 return Packet;
877 }
878
879 static unsigned int ether_thread_write_packets(void *arg)
880 {
881 LPPACKET Packet;
882
883 thread_active_1 = true;
884
885 D(bug("ether_thread_write_packets start\n"));
886
887 while(thread_active) {
888 // must be alertable, otherwise write completion is never called
889 WaitForSingleObjectEx(int_send_now,INFINITE,TRUE);
890 while( thread_active && (Packet = get_send_head()) != 0 ) {
891 switch (net_if_type) {
892 case NET_IF_ROUTER:
893 if(router_write_packet((uint8 *)Packet->Buffer, Packet->Length)) {
894 Packet->bIoComplete = TRUE;
895 recycle_write_packet(Packet);
896 }
897 break;
898 case NET_IF_FAKE:
899 Packet->bIoComplete = TRUE;
900 recycle_write_packet(Packet);
901 break;
902 case NET_IF_B2ETHER:
903 if(!PacketSendPacket( fd, Packet, FALSE, TRUE )) {
904 // already recycled if async
905 }
906 break;
907 case NET_IF_SLIRP:
908 slirp_input((uint8 *)Packet->Buffer, Packet->Length);
909 Packet->bIoComplete = TRUE;
910 recycle_write_packet(Packet);
911 break;
912 }
913 }
914 }
915
916 D(bug("ether_thread_write_packets exit\n"));
917
918 thread_active_1 = false;
919
920 return(0);
921 }
922
923 static BOOL write_packet( uint8 *packet, int len )
924 {
925 LPPACKET Packet;
926
927 D(bug("write_packet\n"));
928
929 Packet = get_write_packet(len);
930 if(Packet) {
931 memcpy( Packet->Buffer, packet, len );
932
933 EnterCriticalSection( &fetch_csection );
934 pending_packet_sz[echo_count] = min(sizeof(pending_packet),len);
935 memcpy( pending_packet[echo_count], packet, pending_packet_sz[echo_count] );
936 echo_count = (echo_count+1) & (~(MAX_ECHO-1));
937 LeaveCriticalSection( &fetch_csection );
938
939 insert_send_queue( Packet );
940
941 ReleaseSemaphore(int_send_now,1,NULL);
942 return(TRUE);
943 } else {
944 return(FALSE);
945 }
946 }
947
948 static int16 ether_do_write(uint32 arg)
949 {
950 D(bug("ether_write\n"));
951
952 // Copy packet to buffer
953 uint8 packet[1514], *p = packet;
954 int len = ether_arg_to_buffer(arg, p);
955
956 if(len > 1514) {
957 D(bug("illegal packet length: %d\n",len));
958 return eLenErr;
959 } else {
960 #if MONITOR
961 bug("Sending Ethernet packet (%d bytes):\n",(int)len);
962 dump_packet( packet, len );
963 #endif
964 }
965
966 // Transmit packet
967 if (!write_packet(packet, len)) {
968 D(bug("WARNING: couldn't transmit packet\n"));
969 return excessCollsns;
970 } else {
971 // It's up to the protocol drivers to do the error checking. Even if the
972 // i/o completion routine returns ok, there can be errors, so there is
973 // no point to wait for write completion and try to make some sense of the
974 // possible error codes.
975 return noErr;
976 }
977 }
978
979
980 static void init_queue(void)
981 {
982 queue_inx = 0;
983 queue_head = 0;
984
985 for( int i=0; i<MAX_QUEUE_ITEMS; i++ ) {
986 queue[i].buf = (uint8 *)malloc( 1514 );
987 queue[i].sz = 0;
988 }
989 }
990
991 static void final_queue(void)
992 {
993 for( int i=0; i<MAX_QUEUE_ITEMS; i++ ) {
994 if(queue[i].buf) free(queue[i].buf);
995 }
996 }
997
998 void enqueue_packet( const uint8 *buf, int sz )
999 {
1000 EnterCriticalSection( &queue_csection );
1001 if(queue[queue_inx].sz > 0) {
1002 D(bug("ethernet queue full, packet dropped\n"));
1003 } else {
1004 if(sz > 1514) sz = 1514;
1005 queue[queue_inx].sz = sz;
1006 memcpy( queue[queue_inx].buf, buf, sz );
1007 queue_inx++;
1008 if(queue_inx >= MAX_QUEUE_ITEMS) queue_inx = 0;
1009 if(wait_request) {
1010 wait_request = false;
1011 ReleaseSemaphore(int_sig,1,NULL);
1012 }
1013 }
1014 LeaveCriticalSection( &queue_csection );
1015 }
1016
1017 static int dequeue_packet( uint8 *buf )
1018 {
1019 int sz;
1020
1021 if(!thread_active) return(0);
1022
1023 EnterCriticalSection( &queue_csection );
1024 sz = queue[queue_head].sz;
1025 if(sz > 0) {
1026 memcpy( buf, queue[queue_head].buf, sz );
1027 queue[queue_head].sz = 0;
1028 queue_head++;
1029 if(queue_head >= MAX_QUEUE_ITEMS) queue_head = 0;
1030 }
1031 LeaveCriticalSection( &queue_csection );
1032 return(sz);
1033 }
1034
1035 static void trigger_queue(void)
1036 {
1037 EnterCriticalSection( &queue_csection );
1038 if( queue[queue_head].sz > 0 ) {
1039 D(bug(" packet received, triggering Ethernet interrupt\n"));
1040 SetInterruptFlag(INTFLAG_ETHER);
1041 TriggerInterrupt();
1042 // of course can't wait here.
1043 }
1044 LeaveCriticalSection( &queue_csection );
1045 }
1046
1047 static bool set_wait_request(void)
1048 {
1049 bool result;
1050 EnterCriticalSection( &queue_csection );
1051 if(queue[queue_head].sz) {
1052 result = true;
1053 } else {
1054 result = false;
1055 wait_request = true;
1056 }
1057 LeaveCriticalSection( &queue_csection );
1058 return(result);
1059 }
1060
1061
1062 /*
1063 * SLIRP output buffer glue
1064 */
1065
1066 int slirp_can_output(void)
1067 {
1068 return 1;
1069 }
1070
1071 void slirp_output(const uint8 *packet, int len)
1072 {
1073 enqueue_packet(packet, len);
1074 }
1075
1076 unsigned int slirp_receive_func(void *arg)
1077 {
1078 D(bug("slirp_receive_func\n"));
1079 thread_active_2 = true;
1080
1081 while (thread_active) {
1082 // Wait for packets to arrive
1083 fd_set rfds, wfds, xfds;
1084 int nfds, ret, timeout;
1085
1086 // ... in the output queue
1087 nfds = -1;
1088 FD_ZERO(&rfds);
1089 FD_ZERO(&wfds);
1090 FD_ZERO(&xfds);
1091 timeout = slirp_select_fill(&nfds, &rfds, &wfds, &xfds);
1092 #if ! USE_SLIRP_TIMEOUT
1093 timeout = 10000;
1094 #endif
1095 if (nfds < 0) {
1096 /* Windows does not honour the timeout if there is not
1097 descriptor to wait for */
1098 Delay_usec(timeout);
1099 ret = 0;
1100 }
1101 else {
1102 struct timeval tv;
1103 tv.tv_sec = 0;
1104 tv.tv_usec = timeout;
1105 ret = select(0, &rfds, &wfds, &xfds, &tv);
1106 }
1107 if (ret >= 0)
1108 slirp_select_poll(&rfds, &wfds, &xfds);
1109 }
1110
1111 D(bug("slirp_receive_func exit\n"));
1112 thread_active_2 = false;
1113 return 0;
1114 }
1115
1116
1117 /*
1118 * Packet reception threads
1119 */
1120
1121 VOID CALLBACK packet_read_completion(
1122 DWORD dwErrorCode,
1123 DWORD dwNumberOfBytesTransfered,
1124 LPOVERLAPPED lpOverlapped
1125 )
1126 {
1127 EnterCriticalSection( &fetch_csection );
1128
1129 LPPACKET lpPacket = CONTAINING_RECORD(lpOverlapped,PACKET,OverLapped);
1130
1131 D(bug("packet_read_completion bytes=%d, error code=%d\n",dwNumberOfBytesTransfered,dwErrorCode));
1132
1133 if(thread_active && !dwErrorCode) {
1134 int count = min(dwNumberOfBytesTransfered,1514);
1135 if(count) {
1136 int j = echo_count;
1137 for(int i=MAX_ECHO; i; i--) {
1138 j--;
1139 if(j < 0) j = MAX_ECHO-1;
1140 if(count == pending_packet_sz[j] &&
1141 memcmp(pending_packet[j],lpPacket->Buffer,count) == 0)
1142 {
1143 D(bug("packet_read_completion discarding own packet.\n"));
1144 dwNumberOfBytesTransfered = 0;
1145
1146 j = (j+1) & (~(MAX_ECHO-1));
1147 if(j != echo_count) {
1148 D(bug("Wow, this fix made some good after all...\n"));
1149 }
1150
1151 break;
1152 }
1153 }
1154 if(dwNumberOfBytesTransfered) {
1155 if(net_if_type != NET_IF_ROUTER || !router_read_packet((uint8 *)lpPacket->Buffer, dwNumberOfBytesTransfered)) {
1156 enqueue_packet( (LPBYTE)lpPacket->Buffer, dwNumberOfBytesTransfered );
1157 }
1158 }
1159 }
1160 }
1161
1162 // actually an auto-reset event.
1163 if(lpPacket->OverLapped.hEvent) ResetEvent(lpPacket->OverLapped.hEvent);
1164
1165 lpPacket->free = TRUE;
1166 lpPacket->bIoComplete = TRUE;
1167
1168 if(wait_request2) {
1169 wait_request2 = false;
1170 ReleaseSemaphore(int_sig2,1,NULL);
1171 }
1172
1173 LeaveCriticalSection( &fetch_csection );
1174 }
1175
1176 static BOOL has_no_completed_io(void)
1177 {
1178 BOOL result = TRUE;
1179
1180 EnterCriticalSection( &fetch_csection );
1181
1182 for( int i=0; i<PACKET_POOL_COUNT; i++ ) {
1183 if(packets[i]->bIoComplete) {
1184 result = FALSE;
1185 break;
1186 }
1187 }
1188 if(result) wait_request2 = true;
1189
1190 LeaveCriticalSection( &fetch_csection );
1191 return(result);
1192 }
1193
1194 static bool allocate_read_packets(void)
1195 {
1196 for( int i=0; i<PACKET_POOL_COUNT; i++ ) {
1197 packets[i] = PacketAllocatePacket(fd,1514);
1198 if(!packets[i]) {
1199 D(bug("allocate_read_packets: out of memory\n"));
1200 return(false);
1201 }
1202 }
1203 return(true);
1204 }
1205
1206 static void free_read_packets(void)
1207 {
1208 for( int i=0; i<PACKET_POOL_COUNT; i++ ) {
1209 PacketFreePacket(packets[i]);
1210 }
1211 }
1212
1213 static unsigned int ether_thread_get_packets_nt(void *arg)
1214 {
1215 static uint8 packet[1514];
1216 int i, packet_sz = 0;
1217
1218 thread_active_2 = true;
1219
1220 D(bug("ether_thread_get_packets_nt start\n"));
1221
1222 // Wait for packets to arrive.
1223 // Obey the golden rules; keep the reads pending.
1224 while(thread_active) {
1225
1226 if(net_if_type == NET_IF_B2ETHER) {
1227 D(bug("Pending reads\n"));
1228 for( i=0; thread_active && i<PACKET_POOL_COUNT; i++ ) {
1229 if(packets[i]->free) {
1230 packets[i]->free = FALSE;
1231 if(PacketReceivePacket(fd,packets[i],FALSE)) {
1232 if(packets[i]->bIoComplete) {
1233 D(bug("Early io completion...\n"));
1234 packet_read_completion(
1235 ERROR_SUCCESS,
1236 packets[i]->BytesReceived,
1237 &packets[i]->OverLapped
1238 );
1239 }
1240 } else {
1241 packets[i]->free = TRUE;
1242 }
1243 }
1244 }
1245 }
1246
1247 if(thread_active && has_no_completed_io()) {
1248 D(bug("Waiting for int_sig2\n"));
1249 // "problem": awakens twice in a row. Fix if you increase the pool size.
1250 WaitForSingleObjectEx(int_sig2,INFINITE,TRUE);
1251 }
1252 }
1253
1254 D(bug("ether_thread_get_packets_nt exit\n"));
1255
1256 thread_active_2 = false;
1257
1258 return 0;
1259 }
1260
1261 static unsigned int ether_thread_feed_int(void *arg)
1262 {
1263 bool looping;
1264
1265 thread_active_3 = true;
1266
1267 D(bug("ether_thread_feed_int start\n"));
1268
1269 while(thread_active) {
1270 D(bug("Waiting for int_sig\n"));
1271 WaitForSingleObject(int_sig,INFINITE);
1272 // Looping this way to avoid a race condition.
1273 D(bug("Triggering\n"));
1274 looping = true;
1275 while(thread_active && looping) {
1276 trigger_queue();
1277 // Wait for interrupt acknowledge by EtherInterrupt()
1278 WaitForSingleObject(int_ack,INFINITE);
1279 if(thread_active) looping = set_wait_request();
1280 }
1281 D(bug("Queue empty.\n"));
1282 }
1283
1284 D(bug("ether_thread_feed_int exit\n"));
1285
1286 thread_active_3 = false;
1287
1288 return 0;
1289 }
1290
1291
1292 /*
1293 * Ethernet interrupt - activate deferred tasks to call IODone or protocol handlers
1294 */
1295
1296 static void ether_do_interrupt(void)
1297 {
1298 // Call protocol handler for received packets
1299 EthernetPacket ether_packet;
1300 uint32 packet = ether_packet.addr();
1301 ssize_t length;
1302 for (;;) {
1303
1304 // Read packet from Ethernet device
1305 length = dequeue_packet(Mac2HostAddr(packet));
1306 if (length < 14)
1307 break;
1308
1309 #if MONITOR
1310 bug("Receiving Ethernet packet (%d bytes):\n",(int)length);
1311 dump_packet( Mac2HostAddr(packet), length );
1312 #endif
1313
1314 // Dispatch packet
1315 ether_dispatch_packet(packet, length);
1316 }
1317 }
1318
1319 #if DEBUG
1320 #pragma optimize("",on)
1321 #endif