net.c 42.3 KB
Newer Older
J
Jay Fenlason 已提交
1 2 3 4 5 6 7 8
/*
 * IPv4 over IEEE 1394, per RFC 2734
 *
 * Copyright (C) 2009 Jay Fenlason <fenlason@redhat.com>
 *
 * based on eth1394 by Ben Collins et al
 */

S
Stefan Richter 已提交
9
#include <linux/bug.h>
J
Jay Fenlason 已提交
10 11 12 13 14 15 16
#include <linux/device.h>
#include <linux/ethtool.h>
#include <linux/firewire.h>
#include <linux/firewire-constants.h>
#include <linux/highmem.h>
#include <linux/in.h>
#include <linux/ip.h>
S
Stefan Richter 已提交
17
#include <linux/jiffies.h>
J
Jay Fenlason 已提交
18 19 20
#include <linux/mod_devicetable.h>
#include <linux/module.h>
#include <linux/moduleparam.h>
21
#include <linux/mutex.h>
J
Jay Fenlason 已提交
22 23
#include <linux/netdevice.h>
#include <linux/skbuff.h>
24
#include <linux/spinlock.h>
J
Jay Fenlason 已提交
25 26 27 28

#include <asm/unaligned.h>
#include <net/arp.h>

S
Stefan Richter 已提交
29 30
#define FWNET_MAX_FRAGMENTS	25	/* arbitrary limit */
#define FWNET_ISO_PAGE_COUNT	(PAGE_SIZE < 16 * 1024 ? 4 : 2)
J
Jay Fenlason 已提交
31

S
Stefan Richter 已提交
32 33 34 35
#define IEEE1394_BROADCAST_CHANNEL	31
#define IEEE1394_ALL_NODES		(0xffc0 | 0x003f)
#define IEEE1394_MAX_PAYLOAD_S100	512
#define FWNET_NO_FIFO_ADDR		(~0ULL)
J
Jay Fenlason 已提交
36

S
Stefan Richter 已提交
37 38
#define IANA_SPECIFIER_ID		0x00005eU
#define RFC2734_SW_VERSION		0x000001U
J
Jay Fenlason 已提交
39

S
Stefan Richter 已提交
40
#define IEEE1394_GASP_HDR_SIZE	8
J
Jay Fenlason 已提交
41

S
Stefan Richter 已提交
42 43 44
#define RFC2374_UNFRAG_HDR_SIZE	4
#define RFC2374_FRAG_HDR_SIZE	8
#define RFC2374_FRAG_OVERHEAD	4
J
Jay Fenlason 已提交
45

S
Stefan Richter 已提交
46 47 48 49
#define RFC2374_HDR_UNFRAG	0	/* unfragmented		*/
#define RFC2374_HDR_FIRSTFRAG	1	/* first fragment	*/
#define RFC2374_HDR_LASTFRAG	2	/* last fragment	*/
#define RFC2374_HDR_INTFRAG	3	/* interior fragment	*/
J
Jay Fenlason 已提交
50

S
Stefan Richter 已提交
51
#define RFC2734_HW_ADDR_LEN	16
J
Jay Fenlason 已提交
52

S
Stefan Richter 已提交
53 54 55 56 57 58 59
struct rfc2734_arp {
	__be16 hw_type;		/* 0x0018	*/
	__be16 proto_type;	/* 0x0806       */
	u8 hw_addr_len;		/* 16		*/
	u8 ip_addr_len;		/* 4		*/
	__be16 opcode;		/* ARP Opcode	*/
	/* Above is exactly the same format as struct arphdr */
J
Jay Fenlason 已提交
60

S
Stefan Richter 已提交
61 62 63 64 65 66 67 68
	__be64 s_uniq_id;	/* Sender's 64bit EUI			*/
	u8 max_rec;		/* Sender's max packet size		*/
	u8 sspd;		/* Sender's max speed			*/
	__be16 fifo_hi;		/* hi 16bits of sender's FIFO addr	*/
	__be32 fifo_lo;		/* lo 32bits of sender's FIFO addr	*/
	__be32 sip;		/* Sender's IP Address			*/
	__be32 tip;		/* IP Address of requested hw addr	*/
} __attribute__((packed));
J
Jay Fenlason 已提交
69

S
Stefan Richter 已提交
70 71 72 73 74 75 76
/* This header format is specific to this driver implementation. */
#define FWNET_ALEN	8
#define FWNET_HLEN	10
struct fwnet_header {
	u8 h_dest[FWNET_ALEN];	/* destination address */
	__be16 h_proto;		/* packet type ID field */
} __attribute__((packed));
J
Jay Fenlason 已提交
77

S
Stefan Richter 已提交
78 79
/* IPv4 and IPv6 encapsulation header */
struct rfc2734_header {
J
Jay Fenlason 已提交
80 81 82 83
	u32 w0;
	u32 w1;
};

S
Stefan Richter 已提交
84 85 86 87 88
#define fwnet_get_hdr_lf(h)		(((h)->w0 & 0xc0000000) >> 30)
#define fwnet_get_hdr_ether_type(h)	(((h)->w0 & 0x0000ffff))
#define fwnet_get_hdr_dg_size(h)	(((h)->w0 & 0x0fff0000) >> 16)
#define fwnet_get_hdr_fg_off(h)		(((h)->w0 & 0x00000fff))
#define fwnet_get_hdr_dgl(h)		(((h)->w1 & 0xffff0000) >> 16)
J
Jay Fenlason 已提交
89

S
Stefan Richter 已提交
90 91 92 93
#define fwnet_set_hdr_lf(lf)		((lf)  << 30)
#define fwnet_set_hdr_ether_type(et)	(et)
#define fwnet_set_hdr_dg_size(dgs)	((dgs) << 16)
#define fwnet_set_hdr_fg_off(fgo)	(fgo)
J
Jay Fenlason 已提交
94

S
Stefan Richter 已提交
95
#define fwnet_set_hdr_dgl(dgl)		((dgl) << 16)
J
Jay Fenlason 已提交
96

S
Stefan Richter 已提交
97 98 99 100 101 102
static inline void fwnet_make_uf_hdr(struct rfc2734_header *hdr,
		unsigned ether_type)
{
	hdr->w0 = fwnet_set_hdr_lf(RFC2374_HDR_UNFRAG)
		  | fwnet_set_hdr_ether_type(ether_type);
}
J
Jay Fenlason 已提交
103

S
Stefan Richter 已提交
104 105 106 107 108 109 110 111
static inline void fwnet_make_ff_hdr(struct rfc2734_header *hdr,
		unsigned ether_type, unsigned dg_size, unsigned dgl)
{
	hdr->w0 = fwnet_set_hdr_lf(RFC2374_HDR_FIRSTFRAG)
		  | fwnet_set_hdr_dg_size(dg_size)
		  | fwnet_set_hdr_ether_type(ether_type);
	hdr->w1 = fwnet_set_hdr_dgl(dgl);
}
J
Jay Fenlason 已提交
112

S
Stefan Richter 已提交
113 114 115 116 117 118 119 120
static inline void fwnet_make_sf_hdr(struct rfc2734_header *hdr,
		unsigned lf, unsigned dg_size, unsigned fg_off, unsigned dgl)
{
	hdr->w0 = fwnet_set_hdr_lf(lf)
		  | fwnet_set_hdr_dg_size(dg_size)
		  | fwnet_set_hdr_fg_off(fg_off);
	hdr->w1 = fwnet_set_hdr_dgl(dgl);
}
J
Jay Fenlason 已提交
121 122

/* This list keeps track of what parts of the datagram have been filled in */
S
Stefan Richter 已提交
123 124
struct fwnet_fragment_info {
	struct list_head fi_link;
J
Jay Fenlason 已提交
125 126 127 128
	u16 offset;
	u16 len;
};

S
Stefan Richter 已提交
129 130 131
struct fwnet_partial_datagram {
	struct list_head pd_link;
	struct list_head fi_list;
J
Jay Fenlason 已提交
132 133 134 135 136 137 138 139
	struct sk_buff *skb;
	/* FIXME Why not use skb->data? */
	char *pbuf;
	u16 datagram_label;
	u16 ether_type;
	u16 datagram_size;
};

140 141
static DEFINE_MUTEX(fwnet_device_mutex);
static LIST_HEAD(fwnet_device_list);
J
Jay Fenlason 已提交
142

S
Stefan Richter 已提交
143
struct fwnet_device {
144
	struct list_head dev_link;
J
Jay Fenlason 已提交
145
	spinlock_t lock;
S
Stefan Richter 已提交
146 147 148 149 150
	enum {
		FWNET_BROADCAST_ERROR,
		FWNET_BROADCAST_RUNNING,
		FWNET_BROADCAST_STOPPED,
	} broadcast_state;
J
Jay Fenlason 已提交
151 152 153 154 155 156 157 158 159 160 161 162 163 164 165
	struct fw_iso_context *broadcast_rcv_context;
	struct fw_iso_buffer broadcast_rcv_buffer;
	void **broadcast_rcv_buffer_ptrs;
	unsigned broadcast_rcv_next_ptr;
	unsigned num_broadcast_rcv_ptrs;
	unsigned rcv_buffer_size;
	/*
	 * This value is the maximum unfragmented datagram size that can be
	 * sent by the hardware.  It already has the GASP overhead and the
	 * unfragmented datagram header overhead calculated into it.
	 */
	unsigned broadcast_xmt_max_payload;
	u16 broadcast_xmt_datagramlabel;

	/*
S
Stefan Richter 已提交
166
	 * The CSR address that remote nodes must send datagrams to for us to
J
Jay Fenlason 已提交
167 168 169 170 171 172 173 174 175 176 177 178 179 180 181
	 * receive them.
	 */
	struct fw_address_handler handler;
	u64 local_fifo;

	/* List of packets to be sent */
	struct list_head packet_list;
	/*
	 * List of packets that were broadcasted.  When we get an ISO interrupt
	 * one of them has been sent
	 */
	struct list_head broadcasted_list;
	/* List of packets that have been sent but not yet acked */
	struct list_head sent_list;

182
	struct list_head peer_list;
J
Jay Fenlason 已提交
183
	struct fw_card *card;
184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201
	struct net_device *netdev;
};

struct fwnet_peer {
	struct list_head peer_link;
	struct fwnet_device *dev;
	u64 guid;
	u64 fifo;

	/* guarded by dev->lock */
	struct list_head pd_list; /* received partial datagrams */
	unsigned pdg_size;        /* pd_list size */

	u16 datagram_label;       /* outgoing datagram label */
	unsigned max_payload;     /* includes RFC2374_FRAG_HDR_SIZE overhead */
	int node_id;
	int generation;
	unsigned speed;
J
Jay Fenlason 已提交
202 203 204
};

/* This is our task struct. It's used for the packet complete callback.  */
S
Stefan Richter 已提交
205
struct fwnet_packet_task {
J
Jay Fenlason 已提交
206
	/*
S
Stefan Richter 已提交
207 208
	 * ptask can actually be on dev->packet_list, dev->broadcasted_list,
	 * or dev->sent_list depending on its current state.
J
Jay Fenlason 已提交
209
	 */
S
Stefan Richter 已提交
210
	struct list_head pt_link;
J
Jay Fenlason 已提交
211
	struct fw_transaction transaction;
S
Stefan Richter 已提交
212
	struct rfc2734_header hdr;
J
Jay Fenlason 已提交
213
	struct sk_buff *skb;
S
Stefan Richter 已提交
214 215
	struct fwnet_device *dev;

J
Jay Fenlason 已提交
216 217 218 219 220 221 222 223
	int outstanding_pkts;
	unsigned max_payload;
	u64 fifo_addr;
	u16 dest_node;
	u8 generation;
	u8 speed;
};

S
Stefan Richter 已提交
224 225 226 227 228 229 230 231 232
/*
 * saddr == NULL means use device source address.
 * daddr == NULL means leave destination address (eg unresolved arp).
 */
static int fwnet_header_create(struct sk_buff *skb, struct net_device *net,
			unsigned short type, const void *daddr,
			const void *saddr, unsigned len)
{
	struct fwnet_header *h;
J
Jay Fenlason 已提交
233

S
Stefan Richter 已提交
234 235
	h = (struct fwnet_header *)skb_push(skb, sizeof(*h));
	put_unaligned_be16(type, &h->h_proto);
J
Jay Fenlason 已提交
236

S
Stefan Richter 已提交
237 238
	if (net->flags & (IFF_LOOPBACK | IFF_NOARP)) {
		memset(h->h_dest, 0, net->addr_len);
J
Jay Fenlason 已提交
239

S
Stefan Richter 已提交
240
		return net->hard_header_len;
J
Jay Fenlason 已提交
241 242 243
	}

	if (daddr) {
S
Stefan Richter 已提交
244 245 246
		memcpy(h->h_dest, daddr, net->addr_len);

		return net->hard_header_len;
J
Jay Fenlason 已提交
247 248
	}

S
Stefan Richter 已提交
249
	return -net->hard_header_len;
J
Jay Fenlason 已提交
250 251
}

S
Stefan Richter 已提交
252
static int fwnet_header_rebuild(struct sk_buff *skb)
J
Jay Fenlason 已提交
253
{
S
Stefan Richter 已提交
254
	struct fwnet_header *h = (struct fwnet_header *)skb->data;
J
Jay Fenlason 已提交
255

S
Stefan Richter 已提交
256 257
	if (get_unaligned_be16(&h->h_proto) == ETH_P_IP)
		return arp_find((unsigned char *)&h->h_dest, skb);
J
Jay Fenlason 已提交
258

S
Stefan Richter 已提交
259 260
	fw_notify("%s: unable to resolve type %04x addresses\n",
		  skb->dev->name, be16_to_cpu(h->h_proto));
J
Jay Fenlason 已提交
261 262 263
	return 0;
}

S
Stefan Richter 已提交
264 265 266 267 268
static int fwnet_header_cache(const struct neighbour *neigh,
			      struct hh_cache *hh)
{
	struct net_device *net;
	struct fwnet_header *h;
J
Jay Fenlason 已提交
269

S
Stefan Richter 已提交
270
	if (hh->hh_type == cpu_to_be16(ETH_P_802_3))
J
Jay Fenlason 已提交
271
		return -1;
S
Stefan Richter 已提交
272 273 274 275 276
	net = neigh->dev;
	h = (struct fwnet_header *)((u8 *)hh->hh_data + 16 - sizeof(*h));
	h->h_proto = hh->hh_type;
	memcpy(h->h_dest, neigh->ha, net->addr_len);
	hh->hh_len = FWNET_HLEN;
J
Jay Fenlason 已提交
277 278 279 280 281

	return 0;
}

/* Called by Address Resolution module to notify changes in address. */
S
Stefan Richter 已提交
282 283 284 285
static void fwnet_header_cache_update(struct hh_cache *hh,
		const struct net_device *net, const unsigned char *haddr)
{
	memcpy((u8 *)hh->hh_data + 16 - FWNET_HLEN, haddr, net->addr_len);
J
Jay Fenlason 已提交
286 287
}

S
Stefan Richter 已提交
288 289 290 291 292
static int fwnet_header_parse(const struct sk_buff *skb, unsigned char *haddr)
{
	memcpy(haddr, skb->dev->dev_addr, FWNET_ALEN);

	return FWNET_ALEN;
J
Jay Fenlason 已提交
293 294
}

S
Stefan Richter 已提交
295 296 297 298 299 300
static const struct header_ops fwnet_header_ops = {
	.create         = fwnet_header_create,
	.rebuild        = fwnet_header_rebuild,
	.cache		= fwnet_header_cache,
	.cache_update	= fwnet_header_cache_update,
	.parse          = fwnet_header_parse,
J
Jay Fenlason 已提交
301 302 303
};

/* FIXME: is this correct for all cases? */
S
Stefan Richter 已提交
304 305
static bool fwnet_frag_overlap(struct fwnet_partial_datagram *pd,
			       unsigned offset, unsigned len)
J
Jay Fenlason 已提交
306
{
S
Stefan Richter 已提交
307
	struct fwnet_fragment_info *fi;
J
Jay Fenlason 已提交
308 309
	unsigned end = offset + len;

S
Stefan Richter 已提交
310 311
	list_for_each_entry(fi, &pd->fi_list, fi_link)
		if (offset < fi->offset + fi->len && end > fi->offset)
J
Jay Fenlason 已提交
312
			return true;
S
Stefan Richter 已提交
313

J
Jay Fenlason 已提交
314 315 316 317
	return false;
}

/* Assumes that new fragment does not overlap any existing fragments */
S
Stefan Richter 已提交
318 319 320 321
static struct fwnet_fragment_info *fwnet_frag_new(
	struct fwnet_partial_datagram *pd, unsigned offset, unsigned len)
{
	struct fwnet_fragment_info *fi, *fi2, *new;
J
Jay Fenlason 已提交
322 323
	struct list_head *list;

S
Stefan Richter 已提交
324 325
	list = &pd->fi_list;
	list_for_each_entry(fi, &pd->fi_list, fi_link) {
J
Jay Fenlason 已提交
326 327 328
		if (fi->offset + fi->len == offset) {
			/* The new fragment can be tacked on to the end */
			/* Did the new fragment plug a hole? */
S
Stefan Richter 已提交
329 330
			fi2 = list_entry(fi->fi_link.next,
					 struct fwnet_fragment_info, fi_link);
J
Jay Fenlason 已提交
331 332 333
			if (fi->offset + fi->len == fi2->offset) {
				/* glue fragments together */
				fi->len += len + fi2->len;
S
Stefan Richter 已提交
334
				list_del(&fi2->fi_link);
J
Jay Fenlason 已提交
335 336 337 338
				kfree(fi2);
			} else {
				fi->len += len;
			}
S
Stefan Richter 已提交
339

J
Jay Fenlason 已提交
340 341 342 343 344
			return fi;
		}
		if (offset + len == fi->offset) {
			/* The new fragment can be tacked on to the beginning */
			/* Did the new fragment plug a hole? */
S
Stefan Richter 已提交
345 346
			fi2 = list_entry(fi->fi_link.prev,
					 struct fwnet_fragment_info, fi_link);
J
Jay Fenlason 已提交
347 348 349
			if (fi2->offset + fi2->len == fi->offset) {
				/* glue fragments together */
				fi2->len += fi->len + len;
S
Stefan Richter 已提交
350
				list_del(&fi->fi_link);
J
Jay Fenlason 已提交
351
				kfree(fi);
S
Stefan Richter 已提交
352

J
Jay Fenlason 已提交
353 354 355 356
				return fi2;
			}
			fi->offset = offset;
			fi->len += len;
S
Stefan Richter 已提交
357

J
Jay Fenlason 已提交
358 359 360
			return fi;
		}
		if (offset > fi->offset + fi->len) {
S
Stefan Richter 已提交
361
			list = &fi->fi_link;
J
Jay Fenlason 已提交
362 363 364
			break;
		}
		if (offset + len < fi->offset) {
S
Stefan Richter 已提交
365
			list = fi->fi_link.prev;
J
Jay Fenlason 已提交
366 367 368 369 370 371
			break;
		}
	}

	new = kmalloc(sizeof(*new), GFP_ATOMIC);
	if (!new) {
S
Stefan Richter 已提交
372
		fw_error("out of memory\n");
J
Jay Fenlason 已提交
373 374 375 376 377
		return NULL;
	}

	new->offset = offset;
	new->len = len;
S
Stefan Richter 已提交
378 379
	list_add(&new->fi_link, list);

J
Jay Fenlason 已提交
380 381 382
	return new;
}

S
Stefan Richter 已提交
383 384 385 386 387 388
static struct fwnet_partial_datagram *fwnet_pd_new(struct net_device *net,
		struct fwnet_peer *peer, u16 datagram_label, unsigned dg_size,
		void *frag_buf, unsigned frag_off, unsigned frag_len)
{
	struct fwnet_partial_datagram *new;
	struct fwnet_fragment_info *fi;
J
Jay Fenlason 已提交
389 390 391 392

	new = kmalloc(sizeof(*new), GFP_ATOMIC);
	if (!new)
		goto fail;
S
Stefan Richter 已提交
393 394 395 396

	INIT_LIST_HEAD(&new->fi_list);
	fi = fwnet_frag_new(new, frag_off, frag_len);
	if (fi == NULL)
J
Jay Fenlason 已提交
397
		goto fail_w_new;
S
Stefan Richter 已提交
398

J
Jay Fenlason 已提交
399 400
	new->datagram_label = datagram_label;
	new->datagram_size = dg_size;
S
Stefan Richter 已提交
401 402
	new->skb = dev_alloc_skb(dg_size + net->hard_header_len + 15);
	if (new->skb == NULL)
J
Jay Fenlason 已提交
403
		goto fail_w_fi;
S
Stefan Richter 已提交
404 405

	skb_reserve(new->skb, (net->hard_header_len + 15) & ~15);
J
Jay Fenlason 已提交
406 407
	new->pbuf = skb_put(new->skb, dg_size);
	memcpy(new->pbuf + frag_off, frag_buf, frag_len);
S
Stefan Richter 已提交
408 409
	list_add_tail(&new->pd_link, &peer->pd_list);

J
Jay Fenlason 已提交
410 411 412 413 414 415 416
	return new;

fail_w_fi:
	kfree(fi);
fail_w_new:
	kfree(new);
fail:
S
Stefan Richter 已提交
417 418
	fw_error("out of memory\n");

J
Jay Fenlason 已提交
419 420 421
	return NULL;
}

S
Stefan Richter 已提交
422 423 424 425
static struct fwnet_partial_datagram *fwnet_pd_find(struct fwnet_peer *peer,
						    u16 datagram_label)
{
	struct fwnet_partial_datagram *pd;
J
Jay Fenlason 已提交
426

S
Stefan Richter 已提交
427 428
	list_for_each_entry(pd, &peer->pd_list, pd_link)
		if (pd->datagram_label == datagram_label)
J
Jay Fenlason 已提交
429
			return pd;
S
Stefan Richter 已提交
430

J
Jay Fenlason 已提交
431 432 433 434
	return NULL;
}


S
Stefan Richter 已提交
435 436 437
static void fwnet_pd_delete(struct fwnet_partial_datagram *old)
{
	struct fwnet_fragment_info *fi, *n;
J
Jay Fenlason 已提交
438

S
Stefan Richter 已提交
439
	list_for_each_entry_safe(fi, n, &old->fi_list, fi_link)
J
Jay Fenlason 已提交
440
		kfree(fi);
S
Stefan Richter 已提交
441 442

	list_del(&old->pd_link);
J
Jay Fenlason 已提交
443 444 445 446
	dev_kfree_skb_any(old->skb);
	kfree(old);
}

S
Stefan Richter 已提交
447 448 449 450 451
static bool fwnet_pd_update(struct fwnet_peer *peer,
		struct fwnet_partial_datagram *pd, void *frag_buf,
		unsigned frag_off, unsigned frag_len)
{
	if (fwnet_frag_new(pd, frag_off, frag_len) == NULL)
J
Jay Fenlason 已提交
452
		return false;
S
Stefan Richter 已提交
453

J
Jay Fenlason 已提交
454 455 456 457 458 459
	memcpy(pd->pbuf + frag_off, frag_buf, frag_len);

	/*
	 * Move list entry to beginnig of list so that oldest partial
	 * datagrams percolate to the end of the list
	 */
S
Stefan Richter 已提交
460 461
	list_move_tail(&pd->pd_link, &peer->pd_list);

J
Jay Fenlason 已提交
462 463 464
	return true;
}

S
Stefan Richter 已提交
465 466 467
static bool fwnet_pd_is_complete(struct fwnet_partial_datagram *pd)
{
	struct fwnet_fragment_info *fi;
J
Jay Fenlason 已提交
468

S
Stefan Richter 已提交
469
	fi = list_entry(pd->fi_list.next, struct fwnet_fragment_info, fi_link);
J
Jay Fenlason 已提交
470

S
Stefan Richter 已提交
471
	return fi->len == pd->datagram_size;
J
Jay Fenlason 已提交
472 473
}

474
/* caller must hold dev->lock */
S
Stefan Richter 已提交
475 476 477
static struct fwnet_peer *fwnet_peer_find_by_guid(struct fwnet_device *dev,
						  u64 guid)
{
478
	struct fwnet_peer *peer;
J
Jay Fenlason 已提交
479

480 481 482
	list_for_each_entry(peer, &dev->peer_list, peer_link)
		if (peer->guid == guid)
			return peer;
J
Jay Fenlason 已提交
483

484
	return NULL;
J
Jay Fenlason 已提交
485 486
}

487
/* caller must hold dev->lock */
S
Stefan Richter 已提交
488
static struct fwnet_peer *fwnet_peer_find_by_node_id(struct fwnet_device *dev,
489
						int node_id, int generation)
S
Stefan Richter 已提交
490
{
491
	struct fwnet_peer *peer;
J
Jay Fenlason 已提交
492

493 494 495 496
	list_for_each_entry(peer, &dev->peer_list, peer_link)
		if (peer->node_id    == node_id &&
		    peer->generation == generation)
			return peer;
S
Stefan Richter 已提交
497

498
	return NULL;
J
Jay Fenlason 已提交
499 500
}

501 502
/* See IEEE 1394-2008 table 6-4, table 8-8, table 16-18. */
static unsigned fwnet_max_payload(unsigned max_rec, unsigned speed)
S
Stefan Richter 已提交
503
{
504 505 506 507 508
	max_rec = min(max_rec, speed + 8);
	max_rec = min(max_rec, 0xbU); /* <= 4096 */
	if (max_rec < 8) {
		fw_notify("max_rec %x out of range\n", max_rec);
		max_rec = 8;
J
Jay Fenlason 已提交
509
	}
510 511

	return (1 << (max_rec + 1)) - RFC2374_FRAG_HDR_SIZE;
J
Jay Fenlason 已提交
512 513
}

514

S
Stefan Richter 已提交
515 516 517 518 519 520
static int fwnet_finish_incoming_packet(struct net_device *net,
					struct sk_buff *skb, u16 source_node_id,
					bool is_broadcast, u16 ether_type)
{
	struct fwnet_device *dev;
	static const __be64 broadcast_hw = cpu_to_be64(~0ULL);
J
Jay Fenlason 已提交
521
	int status;
S
Stefan Richter 已提交
522
	__be64 guid;
J
Jay Fenlason 已提交
523

S
Stefan Richter 已提交
524
	dev = netdev_priv(net);
J
Jay Fenlason 已提交
525
	/* Write metadata, and then pass to the receive level */
S
Stefan Richter 已提交
526
	skb->dev = net;
J
Jay Fenlason 已提交
527 528 529 530 531 532 533 534 535 536 537 538 539 540
	skb->ip_summed = CHECKSUM_UNNECESSARY;  /* don't check it */

	/*
	 * Parse the encapsulation header. This actually does the job of
	 * converting to an ethernet frame header, as well as arp
	 * conversion if needed. ARP conversion is easier in this
	 * direction, since we are using ethernet as our backend.
	 */
	/*
	 * If this is an ARP packet, convert it. First, we want to make
	 * use of some of the fields, since they tell us a little bit
	 * about the sending machine.
	 */
	if (ether_type == ETH_P_ARP) {
S
Stefan Richter 已提交
541
		struct rfc2734_arp *arp1394;
J
Jay Fenlason 已提交
542 543 544
		struct arphdr *arp;
		unsigned char *arp_ptr;
		u64 fifo_addr;
S
Stefan Richter 已提交
545
		u64 peer_guid;
546
		unsigned sspd;
J
Jay Fenlason 已提交
547
		u16 max_payload;
S
Stefan Richter 已提交
548
		struct fwnet_peer *peer;
549 550 551 552 553 554 555 556
		unsigned long flags;

		arp1394   = (struct rfc2734_arp *)skb->data;
		arp       = (struct arphdr *)skb->data;
		arp_ptr   = (unsigned char *)(arp + 1);
		peer_guid = get_unaligned_be64(&arp1394->s_uniq_id);
		fifo_addr = (u64)get_unaligned_be16(&arp1394->fifo_hi) << 32
				| get_unaligned_be32(&arp1394->fifo_lo);
J
Jay Fenlason 已提交
557 558

		sspd = arp1394->sspd;
S
Stefan Richter 已提交
559 560 561
		/* Sanity check.  OS X 10.3 PPC reportedly sends 131. */
		if (sspd > SCODE_3200) {
			fw_notify("sspd %x out of range\n", sspd);
562
			sspd = SCODE_3200;
J
Jay Fenlason 已提交
563
		}
564
		max_payload = fwnet_max_payload(arp1394->max_rec, sspd);
J
Jay Fenlason 已提交
565

566
		spin_lock_irqsave(&dev->lock, flags);
S
Stefan Richter 已提交
567
		peer = fwnet_peer_find_by_guid(dev, peer_guid);
568 569 570 571 572 573 574 575 576 577
		if (peer) {
			peer->fifo = fifo_addr;

			if (peer->speed > sspd)
				peer->speed = sspd;
			if (peer->max_payload > max_payload)
				peer->max_payload = max_payload;
		}
		spin_unlock_irqrestore(&dev->lock, flags);

S
Stefan Richter 已提交
578 579 580
		if (!peer) {
			fw_notify("No peer for ARP packet from %016llx\n",
				  (unsigned long long)peer_guid);
J
Jay Fenlason 已提交
581 582
			goto failed_proto;
		}
S
Stefan Richter 已提交
583

J
Jay Fenlason 已提交
584 585 586 587 588 589 590 591 592 593 594 595 596
		/*
		 * Now that we're done with the 1394 specific stuff, we'll
		 * need to alter some of the data.  Believe it or not, all
		 * that needs to be done is sender_IP_address needs to be
		 * moved, the destination hardware address get stuffed
		 * in and the hardware address length set to 8.
		 *
		 * IMPORTANT: The code below overwrites 1394 specific data
		 * needed above so keep the munging of the data for the
		 * higher level IP stack last.
		 */

		arp->ar_hln = 8;
S
Stefan Richter 已提交
597 598 599 600 601 602
		/* skip over sender unique id */
		arp_ptr += arp->ar_hln;
		/* move sender IP addr */
		put_unaligned(arp1394->sip, (u32 *)arp_ptr);
		/* skip over sender IP addr */
		arp_ptr += arp->ar_pln;
J
Jay Fenlason 已提交
603 604 605 606

		if (arp->ar_op == htons(ARPOP_REQUEST))
			memset(arp_ptr, 0, sizeof(u64));
		else
S
Stefan Richter 已提交
607
			memcpy(arp_ptr, net->dev_addr, sizeof(u64));
J
Jay Fenlason 已提交
608 609 610
	}

	/* Now add the ethernet header. */
S
Stefan Richter 已提交
611 612 613 614 615
	guid = cpu_to_be64(dev->card->guid);
	if (dev_hard_header(skb, net, ether_type,
			   is_broadcast ? &broadcast_hw : &guid,
			   NULL, skb->len) >= 0) {
		struct fwnet_header *eth;
J
Jay Fenlason 已提交
616 617 618 619 620
		u16 *rawp;
		__be16 protocol;

		skb_reset_mac_header(skb);
		skb_pull(skb, sizeof(*eth));
S
Stefan Richter 已提交
621
		eth = (struct fwnet_header *)skb_mac_header(skb);
J
Jay Fenlason 已提交
622
		if (*eth->h_dest & 1) {
S
Stefan Richter 已提交
623 624
			if (memcmp(eth->h_dest, net->broadcast,
				   net->addr_len) == 0)
J
Jay Fenlason 已提交
625 626 627 628 629 630
				skb->pkt_type = PACKET_BROADCAST;
#if 0
			else
				skb->pkt_type = PACKET_MULTICAST;
#endif
		} else {
S
Stefan Richter 已提交
631
			if (memcmp(eth->h_dest, net->dev_addr, net->addr_len)) {
J
Jay Fenlason 已提交
632 633
				u64 a1, a2;

S
Stefan Richter 已提交
634 635
				memcpy(&a1, eth->h_dest, sizeof(u64));
				memcpy(&a2, net->dev_addr, sizeof(u64));
J
Jay Fenlason 已提交
636 637 638 639 640 641 642
				skb->pkt_type = PACKET_OTHERHOST;
			}
		}
		if (ntohs(eth->h_proto) >= 1536) {
			protocol = eth->h_proto;
		} else {
			rawp = (u16 *)skb->data;
S
Stefan Richter 已提交
643
			if (*rawp == 0xffff)
J
Jay Fenlason 已提交
644
				protocol = htons(ETH_P_802_3);
S
Stefan Richter 已提交
645
			else
J
Jay Fenlason 已提交
646 647 648 649 650
				protocol = htons(ETH_P_802_2);
		}
		skb->protocol = protocol;
	}
	status = netif_rx(skb);
S
Stefan Richter 已提交
651 652 653
	if (status == NET_RX_DROP) {
		net->stats.rx_errors++;
		net->stats.rx_dropped++;
J
Jay Fenlason 已提交
654
	} else {
S
Stefan Richter 已提交
655 656
		net->stats.rx_packets++;
		net->stats.rx_bytes += skb->len;
J
Jay Fenlason 已提交
657
	}
S
Stefan Richter 已提交
658 659 660
	if (netif_queue_stopped(net))
		netif_wake_queue(net);

J
Jay Fenlason 已提交
661 662 663
	return 0;

 failed_proto:
S
Stefan Richter 已提交
664 665 666
	net->stats.rx_errors++;
	net->stats.rx_dropped++;

J
Jay Fenlason 已提交
667
	dev_kfree_skb_any(skb);
S
Stefan Richter 已提交
668 669 670 671 672
	if (netif_queue_stopped(net))
		netif_wake_queue(net);

	net->last_rx = jiffies;

J
Jay Fenlason 已提交
673 674 675
	return 0;
}

S
Stefan Richter 已提交
676
static int fwnet_incoming_packet(struct fwnet_device *dev, __be32 *buf, int len,
677 678
				 int source_node_id, int generation,
				 bool is_broadcast)
S
Stefan Richter 已提交
679
{
J
Jay Fenlason 已提交
680
	struct sk_buff *skb;
681
	struct net_device *net = dev->netdev;
S
Stefan Richter 已提交
682
	struct rfc2734_header hdr;
J
Jay Fenlason 已提交
683 684
	unsigned lf;
	unsigned long flags;
S
Stefan Richter 已提交
685 686
	struct fwnet_peer *peer;
	struct fwnet_partial_datagram *pd;
J
Jay Fenlason 已提交
687 688 689 690 691 692
	int fg_off;
	int dg_size;
	u16 datagram_label;
	int retval;
	u16 ether_type;

S
Stefan Richter 已提交
693 694 695
	hdr.w0 = be32_to_cpu(buf[0]);
	lf = fwnet_get_hdr_lf(&hdr);
	if (lf == RFC2374_HDR_UNFRAG) {
J
Jay Fenlason 已提交
696 697 698 699 700
		/*
		 * An unfragmented datagram has been received by the ieee1394
		 * bus. Build an skbuff around it so we can pass it to the
		 * high level network layer.
		 */
S
Stefan Richter 已提交
701
		ether_type = fwnet_get_hdr_ether_type(&hdr);
J
Jay Fenlason 已提交
702
		buf++;
S
Stefan Richter 已提交
703
		len -= RFC2374_UNFRAG_HDR_SIZE;
J
Jay Fenlason 已提交
704

S
Stefan Richter 已提交
705
		skb = dev_alloc_skb(len + net->hard_header_len + 15);
J
Jay Fenlason 已提交
706
		if (unlikely(!skb)) {
S
Stefan Richter 已提交
707 708 709
			fw_error("out of memory\n");
			net->stats.rx_dropped++;

J
Jay Fenlason 已提交
710 711
			return -1;
		}
S
Stefan Richter 已提交
712 713 714 715 716
		skb_reserve(skb, (net->hard_header_len + 15) & ~15);
		memcpy(skb_put(skb, len), buf, len);

		return fwnet_finish_incoming_packet(net, skb, source_node_id,
						    is_broadcast, ether_type);
J
Jay Fenlason 已提交
717 718 719
	}
	/* A datagram fragment has been received, now the fun begins. */
	hdr.w1 = ntohl(buf[1]);
S
Stefan Richter 已提交
720 721 722 723
	buf += 2;
	len -= RFC2374_FRAG_HDR_SIZE;
	if (lf == RFC2374_HDR_FIRSTFRAG) {
		ether_type = fwnet_get_hdr_ether_type(&hdr);
J
Jay Fenlason 已提交
724 725
		fg_off = 0;
	} else {
S
Stefan Richter 已提交
726 727
		ether_type = 0;
		fg_off = fwnet_get_hdr_fg_off(&hdr);
J
Jay Fenlason 已提交
728
	}
S
Stefan Richter 已提交
729 730 731
	datagram_label = fwnet_get_hdr_dgl(&hdr);
	dg_size = fwnet_get_hdr_dg_size(&hdr); /* ??? + 1 */

732 733 734 735 736
	spin_lock_irqsave(&dev->lock, flags);

	peer = fwnet_peer_find_by_node_id(dev, source_node_id, generation);
	if (!peer)
		goto bad_proto;
S
Stefan Richter 已提交
737 738

	pd = fwnet_pd_find(peer, datagram_label);
J
Jay Fenlason 已提交
739
	if (pd == NULL) {
S
Stefan Richter 已提交
740
		while (peer->pdg_size >= FWNET_MAX_FRAGMENTS) {
J
Jay Fenlason 已提交
741
			/* remove the oldest */
S
Stefan Richter 已提交
742 743 744
			fwnet_pd_delete(list_first_entry(&peer->pd_list,
				struct fwnet_partial_datagram, pd_link));
			peer->pdg_size--;
J
Jay Fenlason 已提交
745
		}
S
Stefan Richter 已提交
746 747 748
		pd = fwnet_pd_new(net, peer, datagram_label,
				  dg_size, buf, fg_off, len);
		if (pd == NULL) {
J
Jay Fenlason 已提交
749 750 751
			retval = -ENOMEM;
			goto bad_proto;
		}
S
Stefan Richter 已提交
752
		peer->pdg_size++;
J
Jay Fenlason 已提交
753
	} else {
S
Stefan Richter 已提交
754 755
		if (fwnet_frag_overlap(pd, fg_off, len) ||
		    pd->datagram_size != dg_size) {
J
Jay Fenlason 已提交
756 757
			/*
			 * Differing datagram sizes or overlapping fragments,
S
Stefan Richter 已提交
758
			 * discard old datagram and start a new one.
J
Jay Fenlason 已提交
759
			 */
S
Stefan Richter 已提交
760 761 762 763
			fwnet_pd_delete(pd);
			pd = fwnet_pd_new(net, peer, datagram_label,
					  dg_size, buf, fg_off, len);
			if (pd == NULL) {
J
Jay Fenlason 已提交
764
				retval = -ENOMEM;
S
Stefan Richter 已提交
765
				peer->pdg_size--;
J
Jay Fenlason 已提交
766 767 768
				goto bad_proto;
			}
		} else {
S
Stefan Richter 已提交
769
			if (!fwnet_pd_update(peer, pd, buf, fg_off, len)) {
J
Jay Fenlason 已提交
770 771 772 773 774
				/*
				 * Couldn't save off fragment anyway
				 * so might as well obliterate the
				 * datagram now.
				 */
S
Stefan Richter 已提交
775 776
				fwnet_pd_delete(pd);
				peer->pdg_size--;
J
Jay Fenlason 已提交
777 778 779 780 781
				goto bad_proto;
			}
		}
	} /* new datagram or add to existing one */

S
Stefan Richter 已提交
782
	if (lf == RFC2374_HDR_FIRSTFRAG)
J
Jay Fenlason 已提交
783
		pd->ether_type = ether_type;
S
Stefan Richter 已提交
784 785

	if (fwnet_pd_is_complete(pd)) {
J
Jay Fenlason 已提交
786
		ether_type = pd->ether_type;
S
Stefan Richter 已提交
787
		peer->pdg_size--;
J
Jay Fenlason 已提交
788
		skb = skb_get(pd->skb);
S
Stefan Richter 已提交
789 790
		fwnet_pd_delete(pd);

791
		spin_unlock_irqrestore(&dev->lock, flags);
S
Stefan Richter 已提交
792 793 794

		return fwnet_finish_incoming_packet(net, skb, source_node_id,
						    false, ether_type);
J
Jay Fenlason 已提交
795 796 797 798 799
	}
	/*
	 * Datagram is not complete, we're done for the
	 * moment.
	 */
800
	spin_unlock_irqrestore(&dev->lock, flags);
S
Stefan Richter 已提交
801

J
Jay Fenlason 已提交
802 803 804
	return 0;

 bad_proto:
805
	spin_unlock_irqrestore(&dev->lock, flags);
S
Stefan Richter 已提交
806 807 808 809

	if (netif_queue_stopped(net))
		netif_wake_queue(net);

J
Jay Fenlason 已提交
810 811 812
	return 0;
}

S
Stefan Richter 已提交
813 814 815 816 817 818
static void fwnet_receive_packet(struct fw_card *card, struct fw_request *r,
		int tcode, int destination, int source, int generation,
		int speed, unsigned long long offset, void *payload,
		size_t length, void *callback_data)
{
	struct fwnet_device *dev;
J
Jay Fenlason 已提交
819 820
	int status;

S
Stefan Richter 已提交
821 822 823 824 825
	dev = callback_data;
	if (tcode != TCODE_WRITE_BLOCK_REQUEST
	    || destination != card->node_id	/* <- FIXME */
	    || generation != card->generation	/* <- FIXME */
	    || offset != dev->handler.offset) {
J
Jay Fenlason 已提交
826
		fw_send_response(card, r, RCODE_CONFLICT_ERROR);
S
Stefan Richter 已提交
827

J
Jay Fenlason 已提交
828 829
		return;
	}
S
Stefan Richter 已提交
830

831 832
	status = fwnet_incoming_packet(dev, payload, length,
				       source, generation, false);
S
Stefan Richter 已提交
833 834 835 836
	if (status != 0) {
		fw_error("Incoming packet failure\n");
		fw_send_response(card, r, RCODE_CONFLICT_ERROR);

J
Jay Fenlason 已提交
837 838
		return;
	}
S
Stefan Richter 已提交
839 840

	fw_send_response(card, r, RCODE_COMPLETE);
J
Jay Fenlason 已提交
841 842
}

S
Stefan Richter 已提交
843 844 845 846
static void fwnet_receive_broadcast(struct fw_iso_context *context,
		u32 cycle, size_t header_length, void *header, void *data)
{
	struct fwnet_device *dev;
J
Jay Fenlason 已提交
847 848
	struct fw_iso_packet packet;
	struct fw_card *card;
S
Stefan Richter 已提交
849 850
	__be16 *hdr_ptr;
	__be32 *buf_ptr;
J
Jay Fenlason 已提交
851 852 853 854 855 856 857 858
	int retval;
	u32 length;
	u16 source_node_id;
	u32 specifier_id;
	u32 ver;
	unsigned long offset;
	unsigned long flags;

S
Stefan Richter 已提交
859 860
	dev = data;
	card = dev->card;
J
Jay Fenlason 已提交
861
	hdr_ptr = header;
S
Stefan Richter 已提交
862 863 864 865 866 867 868 869 870 871
	length = be16_to_cpup(hdr_ptr);

	spin_lock_irqsave(&dev->lock, flags);

	offset = dev->rcv_buffer_size * dev->broadcast_rcv_next_ptr;
	buf_ptr = dev->broadcast_rcv_buffer_ptrs[dev->broadcast_rcv_next_ptr++];
	if (dev->broadcast_rcv_next_ptr == dev->num_broadcast_rcv_ptrs)
		dev->broadcast_rcv_next_ptr = 0;

	spin_unlock_irqrestore(&dev->lock, flags);
J
Jay Fenlason 已提交
872 873 874

	specifier_id =    (be32_to_cpu(buf_ptr[0]) & 0xffff) << 8
			| (be32_to_cpu(buf_ptr[1]) & 0xff000000) >> 24;
S
Stefan Richter 已提交
875
	ver = be32_to_cpu(buf_ptr[1]) & 0xffffff;
J
Jay Fenlason 已提交
876
	source_node_id = be32_to_cpu(buf_ptr[0]) >> 16;
S
Stefan Richter 已提交
877 878

	if (specifier_id == IANA_SPECIFIER_ID && ver == RFC2734_SW_VERSION) {
J
Jay Fenlason 已提交
879
		buf_ptr += 2;
S
Stefan Richter 已提交
880 881
		length -= IEEE1394_GASP_HDR_SIZE;
		fwnet_incoming_packet(dev, buf_ptr, length,
882
				      source_node_id, -1, true);
S
Stefan Richter 已提交
883 884 885
	}

	packet.payload_length = dev->rcv_buffer_size;
J
Jay Fenlason 已提交
886 887 888 889
	packet.interrupt = 1;
	packet.skip = 0;
	packet.tag = 3;
	packet.sy = 0;
S
Stefan Richter 已提交
890 891 892
	packet.header_length = IEEE1394_GASP_HDR_SIZE;

	spin_lock_irqsave(&dev->lock, flags);
J
Jay Fenlason 已提交
893

S
Stefan Richter 已提交
894 895 896 897 898 899 900
	retval = fw_iso_context_queue(dev->broadcast_rcv_context, &packet,
				      &dev->broadcast_rcv_buffer, offset);

	spin_unlock_irqrestore(&dev->lock, flags);

	if (retval < 0)
		fw_error("requeue failed\n");
J
Jay Fenlason 已提交
901 902
}

S
Stefan Richter 已提交
903 904 905 906 907 908 909
static struct kmem_cache *fwnet_packet_task_cache;

static int fwnet_send_packet(struct fwnet_packet_task *ptask);

static void fwnet_transmit_packet_done(struct fwnet_packet_task *ptask)
{
	struct fwnet_device *dev;
J
Jay Fenlason 已提交
910 911
	unsigned long flags;

S
Stefan Richter 已提交
912 913 914 915 916 917 918 919 920
	dev = ptask->dev;

	spin_lock_irqsave(&dev->lock, flags);
	list_del(&ptask->pt_link);
	spin_unlock_irqrestore(&dev->lock, flags);

	ptask->outstanding_pkts--; /* FIXME access inside lock */

	if (ptask->outstanding_pkts > 0) {
J
Jay Fenlason 已提交
921 922 923 924 925 926 927
		u16 dg_size;
		u16 fg_off;
		u16 datagram_label;
		u16 lf;
		struct sk_buff *skb;

		/* Update the ptask to point to the next fragment and send it */
S
Stefan Richter 已提交
928
		lf = fwnet_get_hdr_lf(&ptask->hdr);
J
Jay Fenlason 已提交
929
		switch (lf) {
S
Stefan Richter 已提交
930 931
		case RFC2374_HDR_LASTFRAG:
		case RFC2374_HDR_UNFRAG:
J
Jay Fenlason 已提交
932
		default:
S
Stefan Richter 已提交
933 934 935
			fw_error("Outstanding packet %x lf %x, header %x,%x\n",
				 ptask->outstanding_pkts, lf, ptask->hdr.w0,
				 ptask->hdr.w1);
J
Jay Fenlason 已提交
936 937
			BUG();

S
Stefan Richter 已提交
938
		case RFC2374_HDR_FIRSTFRAG:
J
Jay Fenlason 已提交
939
			/* Set frag type here for future interior fragments */
S
Stefan Richter 已提交
940 941 942
			dg_size = fwnet_get_hdr_dg_size(&ptask->hdr);
			fg_off = ptask->max_payload - RFC2374_FRAG_HDR_SIZE;
			datagram_label = fwnet_get_hdr_dgl(&ptask->hdr);
J
Jay Fenlason 已提交
943 944
			break;

S
Stefan Richter 已提交
945 946 947 948 949
		case RFC2374_HDR_INTFRAG:
			dg_size = fwnet_get_hdr_dg_size(&ptask->hdr);
			fg_off = fwnet_get_hdr_fg_off(&ptask->hdr)
				  + ptask->max_payload - RFC2374_FRAG_HDR_SIZE;
			datagram_label = fwnet_get_hdr_dgl(&ptask->hdr);
J
Jay Fenlason 已提交
950 951 952
			break;
		}
		skb = ptask->skb;
S
Stefan Richter 已提交
953 954 955 956
		skb_pull(skb, ptask->max_payload);
		if (ptask->outstanding_pkts > 1) {
			fwnet_make_sf_hdr(&ptask->hdr, RFC2374_HDR_INTFRAG,
					  dg_size, fg_off, datagram_label);
J
Jay Fenlason 已提交
957
		} else {
S
Stefan Richter 已提交
958 959 960
			fwnet_make_sf_hdr(&ptask->hdr, RFC2374_HDR_LASTFRAG,
					  dg_size, fg_off, datagram_label);
			ptask->max_payload = skb->len + RFC2374_FRAG_HDR_SIZE;
J
Jay Fenlason 已提交
961
		}
S
Stefan Richter 已提交
962
		fwnet_send_packet(ptask);
J
Jay Fenlason 已提交
963
	} else {
S
Stefan Richter 已提交
964 965
		dev_kfree_skb_any(ptask->skb);
		kmem_cache_free(fwnet_packet_task_cache, ptask);
J
Jay Fenlason 已提交
966 967 968
	}
}

S
Stefan Richter 已提交
969 970 971 972
static void fwnet_write_complete(struct fw_card *card, int rcode,
				 void *payload, size_t length, void *data)
{
	struct fwnet_packet_task *ptask;
J
Jay Fenlason 已提交
973 974 975

	ptask = data;

S
Stefan Richter 已提交
976 977 978 979
	if (rcode == RCODE_COMPLETE)
		fwnet_transmit_packet_done(ptask);
	else
		fw_error("fwnet_write_complete: failed: %x\n", rcode);
J
Jay Fenlason 已提交
980 981 982
		/* ??? error recovery */
}

S
Stefan Richter 已提交
983 984 985
static int fwnet_send_packet(struct fwnet_packet_task *ptask)
{
	struct fwnet_device *dev;
J
Jay Fenlason 已提交
986
	unsigned tx_len;
S
Stefan Richter 已提交
987
	struct rfc2734_header *bufhdr;
J
Jay Fenlason 已提交
988 989
	unsigned long flags;

S
Stefan Richter 已提交
990
	dev = ptask->dev;
J
Jay Fenlason 已提交
991
	tx_len = ptask->max_payload;
S
Stefan Richter 已提交
992 993 994 995 996
	switch (fwnet_get_hdr_lf(&ptask->hdr)) {
	case RFC2374_HDR_UNFRAG:
		bufhdr = (struct rfc2734_header *)
				skb_push(ptask->skb, RFC2374_UNFRAG_HDR_SIZE);
		put_unaligned_be32(ptask->hdr.w0, &bufhdr->w0);
J
Jay Fenlason 已提交
997 998
		break;

S
Stefan Richter 已提交
999 1000 1001 1002 1003 1004 1005
	case RFC2374_HDR_FIRSTFRAG:
	case RFC2374_HDR_INTFRAG:
	case RFC2374_HDR_LASTFRAG:
		bufhdr = (struct rfc2734_header *)
				skb_push(ptask->skb, RFC2374_FRAG_HDR_SIZE);
		put_unaligned_be32(ptask->hdr.w0, &bufhdr->w0);
		put_unaligned_be32(ptask->hdr.w1, &bufhdr->w1);
J
Jay Fenlason 已提交
1006 1007 1008 1009 1010
		break;

	default:
		BUG();
	}
S
Stefan Richter 已提交
1011 1012
	if (ptask->dest_node == IEEE1394_ALL_NODES) {
		u8 *p;
J
Jay Fenlason 已提交
1013
		int generation;
S
Stefan Richter 已提交
1014
		int node_id;
J
Jay Fenlason 已提交
1015 1016

		/* ptask->generation may not have been set yet */
S
Stefan Richter 已提交
1017
		generation = dev->card->generation;
J
Jay Fenlason 已提交
1018
		smp_rmb();
S
Stefan Richter 已提交
1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038
		node_id = dev->card->node_id;

		p = skb_push(ptask->skb, 8);
		put_unaligned_be32(node_id << 16 | IANA_SPECIFIER_ID >> 8, p);
		put_unaligned_be32((IANA_SPECIFIER_ID & 0xff) << 24
						| RFC2734_SW_VERSION, &p[4]);

		/* We should not transmit if broadcast_channel.valid == 0. */
		fw_send_request(dev->card, &ptask->transaction,
				TCODE_STREAM_DATA,
				fw_stream_packet_destination_id(3,
						IEEE1394_BROADCAST_CHANNEL, 0),
				generation, SCODE_100, 0ULL, ptask->skb->data,
				tx_len + 8, fwnet_write_complete, ptask);

		/* FIXME race? */
		spin_lock_irqsave(&dev->lock, flags);
		list_add_tail(&ptask->pt_link, &dev->broadcasted_list);
		spin_unlock_irqrestore(&dev->lock, flags);

J
Jay Fenlason 已提交
1039 1040
		return 0;
	}
S
Stefan Richter 已提交
1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051

	fw_send_request(dev->card, &ptask->transaction,
			TCODE_WRITE_BLOCK_REQUEST, ptask->dest_node,
			ptask->generation, ptask->speed, ptask->fifo_addr,
			ptask->skb->data, tx_len, fwnet_write_complete, ptask);

	/* FIXME race? */
	spin_lock_irqsave(&dev->lock, flags);
	list_add_tail(&ptask->pt_link, &dev->sent_list);
	spin_unlock_irqrestore(&dev->lock, flags);

1052
	dev->netdev->trans_start = jiffies;
S
Stefan Richter 已提交
1053

J
Jay Fenlason 已提交
1054 1055 1056
	return 0;
}

S
Stefan Richter 已提交
1057 1058
static int fwnet_broadcast_start(struct fwnet_device *dev)
{
J
Jay Fenlason 已提交
1059 1060 1061 1062 1063 1064 1065 1066
	struct fw_iso_context *context;
	int retval;
	unsigned num_packets;
	unsigned max_receive;
	struct fw_iso_packet packet;
	unsigned long offset;
	unsigned u;

S
Stefan Richter 已提交
1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079
	if (dev->local_fifo == FWNET_NO_FIFO_ADDR) {
		/* outside OHCI posted write area? */
		static const struct fw_address_region region = {
			.start = 0xffff00000000ULL,
			.end   = CSR_REGISTER_BASE,
		};

		dev->handler.length = 4096;
		dev->handler.address_callback = fwnet_receive_packet;
		dev->handler.callback_data = dev;

		retval = fw_core_add_address_handler(&dev->handler, &region);
		if (retval < 0)
J
Jay Fenlason 已提交
1080
			goto failed_initial;
S
Stefan Richter 已提交
1081 1082

		dev->local_fifo = dev->handler.offset;
J
Jay Fenlason 已提交
1083 1084
	}

S
Stefan Richter 已提交
1085 1086 1087 1088
	max_receive = 1U << (dev->card->max_receive + 1);
	num_packets = (FWNET_ISO_PAGE_COUNT * PAGE_SIZE) / max_receive;

	if (!dev->broadcast_rcv_context) {
J
Jay Fenlason 已提交
1089 1090
		void **ptrptr;

S
Stefan Richter 已提交
1091 1092 1093
		context = fw_iso_context_create(dev->card,
		    FW_ISO_CONTEXT_RECEIVE, IEEE1394_BROADCAST_CHANNEL,
		    dev->card->link_speed, 8, fwnet_receive_broadcast, dev);
J
Jay Fenlason 已提交
1094 1095 1096 1097
		if (IS_ERR(context)) {
			retval = PTR_ERR(context);
			goto failed_context_create;
		}
S
Stefan Richter 已提交
1098 1099 1100 1101

		retval = fw_iso_buffer_init(&dev->broadcast_rcv_buffer,
		    dev->card, FWNET_ISO_PAGE_COUNT, DMA_FROM_DEVICE);
		if (retval < 0)
J
Jay Fenlason 已提交
1102
			goto failed_buffer_init;
S
Stefan Richter 已提交
1103 1104 1105

		ptrptr = kmalloc(sizeof(void *) * num_packets, GFP_KERNEL);
		if (!ptrptr) {
J
Jay Fenlason 已提交
1106 1107 1108
			retval = -ENOMEM;
			goto failed_ptrs_alloc;
		}
S
Stefan Richter 已提交
1109 1110 1111

		dev->broadcast_rcv_buffer_ptrs = ptrptr;
		for (u = 0; u < FWNET_ISO_PAGE_COUNT; u++) {
J
Jay Fenlason 已提交
1112 1113 1114
			void *ptr;
			unsigned v;

S
Stefan Richter 已提交
1115 1116 1117 1118
			ptr = kmap(dev->broadcast_rcv_buffer.pages[u]);
			for (v = 0; v < num_packets / FWNET_ISO_PAGE_COUNT; v++)
				*ptrptr++ = (void *)
						((char *)ptr + v * max_receive);
J
Jay Fenlason 已提交
1119
		}
S
Stefan Richter 已提交
1120 1121 1122 1123
		dev->broadcast_rcv_context = context;
	} else {
		context = dev->broadcast_rcv_context;
	}
J
Jay Fenlason 已提交
1124 1125 1126 1127 1128 1129

	packet.payload_length = max_receive;
	packet.interrupt = 1;
	packet.skip = 0;
	packet.tag = 3;
	packet.sy = 0;
S
Stefan Richter 已提交
1130
	packet.header_length = IEEE1394_GASP_HDR_SIZE;
J
Jay Fenlason 已提交
1131
	offset = 0;
S
Stefan Richter 已提交
1132 1133 1134 1135 1136

	for (u = 0; u < num_packets; u++) {
		retval = fw_iso_context_queue(context, &packet,
				&dev->broadcast_rcv_buffer, offset);
		if (retval < 0)
J
Jay Fenlason 已提交
1137
			goto failed_rcv_queue;
S
Stefan Richter 已提交
1138

J
Jay Fenlason 已提交
1139 1140
		offset += max_receive;
	}
S
Stefan Richter 已提交
1141 1142 1143 1144 1145 1146
	dev->num_broadcast_rcv_ptrs = num_packets;
	dev->rcv_buffer_size = max_receive;
	dev->broadcast_rcv_next_ptr = 0U;
	retval = fw_iso_context_start(context, -1, 0,
			FW_ISO_CONTEXT_MATCH_ALL_TAGS); /* ??? sync */
	if (retval < 0)
J
Jay Fenlason 已提交
1147
		goto failed_rcv_queue;
S
Stefan Richter 已提交
1148 1149 1150 1151 1152 1153

	/* FIXME: adjust it according to the min. speed of all known peers? */
	dev->broadcast_xmt_max_payload = IEEE1394_MAX_PAYLOAD_S100
			- IEEE1394_GASP_HDR_SIZE - RFC2374_UNFRAG_HDR_SIZE;
	dev->broadcast_state = FWNET_BROADCAST_RUNNING;

J
Jay Fenlason 已提交
1154 1155 1156
	return 0;

 failed_rcv_queue:
S
Stefan Richter 已提交
1157 1158
	kfree(dev->broadcast_rcv_buffer_ptrs);
	dev->broadcast_rcv_buffer_ptrs = NULL;
J
Jay Fenlason 已提交
1159
 failed_ptrs_alloc:
S
Stefan Richter 已提交
1160
	fw_iso_buffer_destroy(&dev->broadcast_rcv_buffer, dev->card);
J
Jay Fenlason 已提交
1161
 failed_buffer_init:
S
Stefan Richter 已提交
1162 1163
	fw_iso_context_destroy(context);
	dev->broadcast_rcv_context = NULL;
J
Jay Fenlason 已提交
1164
 failed_context_create:
S
Stefan Richter 已提交
1165
	fw_core_remove_address_handler(&dev->handler);
J
Jay Fenlason 已提交
1166
 failed_initial:
S
Stefan Richter 已提交
1167 1168
	dev->local_fifo = FWNET_NO_FIFO_ADDR;

J
Jay Fenlason 已提交
1169 1170 1171
	return retval;
}

S
Stefan Richter 已提交
1172 1173 1174 1175
/* ifup */
static int fwnet_open(struct net_device *net)
{
	struct fwnet_device *dev = netdev_priv(net);
J
Jay Fenlason 已提交
1176 1177
	int ret;

S
Stefan Richter 已提交
1178 1179
	if (dev->broadcast_state == FWNET_BROADCAST_ERROR) {
		ret = fwnet_broadcast_start(dev);
J
Jay Fenlason 已提交
1180 1181 1182
		if (ret)
			return ret;
	}
S
Stefan Richter 已提交
1183 1184
	netif_start_queue(net);

J
Jay Fenlason 已提交
1185 1186 1187
	return 0;
}

S
Stefan Richter 已提交
1188 1189
/* ifdown */
static int fwnet_stop(struct net_device *net)
J
Jay Fenlason 已提交
1190
{
S
Stefan Richter 已提交
1191 1192 1193
	netif_stop_queue(net);

	/* Deallocate iso context for use by other applications? */
J
Jay Fenlason 已提交
1194 1195 1196 1197

	return 0;
}

S
Stefan Richter 已提交
1198
static int fwnet_tx(struct sk_buff *skb, struct net_device *net)
J
Jay Fenlason 已提交
1199
{
S
Stefan Richter 已提交
1200 1201
	struct fwnet_header hdr_buf;
	struct fwnet_device *dev = netdev_priv(net);
J
Jay Fenlason 已提交
1202 1203 1204 1205 1206
	__be16 proto;
	u16 dest_node;
	unsigned max_payload;
	u16 dg_size;
	u16 *datagram_label_ptr;
S
Stefan Richter 已提交
1207
	struct fwnet_packet_task *ptask;
1208 1209
	struct fwnet_peer *peer;
	unsigned long flags;
J
Jay Fenlason 已提交
1210

S
Stefan Richter 已提交
1211
	ptask = kmem_cache_alloc(fwnet_packet_task_cache, GFP_ATOMIC);
J
Jay Fenlason 已提交
1212 1213 1214 1215 1216 1217 1218 1219
	if (ptask == NULL)
		goto fail;

	skb = skb_share_check(skb, GFP_ATOMIC);
	if (!skb)
		goto fail;

	/*
S
Stefan Richter 已提交
1220
	 * Make a copy of the driver-specific header.
J
Jay Fenlason 已提交
1221 1222 1223 1224 1225 1226 1227 1228
	 * We might need to rebuild the header on tx failure.
	 */
	memcpy(&hdr_buf, skb->data, sizeof(hdr_buf));
	skb_pull(skb, sizeof(hdr_buf));

	proto = hdr_buf.h_proto;
	dg_size = skb->len;

1229 1230 1231
	/* serialize access to peer, including peer->datagram_label */
	spin_lock_irqsave(&dev->lock, flags);

J
Jay Fenlason 已提交
1232 1233 1234 1235
	/*
	 * Set the transmission type for the packet.  ARP packets and IP
	 * broadcast packets are sent via GASP.
	 */
S
Stefan Richter 已提交
1236
	if (memcmp(hdr_buf.h_dest, net->broadcast, FWNET_ALEN) == 0
J
Jay Fenlason 已提交
1237
	    || proto == htons(ETH_P_ARP)
S
Stefan Richter 已提交
1238 1239
	    || (proto == htons(ETH_P_IP)
		&& IN_MULTICAST(ntohl(ip_hdr(skb)->daddr)))) {
1240
		max_payload        = dev->broadcast_xmt_max_payload;
S
Stefan Richter 已提交
1241 1242
		datagram_label_ptr = &dev->broadcast_xmt_datagramlabel;

1243 1244 1245 1246
		ptask->fifo_addr   = FWNET_NO_FIFO_ADDR;
		ptask->generation  = 0;
		ptask->dest_node   = IEEE1394_ALL_NODES;
		ptask->speed       = SCODE_100;
J
Jay Fenlason 已提交
1247
	} else {
S
Stefan Richter 已提交
1248
		__be64 guid = get_unaligned((__be64 *)hdr_buf.h_dest);
J
Jay Fenlason 已提交
1249 1250
		u8 generation;

S
Stefan Richter 已提交
1251
		peer = fwnet_peer_find_by_guid(dev, be64_to_cpu(guid));
1252 1253
		if (!peer || peer->fifo == FWNET_NO_FIFO_ADDR)
			goto fail_unlock;
J
Jay Fenlason 已提交
1254

1255 1256 1257
		generation         = peer->generation;
		dest_node          = peer->node_id;
		max_payload        = peer->max_payload;
S
Stefan Richter 已提交
1258
		datagram_label_ptr = &peer->datagram_label;
J
Jay Fenlason 已提交
1259

1260 1261 1262 1263
		ptask->fifo_addr   = peer->fifo;
		ptask->generation  = generation;
		ptask->dest_node   = dest_node;
		ptask->speed       = peer->speed;
J
Jay Fenlason 已提交
1264 1265 1266 1267 1268 1269
	}

	/* If this is an ARP packet, convert it */
	if (proto == htons(ETH_P_ARP)) {
		struct arphdr *arp = (struct arphdr *)skb->data;
		unsigned char *arp_ptr = (unsigned char *)(arp + 1);
S
Stefan Richter 已提交
1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283
		struct rfc2734_arp *arp1394 = (struct rfc2734_arp *)skb->data;
		__be32 ipaddr;

		ipaddr = get_unaligned((__be32 *)(arp_ptr + FWNET_ALEN));

		arp1394->hw_addr_len    = RFC2734_HW_ADDR_LEN;
		arp1394->max_rec        = dev->card->max_receive;
		arp1394->sspd		= dev->card->link_speed;

		put_unaligned_be16(dev->local_fifo >> 32,
				   &arp1394->fifo_hi);
		put_unaligned_be32(dev->local_fifo & 0xffffffff,
				   &arp1394->fifo_lo);
		put_unaligned(ipaddr, &arp1394->sip);
J
Jay Fenlason 已提交
1284 1285 1286 1287 1288
	}

	ptask->hdr.w0 = 0;
	ptask->hdr.w1 = 0;
	ptask->skb = skb;
S
Stefan Richter 已提交
1289 1290
	ptask->dev = dev;

J
Jay Fenlason 已提交
1291
	/* Does it all fit in one packet? */
S
Stefan Richter 已提交
1292 1293
	if (dg_size <= max_payload) {
		fwnet_make_uf_hdr(&ptask->hdr, ntohs(proto));
J
Jay Fenlason 已提交
1294
		ptask->outstanding_pkts = 1;
S
Stefan Richter 已提交
1295
		max_payload = dg_size + RFC2374_UNFRAG_HDR_SIZE;
J
Jay Fenlason 已提交
1296 1297 1298
	} else {
		u16 datagram_label;

S
Stefan Richter 已提交
1299
		max_payload -= RFC2374_FRAG_OVERHEAD;
J
Jay Fenlason 已提交
1300
		datagram_label = (*datagram_label_ptr)++;
S
Stefan Richter 已提交
1301 1302
		fwnet_make_ff_hdr(&ptask->hdr, ntohs(proto), dg_size,
				  datagram_label);
J
Jay Fenlason 已提交
1303
		ptask->outstanding_pkts = DIV_ROUND_UP(dg_size, max_payload);
S
Stefan Richter 已提交
1304
		max_payload += RFC2374_FRAG_HDR_SIZE;
J
Jay Fenlason 已提交
1305
	}
1306 1307 1308

	spin_unlock_irqrestore(&dev->lock, flags);

J
Jay Fenlason 已提交
1309
	ptask->max_payload = max_payload;
S
Stefan Richter 已提交
1310 1311
	fwnet_send_packet(ptask);

J
Jay Fenlason 已提交
1312 1313
	return NETDEV_TX_OK;

1314 1315
 fail_unlock:
	spin_unlock_irqrestore(&dev->lock, flags);
J
Jay Fenlason 已提交
1316 1317
 fail:
	if (ptask)
S
Stefan Richter 已提交
1318
		kmem_cache_free(fwnet_packet_task_cache, ptask);
J
Jay Fenlason 已提交
1319 1320 1321 1322

	if (skb != NULL)
		dev_kfree_skb(skb);

S
Stefan Richter 已提交
1323 1324
	net->stats.tx_dropped++;
	net->stats.tx_errors++;
J
Jay Fenlason 已提交
1325 1326 1327 1328 1329 1330 1331 1332 1333 1334 1335

	/*
	 * FIXME: According to a patch from 2003-02-26, "returning non-zero
	 * causes serious problems" here, allegedly.  Before that patch,
	 * -ERRNO was returned which is not appropriate under Linux 2.6.
	 * Perhaps more needs to be done?  Stop the queue in serious
	 * conditions and restart it elsewhere?
	 */
	return NETDEV_TX_OK;
}

S
Stefan Richter 已提交
1336 1337 1338
static void fwnet_tx_timeout(struct net_device *net)
{
	fw_error("%s: timeout\n", net->name);
J
Jay Fenlason 已提交
1339

S
Stefan Richter 已提交
1340
	/* FIXME: What to do if we timeout? */
J
Jay Fenlason 已提交
1341 1342
}

S
Stefan Richter 已提交
1343 1344
static int fwnet_change_mtu(struct net_device *net, int new_mtu)
{
J
Jay Fenlason 已提交
1345 1346 1347
	if (new_mtu < 68)
		return -EINVAL;

S
Stefan Richter 已提交
1348
	net->mtu = new_mtu;
J
Jay Fenlason 已提交
1349 1350 1351
	return 0;
}

S
Stefan Richter 已提交
1352 1353 1354 1355 1356
static void fwnet_get_drvinfo(struct net_device *net,
			      struct ethtool_drvinfo *info)
{
	strcpy(info->driver, KBUILD_MODNAME);
	strcpy(info->bus_info, "ieee1394");
J
Jay Fenlason 已提交
1357 1358
}

S
Stefan Richter 已提交
1359 1360
static struct ethtool_ops fwnet_ethtool_ops = {
	.get_drvinfo = fwnet_get_drvinfo,
J
Jay Fenlason 已提交
1361 1362
};

S
Stefan Richter 已提交
1363 1364 1365 1366 1367 1368
static const struct net_device_ops fwnet_netdev_ops = {
	.ndo_open       = fwnet_open,
	.ndo_stop	= fwnet_stop,
	.ndo_start_xmit = fwnet_tx,
	.ndo_tx_timeout = fwnet_tx_timeout,
	.ndo_change_mtu = fwnet_change_mtu,
J
Jay Fenlason 已提交
1369 1370
};

S
Stefan Richter 已提交
1371 1372 1373 1374 1375 1376 1377 1378 1379 1380 1381 1382
static void fwnet_init_dev(struct net_device *net)
{
	net->header_ops		= &fwnet_header_ops;
	net->netdev_ops		= &fwnet_netdev_ops;
	net->watchdog_timeo	= 100000; /* ? FIXME */
	net->flags		= IFF_BROADCAST | IFF_MULTICAST;
	net->features		= NETIF_F_HIGHDMA;
	net->addr_len		= FWNET_ALEN;
	net->hard_header_len	= FWNET_HLEN;
	net->type		= ARPHRD_IEEE1394;
	net->tx_queue_len	= 1000; /* ? FIXME */
	SET_ETHTOOL_OPS(net, &fwnet_ethtool_ops);
J
Jay Fenlason 已提交
1383 1384
}

1385 1386 1387 1388 1389 1390 1391 1392 1393 1394 1395 1396 1397 1398 1399 1400 1401 1402 1403 1404 1405 1406 1407 1408 1409 1410 1411 1412 1413 1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425 1426
/* caller must hold fwnet_device_mutex */
static struct fwnet_device *fwnet_dev_find(struct fw_card *card)
{
	struct fwnet_device *dev;

	list_for_each_entry(dev, &fwnet_device_list, dev_link)
		if (dev->card == card)
			return dev;

	return NULL;
}

static int fwnet_add_peer(struct fwnet_device *dev,
			  struct fw_unit *unit, struct fw_device *device)
{
	struct fwnet_peer *peer;

	peer = kmalloc(sizeof(*peer), GFP_KERNEL);
	if (!peer)
		return -ENOMEM;

	unit->device.driver_data = peer;
	peer->dev = dev;
	peer->guid = (u64)device->config_rom[3] << 32 | device->config_rom[4];
	peer->fifo = FWNET_NO_FIFO_ADDR;
	INIT_LIST_HEAD(&peer->pd_list);
	peer->pdg_size = 0;
	peer->datagram_label = 0;
	peer->speed = device->max_speed;
	peer->max_payload = fwnet_max_payload(device->max_rec, peer->speed);

	peer->generation = device->generation;
	smp_rmb();
	peer->node_id = device->node_id;

	spin_lock_irq(&dev->lock);
	list_add_tail(&peer->peer_link, &dev->peer_list);
	spin_unlock_irq(&dev->lock);

	return 0;
}

S
Stefan Richter 已提交
1427 1428 1429 1430 1431 1432 1433
static int fwnet_probe(struct device *_dev)
{
	struct fw_unit *unit = fw_unit(_dev);
	struct fw_device *device = fw_parent_device(unit);
	struct fw_card *card = device->card;
	struct net_device *net;
	struct fwnet_device *dev;
J
Jay Fenlason 已提交
1434
	unsigned max_mtu;
1435 1436
	bool new_netdev;
	int ret;
J
Jay Fenlason 已提交
1437

1438
	mutex_lock(&fwnet_device_mutex);
J
Jay Fenlason 已提交
1439

1440 1441 1442 1443 1444
	dev = fwnet_dev_find(card);
	if (dev) {
		new_netdev = false;
		net = dev->netdev;
		goto have_dev;
J
Jay Fenlason 已提交
1445
	}
1446 1447

	new_netdev = true;
S
Stefan Richter 已提交
1448 1449
	net = alloc_netdev(sizeof(*dev), "firewire%d", fwnet_init_dev);
	if (net == NULL) {
1450
		ret = -ENOMEM;
J
Jay Fenlason 已提交
1451 1452 1453
		goto out;
	}

S
Stefan Richter 已提交
1454 1455
	SET_NETDEV_DEV(net, card->device);
	dev = netdev_priv(net);
J
Jay Fenlason 已提交
1456

S
Stefan Richter 已提交
1457 1458 1459 1460 1461
	spin_lock_init(&dev->lock);
	dev->broadcast_state = FWNET_BROADCAST_ERROR;
	dev->broadcast_rcv_context = NULL;
	dev->broadcast_xmt_max_payload = 0;
	dev->broadcast_xmt_datagramlabel = 0;
J
Jay Fenlason 已提交
1462

S
Stefan Richter 已提交
1463
	dev->local_fifo = FWNET_NO_FIFO_ADDR;
J
Jay Fenlason 已提交
1464

S
Stefan Richter 已提交
1465 1466 1467
	INIT_LIST_HEAD(&dev->packet_list);
	INIT_LIST_HEAD(&dev->broadcasted_list);
	INIT_LIST_HEAD(&dev->sent_list);
1468
	INIT_LIST_HEAD(&dev->peer_list);
J
Jay Fenlason 已提交
1469

S
Stefan Richter 已提交
1470
	dev->card = card;
1471
	dev->netdev = net;
J
Jay Fenlason 已提交
1472 1473 1474 1475 1476 1477

	/*
	 * Use the RFC 2734 default 1500 octets or the maximum payload
	 * as initial MTU
	 */
	max_mtu = (1 << (card->max_receive + 1))
S
Stefan Richter 已提交
1478 1479
		  - sizeof(struct rfc2734_header) - IEEE1394_GASP_HDR_SIZE;
	net->mtu = min(1500U, max_mtu);
J
Jay Fenlason 已提交
1480 1481

	/* Set our hardware address while we're at it */
S
Stefan Richter 已提交
1482 1483
	put_unaligned_be64(card->guid, net->dev_addr);
	put_unaligned_be64(~0ULL, net->broadcast);
1484 1485
	ret = register_netdev(net);
	if (ret) {
S
Stefan Richter 已提交
1486
		fw_error("Cannot register the driver\n");
J
Jay Fenlason 已提交
1487 1488 1489
		goto out;
	}

1490
	list_add_tail(&dev->dev_link, &fwnet_device_list);
S
Stefan Richter 已提交
1491 1492
	fw_notify("%s: IPv4 over FireWire on device %016llx\n",
		  net->name, (unsigned long long)card->guid);
1493 1494 1495 1496 1497 1498
 have_dev:
	ret = fwnet_add_peer(dev, unit, device);
	if (ret && new_netdev) {
		unregister_netdev(net);
		list_del(&dev->dev_link);
	}
J
Jay Fenlason 已提交
1499
 out:
1500
	if (ret && new_netdev)
S
Stefan Richter 已提交
1501 1502
		free_netdev(net);

1503 1504 1505 1506 1507 1508 1509 1510 1511 1512 1513 1514 1515 1516 1517 1518 1519
	mutex_unlock(&fwnet_device_mutex);

	return ret;
}

static void fwnet_remove_peer(struct fwnet_peer *peer)
{
	struct fwnet_partial_datagram *pd, *pd_next;

	spin_lock_irq(&peer->dev->lock);
	list_del(&peer->peer_link);
	spin_unlock_irq(&peer->dev->lock);

	list_for_each_entry_safe(pd, pd_next, &peer->pd_list, pd_link)
		fwnet_pd_delete(pd);

	kfree(peer);
J
Jay Fenlason 已提交
1520 1521
}

S
Stefan Richter 已提交
1522 1523
static int fwnet_remove(struct device *_dev)
{
1524 1525
	struct fwnet_peer *peer = _dev->driver_data;
	struct fwnet_device *dev = peer->dev;
S
Stefan Richter 已提交
1526 1527 1528
	struct net_device *net;
	struct fwnet_packet_task *ptask, *pt_next;

1529
	mutex_lock(&fwnet_device_mutex);
J
Jay Fenlason 已提交
1530

1531
	fwnet_remove_peer(peer);
S
Stefan Richter 已提交
1532

1533 1534
	if (list_empty(&dev->peer_list)) {
		net = dev->netdev;
S
Stefan Richter 已提交
1535 1536 1537 1538 1539 1540 1541 1542 1543
		unregister_netdev(net);

		if (dev->local_fifo != FWNET_NO_FIFO_ADDR)
			fw_core_remove_address_handler(&dev->handler);
		if (dev->broadcast_rcv_context) {
			fw_iso_context_stop(dev->broadcast_rcv_context);
			fw_iso_buffer_destroy(&dev->broadcast_rcv_buffer,
					      dev->card);
			fw_iso_context_destroy(dev->broadcast_rcv_context);
J
Jay Fenlason 已提交
1544
		}
S
Stefan Richter 已提交
1545 1546 1547 1548
		list_for_each_entry_safe(ptask, pt_next,
					 &dev->packet_list, pt_link) {
			dev_kfree_skb_any(ptask->skb);
			kmem_cache_free(fwnet_packet_task_cache, ptask);
J
Jay Fenlason 已提交
1549
		}
S
Stefan Richter 已提交
1550 1551 1552 1553
		list_for_each_entry_safe(ptask, pt_next,
					 &dev->broadcasted_list, pt_link) {
			dev_kfree_skb_any(ptask->skb);
			kmem_cache_free(fwnet_packet_task_cache, ptask);
J
Jay Fenlason 已提交
1554
		}
S
Stefan Richter 已提交
1555 1556 1557 1558
		list_for_each_entry_safe(ptask, pt_next,
					 &dev->sent_list, pt_link) {
			dev_kfree_skb_any(ptask->skb);
			kmem_cache_free(fwnet_packet_task_cache, ptask);
J
Jay Fenlason 已提交
1559
		}
S
Stefan Richter 已提交
1560
		free_netdev(net);
J
Jay Fenlason 已提交
1561
	}
S
Stefan Richter 已提交
1562

1563 1564
	mutex_unlock(&fwnet_device_mutex);

J
Jay Fenlason 已提交
1565 1566 1567
	return 0;
}

S
Stefan Richter 已提交
1568 1569 1570 1571 1572 1573 1574
/*
 * FIXME abort partially sent fragmented datagrams,
 * discard partially received fragmented datagrams
 */
static void fwnet_update(struct fw_unit *unit)
{
	struct fw_device *device = fw_parent_device(unit);
1575 1576
	struct fwnet_peer *peer = unit->device.driver_data;
	int generation;
J
Jay Fenlason 已提交
1577

1578 1579 1580 1581 1582 1583
	generation = device->generation;

	spin_lock_irq(&peer->dev->lock);
	peer->node_id    = device->node_id;
	peer->generation = generation;
	spin_unlock_irq(&peer->dev->lock);
J
Jay Fenlason 已提交
1584 1585
}

S
Stefan Richter 已提交
1586 1587 1588 1589 1590 1591 1592 1593 1594 1595 1596
static const struct ieee1394_device_id fwnet_id_table[] = {
	{
		.match_flags  = IEEE1394_MATCH_SPECIFIER_ID |
				IEEE1394_MATCH_VERSION,
		.specifier_id = IANA_SPECIFIER_ID,
		.version      = RFC2734_SW_VERSION,
	},
	{ }
};

static struct fw_driver fwnet_driver = {
J
Jay Fenlason 已提交
1597
	.driver = {
S
Stefan Richter 已提交
1598 1599 1600 1601 1602
		.owner  = THIS_MODULE,
		.name   = "net",
		.bus    = &fw_bus_type,
		.probe  = fwnet_probe,
		.remove = fwnet_remove,
J
Jay Fenlason 已提交
1603
	},
S
Stefan Richter 已提交
1604 1605 1606 1607 1608 1609 1610 1611 1612 1613 1614 1615 1616 1617 1618 1619 1620 1621 1622 1623 1624 1625 1626 1627
	.update   = fwnet_update,
	.id_table = fwnet_id_table,
};

static const u32 rfc2374_unit_directory_data[] = {
	0x00040000,	/* directory_length		*/
	0x1200005e,	/* unit_specifier_id: IANA	*/
	0x81000003,	/* textual descriptor offset	*/
	0x13000001,	/* unit_sw_version: RFC 2734	*/
	0x81000005,	/* textual descriptor offset	*/
	0x00030000,	/* descriptor_length		*/
	0x00000000,	/* text				*/
	0x00000000,	/* minimal ASCII, en		*/
	0x49414e41,	/* I A N A			*/
	0x00030000,	/* descriptor_length		*/
	0x00000000,	/* text				*/
	0x00000000,	/* minimal ASCII, en		*/
	0x49507634,	/* I P v 4			*/
};

static struct fw_descriptor rfc2374_unit_directory = {
	.length = ARRAY_SIZE(rfc2374_unit_directory_data),
	.key    = (CSR_DIRECTORY | CSR_UNIT) << 24,
	.data   = rfc2374_unit_directory_data
J
Jay Fenlason 已提交
1628 1629
};

S
Stefan Richter 已提交
1630 1631 1632 1633 1634 1635 1636
static int __init fwnet_init(void)
{
	int err;

	err = fw_core_add_descriptor(&rfc2374_unit_directory);
	if (err)
		return err;
J
Jay Fenlason 已提交
1637

S
Stefan Richter 已提交
1638 1639 1640 1641 1642 1643 1644 1645 1646 1647 1648 1649 1650 1651 1652 1653
	fwnet_packet_task_cache = kmem_cache_create("packet_task",
			sizeof(struct fwnet_packet_task), 0, 0, NULL);
	if (!fwnet_packet_task_cache) {
		err = -ENOMEM;
		goto out;
	}

	err = driver_register(&fwnet_driver.driver);
	if (!err)
		return 0;

	kmem_cache_destroy(fwnet_packet_task_cache);
out:
	fw_core_remove_descriptor(&rfc2374_unit_directory);

	return err;
J
Jay Fenlason 已提交
1654
}
S
Stefan Richter 已提交
1655
module_init(fwnet_init);
J
Jay Fenlason 已提交
1656

S
Stefan Richter 已提交
1657 1658 1659 1660 1661
static void __exit fwnet_cleanup(void)
{
	driver_unregister(&fwnet_driver.driver);
	kmem_cache_destroy(fwnet_packet_task_cache);
	fw_core_remove_descriptor(&rfc2374_unit_directory);
J
Jay Fenlason 已提交
1662
}
S
Stefan Richter 已提交
1663
module_exit(fwnet_cleanup);
J
Jay Fenlason 已提交
1664

S
Stefan Richter 已提交
1665 1666 1667 1668
MODULE_AUTHOR("Jay Fenlason <fenlason@redhat.com>");
MODULE_DESCRIPTION("IPv4 over IEEE1394 as per RFC 2734");
MODULE_LICENSE("GPL");
MODULE_DEVICE_TABLE(ieee1394, fwnet_id_table);