net.c 42.9 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
#include <linux/device.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 已提交
16
#include <linux/jiffies.h>
J
Jay Fenlason 已提交
17 18 19
#include <linux/mod_devicetable.h>
#include <linux/module.h>
#include <linux/moduleparam.h>
20
#include <linux/mutex.h>
J
Jay Fenlason 已提交
21 22
#include <linux/netdevice.h>
#include <linux/skbuff.h>
23
#include <linux/slab.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);
581
			goto no_peer;
J
Jay Fenlason 已提交
582
		}
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 {
631
			if (memcmp(eth->h_dest, net->dev_addr, net->addr_len))
J
Jay Fenlason 已提交
632 633 634 635 636 637
				skb->pkt_type = PACKET_OTHERHOST;
		}
		if (ntohs(eth->h_proto) >= 1536) {
			protocol = eth->h_proto;
		} else {
			rawp = (u16 *)skb->data;
S
Stefan Richter 已提交
638
			if (*rawp == 0xffff)
J
Jay Fenlason 已提交
639
				protocol = htons(ETH_P_802_3);
S
Stefan Richter 已提交
640
			else
J
Jay Fenlason 已提交
641 642 643 644 645
				protocol = htons(ETH_P_802_2);
		}
		skb->protocol = protocol;
	}
	status = netif_rx(skb);
S
Stefan Richter 已提交
646 647 648
	if (status == NET_RX_DROP) {
		net->stats.rx_errors++;
		net->stats.rx_dropped++;
J
Jay Fenlason 已提交
649
	} else {
S
Stefan Richter 已提交
650 651
		net->stats.rx_packets++;
		net->stats.rx_bytes += skb->len;
J
Jay Fenlason 已提交
652
	}
S
Stefan Richter 已提交
653 654 655
	if (netif_queue_stopped(net))
		netif_wake_queue(net);

J
Jay Fenlason 已提交
656 657
	return 0;

658
 no_peer:
S
Stefan Richter 已提交
659 660 661
	net->stats.rx_errors++;
	net->stats.rx_dropped++;

J
Jay Fenlason 已提交
662
	dev_kfree_skb_any(skb);
S
Stefan Richter 已提交
663 664 665
	if (netif_queue_stopped(net))
		netif_wake_queue(net);

666
	return -ENOENT;
J
Jay Fenlason 已提交
667 668
}

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

S
Stefan Richter 已提交
686 687 688
	hdr.w0 = be32_to_cpu(buf[0]);
	lf = fwnet_get_hdr_lf(&hdr);
	if (lf == RFC2374_HDR_UNFRAG) {
J
Jay Fenlason 已提交
689 690 691 692 693
		/*
		 * 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 已提交
694
		ether_type = fwnet_get_hdr_ether_type(&hdr);
J
Jay Fenlason 已提交
695
		buf++;
S
Stefan Richter 已提交
696
		len -= RFC2374_UNFRAG_HDR_SIZE;
J
Jay Fenlason 已提交
697

S
Stefan Richter 已提交
698
		skb = dev_alloc_skb(len + net->hard_header_len + 15);
J
Jay Fenlason 已提交
699
		if (unlikely(!skb)) {
S
Stefan Richter 已提交
700 701 702
			fw_error("out of memory\n");
			net->stats.rx_dropped++;

703
			return -ENOMEM;
J
Jay Fenlason 已提交
704
		}
S
Stefan Richter 已提交
705 706 707 708 709
		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 已提交
710 711 712
	}
	/* A datagram fragment has been received, now the fun begins. */
	hdr.w1 = ntohl(buf[1]);
S
Stefan Richter 已提交
713 714 715 716
	buf += 2;
	len -= RFC2374_FRAG_HDR_SIZE;
	if (lf == RFC2374_HDR_FIRSTFRAG) {
		ether_type = fwnet_get_hdr_ether_type(&hdr);
J
Jay Fenlason 已提交
717 718
		fg_off = 0;
	} else {
S
Stefan Richter 已提交
719 720
		ether_type = 0;
		fg_off = fwnet_get_hdr_fg_off(&hdr);
J
Jay Fenlason 已提交
721
	}
S
Stefan Richter 已提交
722 723 724
	datagram_label = fwnet_get_hdr_dgl(&hdr);
	dg_size = fwnet_get_hdr_dg_size(&hdr); /* ??? + 1 */

725 726 727
	spin_lock_irqsave(&dev->lock, flags);

	peer = fwnet_peer_find_by_node_id(dev, source_node_id, generation);
728 729 730 731
	if (!peer) {
		retval = -ENOENT;
		goto fail;
	}
S
Stefan Richter 已提交
732 733

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

S
Stefan Richter 已提交
778
	if (lf == RFC2374_HDR_FIRSTFRAG)
J
Jay Fenlason 已提交
779
		pd->ether_type = ether_type;
S
Stefan Richter 已提交
780 781

	if (fwnet_pd_is_complete(pd)) {
J
Jay Fenlason 已提交
782
		ether_type = pd->ether_type;
S
Stefan Richter 已提交
783
		peer->pdg_size--;
J
Jay Fenlason 已提交
784
		skb = skb_get(pd->skb);
S
Stefan Richter 已提交
785 786
		fwnet_pd_delete(pd);

787
		spin_unlock_irqrestore(&dev->lock, flags);
S
Stefan Richter 已提交
788 789 790

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

J
Jay Fenlason 已提交
798
	return 0;
799
 fail:
800
	spin_unlock_irqrestore(&dev->lock, flags);
S
Stefan Richter 已提交
801 802 803 804

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

805
	return retval;
J
Jay Fenlason 已提交
806 807
}

S
Stefan Richter 已提交
808 809
static void fwnet_receive_packet(struct fw_card *card, struct fw_request *r,
		int tcode, int destination, int source, int generation,
810 811
		unsigned long long offset, void *payload, size_t length,
		void *callback_data)
S
Stefan Richter 已提交
812
{
813 814
	struct fwnet_device *dev = callback_data;
	int rcode;
J
Jay Fenlason 已提交
815

816 817
	if (destination == IEEE1394_ALL_NODES) {
		kfree(r);
S
Stefan Richter 已提交
818

J
Jay Fenlason 已提交
819 820
		return;
	}
S
Stefan Richter 已提交
821

822 823 824 825 826 827
	if (offset != dev->handler.offset)
		rcode = RCODE_ADDRESS_ERROR;
	else if (tcode != TCODE_WRITE_BLOCK_REQUEST)
		rcode = RCODE_TYPE_ERROR;
	else if (fwnet_incoming_packet(dev, payload, length,
				       source, generation, false) != 0) {
S
Stefan Richter 已提交
828
		fw_error("Incoming packet failure\n");
829 830 831
		rcode = RCODE_CONFLICT_ERROR;
	} else
		rcode = RCODE_COMPLETE;
S
Stefan Richter 已提交
832

833
	fw_send_response(card, r, rcode);
J
Jay Fenlason 已提交
834 835
}

S
Stefan Richter 已提交
836 837 838 839
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 已提交
840 841
	struct fw_iso_packet packet;
	struct fw_card *card;
S
Stefan Richter 已提交
842 843
	__be16 *hdr_ptr;
	__be32 *buf_ptr;
J
Jay Fenlason 已提交
844 845 846 847 848 849 850 851
	int retval;
	u32 length;
	u16 source_node_id;
	u32 specifier_id;
	u32 ver;
	unsigned long offset;
	unsigned long flags;

S
Stefan Richter 已提交
852 853
	dev = data;
	card = dev->card;
J
Jay Fenlason 已提交
854
	hdr_ptr = header;
S
Stefan Richter 已提交
855 856 857 858 859 860 861 862 863 864
	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 已提交
865 866 867

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

	if (specifier_id == IANA_SPECIFIER_ID && ver == RFC2734_SW_VERSION) {
J
Jay Fenlason 已提交
872
		buf_ptr += 2;
S
Stefan Richter 已提交
873 874
		length -= IEEE1394_GASP_HDR_SIZE;
		fwnet_incoming_packet(dev, buf_ptr, length,
875
				      source_node_id, -1, true);
S
Stefan Richter 已提交
876 877 878
	}

	packet.payload_length = dev->rcv_buffer_size;
J
Jay Fenlason 已提交
879 880 881 882
	packet.interrupt = 1;
	packet.skip = 0;
	packet.tag = 3;
	packet.sy = 0;
S
Stefan Richter 已提交
883 884 885
	packet.header_length = IEEE1394_GASP_HDR_SIZE;

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

S
Stefan Richter 已提交
887 888 889 890 891 892 893
	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 已提交
894 895
}

S
Stefan Richter 已提交
896 897
static struct kmem_cache *fwnet_packet_task_cache;

898 899 900 901 902 903
static void fwnet_free_ptask(struct fwnet_packet_task *ptask)
{
	dev_kfree_skb_any(ptask->skb);
	kmem_cache_free(fwnet_packet_task_cache, ptask);
}

S
Stefan Richter 已提交
904 905 906 907
static int fwnet_send_packet(struct fwnet_packet_task *ptask);

static void fwnet_transmit_packet_done(struct fwnet_packet_task *ptask)
{
908
	struct fwnet_device *dev = ptask->dev;
909
	struct sk_buff *skb = ptask->skb;
J
Jay Fenlason 已提交
910
	unsigned long flags;
911
	bool free;
S
Stefan Richter 已提交
912 913 914

	spin_lock_irqsave(&dev->lock, flags);

915 916 917 918
	ptask->outstanding_pkts--;

	/* Check whether we or the networking TX soft-IRQ is last user. */
	free = (ptask->outstanding_pkts == 0 && !list_empty(&ptask->pt_link));
S
Stefan Richter 已提交
919 920
	if (free)
		list_del(&ptask->pt_link);
921

922 923 924 925
	if (ptask->outstanding_pkts == 0) {
		dev->netdev->stats.tx_packets++;
		dev->netdev->stats.tx_bytes += skb->len;
	}
926 927

	spin_unlock_irqrestore(&dev->lock, flags);
S
Stefan Richter 已提交
928 929

	if (ptask->outstanding_pkts > 0) {
J
Jay Fenlason 已提交
930 931 932 933 934 935
		u16 dg_size;
		u16 fg_off;
		u16 datagram_label;
		u16 lf;

		/* Update the ptask to point to the next fragment and send it */
S
Stefan Richter 已提交
936
		lf = fwnet_get_hdr_lf(&ptask->hdr);
J
Jay Fenlason 已提交
937
		switch (lf) {
S
Stefan Richter 已提交
938 939
		case RFC2374_HDR_LASTFRAG:
		case RFC2374_HDR_UNFRAG:
J
Jay Fenlason 已提交
940
		default:
S
Stefan Richter 已提交
941 942 943
			fw_error("Outstanding packet %x lf %x, header %x,%x\n",
				 ptask->outstanding_pkts, lf, ptask->hdr.w0,
				 ptask->hdr.w1);
J
Jay Fenlason 已提交
944 945
			BUG();

S
Stefan Richter 已提交
946
		case RFC2374_HDR_FIRSTFRAG:
J
Jay Fenlason 已提交
947
			/* Set frag type here for future interior fragments */
S
Stefan Richter 已提交
948 949 950
			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 已提交
951 952
			break;

S
Stefan Richter 已提交
953 954 955 956 957
		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 已提交
958 959
			break;
		}
960

S
Stefan Richter 已提交
961 962 963 964
		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 已提交
965
		} else {
S
Stefan Richter 已提交
966 967 968
			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 已提交
969
		}
S
Stefan Richter 已提交
970
		fwnet_send_packet(ptask);
J
Jay Fenlason 已提交
971
	}
972 973 974

	if (free)
		fwnet_free_ptask(ptask);
J
Jay Fenlason 已提交
975 976
}

S
Stefan Richter 已提交
977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001
static void fwnet_transmit_packet_failed(struct fwnet_packet_task *ptask)
{
	struct fwnet_device *dev = ptask->dev;
	unsigned long flags;
	bool free;

	spin_lock_irqsave(&dev->lock, flags);

	/* One fragment failed; don't try to send remaining fragments. */
	ptask->outstanding_pkts = 0;

	/* Check whether we or the networking TX soft-IRQ is last user. */
	free = !list_empty(&ptask->pt_link);
	if (free)
		list_del(&ptask->pt_link);

	dev->netdev->stats.tx_dropped++;
	dev->netdev->stats.tx_errors++;

	spin_unlock_irqrestore(&dev->lock, flags);

	if (free)
		fwnet_free_ptask(ptask);
}

S
Stefan Richter 已提交
1002 1003 1004 1005
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 已提交
1006 1007 1008

	ptask = data;

S
Stefan Richter 已提交
1009
	if (rcode == RCODE_COMPLETE) {
S
Stefan Richter 已提交
1010
		fwnet_transmit_packet_done(ptask);
S
Stefan Richter 已提交
1011
	} else {
S
Stefan Richter 已提交
1012
		fw_error("fwnet_write_complete: failed: %x\n", rcode);
S
Stefan Richter 已提交
1013 1014
		fwnet_transmit_packet_failed(ptask);
	}
J
Jay Fenlason 已提交
1015 1016
}

S
Stefan Richter 已提交
1017 1018 1019
static int fwnet_send_packet(struct fwnet_packet_task *ptask)
{
	struct fwnet_device *dev;
J
Jay Fenlason 已提交
1020
	unsigned tx_len;
S
Stefan Richter 已提交
1021
	struct rfc2734_header *bufhdr;
J
Jay Fenlason 已提交
1022
	unsigned long flags;
1023
	bool free;
J
Jay Fenlason 已提交
1024

S
Stefan Richter 已提交
1025
	dev = ptask->dev;
J
Jay Fenlason 已提交
1026
	tx_len = ptask->max_payload;
S
Stefan Richter 已提交
1027 1028 1029 1030 1031
	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 已提交
1032 1033
		break;

S
Stefan Richter 已提交
1034 1035 1036 1037 1038 1039 1040
	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 已提交
1041 1042 1043 1044 1045
		break;

	default:
		BUG();
	}
S
Stefan Richter 已提交
1046 1047
	if (ptask->dest_node == IEEE1394_ALL_NODES) {
		u8 *p;
J
Jay Fenlason 已提交
1048
		int generation;
S
Stefan Richter 已提交
1049
		int node_id;
J
Jay Fenlason 已提交
1050 1051

		/* ptask->generation may not have been set yet */
S
Stefan Richter 已提交
1052
		generation = dev->card->generation;
J
Jay Fenlason 已提交
1053
		smp_rmb();
S
Stefan Richter 已提交
1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069
		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);

		spin_lock_irqsave(&dev->lock, flags);
1070 1071 1072 1073 1074 1075

		/* If the AT tasklet already ran, we may be last user. */
		free = (ptask->outstanding_pkts == 0 && list_empty(&ptask->pt_link));
		if (!free)
			list_add_tail(&ptask->pt_link, &dev->broadcasted_list);

S
Stefan Richter 已提交
1076 1077
		spin_unlock_irqrestore(&dev->lock, flags);

1078
		goto out;
J
Jay Fenlason 已提交
1079
	}
S
Stefan Richter 已提交
1080 1081 1082 1083 1084 1085 1086

	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);

	spin_lock_irqsave(&dev->lock, flags);
1087 1088 1089 1090 1091 1092

	/* If the AT tasklet already ran, we may be last user. */
	free = (ptask->outstanding_pkts == 0 && list_empty(&ptask->pt_link));
	if (!free)
		list_add_tail(&ptask->pt_link, &dev->sent_list);

S
Stefan Richter 已提交
1093 1094
	spin_unlock_irqrestore(&dev->lock, flags);

1095
	dev->netdev->trans_start = jiffies;
1096 1097 1098
 out:
	if (free)
		fwnet_free_ptask(ptask);
S
Stefan Richter 已提交
1099

J
Jay Fenlason 已提交
1100 1101 1102
	return 0;
}

S
Stefan Richter 已提交
1103 1104
static int fwnet_broadcast_start(struct fwnet_device *dev)
{
J
Jay Fenlason 已提交
1105 1106 1107 1108 1109 1110 1111 1112
	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 已提交
1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125
	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 已提交
1126
			goto failed_initial;
S
Stefan Richter 已提交
1127 1128

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

S
Stefan Richter 已提交
1131 1132 1133 1134
	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 已提交
1135 1136
		void **ptrptr;

S
Stefan Richter 已提交
1137 1138 1139
		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 已提交
1140 1141 1142 1143
		if (IS_ERR(context)) {
			retval = PTR_ERR(context);
			goto failed_context_create;
		}
S
Stefan Richter 已提交
1144 1145 1146 1147

		retval = fw_iso_buffer_init(&dev->broadcast_rcv_buffer,
		    dev->card, FWNET_ISO_PAGE_COUNT, DMA_FROM_DEVICE);
		if (retval < 0)
J
Jay Fenlason 已提交
1148
			goto failed_buffer_init;
S
Stefan Richter 已提交
1149 1150 1151

		ptrptr = kmalloc(sizeof(void *) * num_packets, GFP_KERNEL);
		if (!ptrptr) {
J
Jay Fenlason 已提交
1152 1153 1154
			retval = -ENOMEM;
			goto failed_ptrs_alloc;
		}
S
Stefan Richter 已提交
1155 1156 1157

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

S
Stefan Richter 已提交
1161 1162 1163 1164
			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 已提交
1165
		}
S
Stefan Richter 已提交
1166 1167 1168 1169
		dev->broadcast_rcv_context = context;
	} else {
		context = dev->broadcast_rcv_context;
	}
J
Jay Fenlason 已提交
1170 1171 1172 1173 1174 1175

	packet.payload_length = max_receive;
	packet.interrupt = 1;
	packet.skip = 0;
	packet.tag = 3;
	packet.sy = 0;
S
Stefan Richter 已提交
1176
	packet.header_length = IEEE1394_GASP_HDR_SIZE;
J
Jay Fenlason 已提交
1177
	offset = 0;
S
Stefan Richter 已提交
1178 1179 1180 1181 1182

	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 已提交
1183
			goto failed_rcv_queue;
S
Stefan Richter 已提交
1184

J
Jay Fenlason 已提交
1185 1186
		offset += max_receive;
	}
S
Stefan Richter 已提交
1187 1188 1189 1190 1191 1192
	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 已提交
1193
		goto failed_rcv_queue;
S
Stefan Richter 已提交
1194 1195 1196 1197 1198 1199

	/* 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 已提交
1200 1201 1202
	return 0;

 failed_rcv_queue:
S
Stefan Richter 已提交
1203 1204
	kfree(dev->broadcast_rcv_buffer_ptrs);
	dev->broadcast_rcv_buffer_ptrs = NULL;
J
Jay Fenlason 已提交
1205
 failed_ptrs_alloc:
S
Stefan Richter 已提交
1206
	fw_iso_buffer_destroy(&dev->broadcast_rcv_buffer, dev->card);
J
Jay Fenlason 已提交
1207
 failed_buffer_init:
S
Stefan Richter 已提交
1208 1209
	fw_iso_context_destroy(context);
	dev->broadcast_rcv_context = NULL;
J
Jay Fenlason 已提交
1210
 failed_context_create:
S
Stefan Richter 已提交
1211
	fw_core_remove_address_handler(&dev->handler);
J
Jay Fenlason 已提交
1212
 failed_initial:
S
Stefan Richter 已提交
1213 1214
	dev->local_fifo = FWNET_NO_FIFO_ADDR;

J
Jay Fenlason 已提交
1215 1216 1217
	return retval;
}

S
Stefan Richter 已提交
1218 1219 1220 1221
/* ifup */
static int fwnet_open(struct net_device *net)
{
	struct fwnet_device *dev = netdev_priv(net);
J
Jay Fenlason 已提交
1222 1223
	int ret;

S
Stefan Richter 已提交
1224 1225
	if (dev->broadcast_state == FWNET_BROADCAST_ERROR) {
		ret = fwnet_broadcast_start(dev);
J
Jay Fenlason 已提交
1226 1227 1228
		if (ret)
			return ret;
	}
S
Stefan Richter 已提交
1229 1230
	netif_start_queue(net);

J
Jay Fenlason 已提交
1231 1232 1233
	return 0;
}

S
Stefan Richter 已提交
1234 1235
/* ifdown */
static int fwnet_stop(struct net_device *net)
J
Jay Fenlason 已提交
1236
{
S
Stefan Richter 已提交
1237 1238 1239
	netif_stop_queue(net);

	/* Deallocate iso context for use by other applications? */
J
Jay Fenlason 已提交
1240 1241 1242 1243

	return 0;
}

1244
static netdev_tx_t fwnet_tx(struct sk_buff *skb, struct net_device *net)
J
Jay Fenlason 已提交
1245
{
S
Stefan Richter 已提交
1246 1247
	struct fwnet_header hdr_buf;
	struct fwnet_device *dev = netdev_priv(net);
J
Jay Fenlason 已提交
1248 1249 1250 1251 1252
	__be16 proto;
	u16 dest_node;
	unsigned max_payload;
	u16 dg_size;
	u16 *datagram_label_ptr;
S
Stefan Richter 已提交
1253
	struct fwnet_packet_task *ptask;
1254 1255
	struct fwnet_peer *peer;
	unsigned long flags;
J
Jay Fenlason 已提交
1256

S
Stefan Richter 已提交
1257
	ptask = kmem_cache_alloc(fwnet_packet_task_cache, GFP_ATOMIC);
J
Jay Fenlason 已提交
1258 1259 1260 1261 1262 1263 1264 1265
	if (ptask == NULL)
		goto fail;

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

	/*
S
Stefan Richter 已提交
1266
	 * Make a copy of the driver-specific header.
J
Jay Fenlason 已提交
1267 1268 1269 1270 1271 1272 1273 1274
	 * 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;

1275 1276 1277
	/* serialize access to peer, including peer->datagram_label */
	spin_lock_irqsave(&dev->lock, flags);

J
Jay Fenlason 已提交
1278 1279 1280 1281
	/*
	 * Set the transmission type for the packet.  ARP packets and IP
	 * broadcast packets are sent via GASP.
	 */
S
Stefan Richter 已提交
1282
	if (memcmp(hdr_buf.h_dest, net->broadcast, FWNET_ALEN) == 0
J
Jay Fenlason 已提交
1283
	    || proto == htons(ETH_P_ARP)
S
Stefan Richter 已提交
1284 1285
	    || (proto == htons(ETH_P_IP)
		&& IN_MULTICAST(ntohl(ip_hdr(skb)->daddr)))) {
1286
		max_payload        = dev->broadcast_xmt_max_payload;
S
Stefan Richter 已提交
1287 1288
		datagram_label_ptr = &dev->broadcast_xmt_datagramlabel;

1289 1290 1291 1292
		ptask->fifo_addr   = FWNET_NO_FIFO_ADDR;
		ptask->generation  = 0;
		ptask->dest_node   = IEEE1394_ALL_NODES;
		ptask->speed       = SCODE_100;
J
Jay Fenlason 已提交
1293
	} else {
S
Stefan Richter 已提交
1294
		__be64 guid = get_unaligned((__be64 *)hdr_buf.h_dest);
J
Jay Fenlason 已提交
1295 1296
		u8 generation;

S
Stefan Richter 已提交
1297
		peer = fwnet_peer_find_by_guid(dev, be64_to_cpu(guid));
1298 1299
		if (!peer || peer->fifo == FWNET_NO_FIFO_ADDR)
			goto fail_unlock;
J
Jay Fenlason 已提交
1300

1301 1302 1303
		generation         = peer->generation;
		dest_node          = peer->node_id;
		max_payload        = peer->max_payload;
S
Stefan Richter 已提交
1304
		datagram_label_ptr = &peer->datagram_label;
J
Jay Fenlason 已提交
1305

1306 1307 1308 1309
		ptask->fifo_addr   = peer->fifo;
		ptask->generation  = generation;
		ptask->dest_node   = dest_node;
		ptask->speed       = peer->speed;
J
Jay Fenlason 已提交
1310 1311 1312 1313 1314 1315
	}

	/* 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 已提交
1316 1317 1318 1319 1320 1321 1322 1323 1324 1325 1326 1327 1328 1329
		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 已提交
1330 1331 1332 1333 1334
	}

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

J
Jay Fenlason 已提交
1337
	/* Does it all fit in one packet? */
S
Stefan Richter 已提交
1338 1339
	if (dg_size <= max_payload) {
		fwnet_make_uf_hdr(&ptask->hdr, ntohs(proto));
J
Jay Fenlason 已提交
1340
		ptask->outstanding_pkts = 1;
S
Stefan Richter 已提交
1341
		max_payload = dg_size + RFC2374_UNFRAG_HDR_SIZE;
J
Jay Fenlason 已提交
1342 1343 1344
	} else {
		u16 datagram_label;

S
Stefan Richter 已提交
1345
		max_payload -= RFC2374_FRAG_OVERHEAD;
J
Jay Fenlason 已提交
1346
		datagram_label = (*datagram_label_ptr)++;
S
Stefan Richter 已提交
1347 1348
		fwnet_make_ff_hdr(&ptask->hdr, ntohs(proto), dg_size,
				  datagram_label);
J
Jay Fenlason 已提交
1349
		ptask->outstanding_pkts = DIV_ROUND_UP(dg_size, max_payload);
S
Stefan Richter 已提交
1350
		max_payload += RFC2374_FRAG_HDR_SIZE;
J
Jay Fenlason 已提交
1351
	}
1352 1353 1354

	spin_unlock_irqrestore(&dev->lock, flags);

J
Jay Fenlason 已提交
1355
	ptask->max_payload = max_payload;
1356 1357
	INIT_LIST_HEAD(&ptask->pt_link);

S
Stefan Richter 已提交
1358 1359
	fwnet_send_packet(ptask);

J
Jay Fenlason 已提交
1360 1361
	return NETDEV_TX_OK;

1362 1363
 fail_unlock:
	spin_unlock_irqrestore(&dev->lock, flags);
J
Jay Fenlason 已提交
1364 1365
 fail:
	if (ptask)
S
Stefan Richter 已提交
1366
		kmem_cache_free(fwnet_packet_task_cache, ptask);
J
Jay Fenlason 已提交
1367 1368 1369 1370

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

S
Stefan Richter 已提交
1371 1372
	net->stats.tx_dropped++;
	net->stats.tx_errors++;
J
Jay Fenlason 已提交
1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 1383

	/*
	 * 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 已提交
1384 1385
static int fwnet_change_mtu(struct net_device *net, int new_mtu)
{
J
Jay Fenlason 已提交
1386 1387 1388
	if (new_mtu < 68)
		return -EINVAL;

S
Stefan Richter 已提交
1389
	net->mtu = new_mtu;
J
Jay Fenlason 已提交
1390 1391 1392
	return 0;
}

S
Stefan Richter 已提交
1393 1394 1395 1396 1397
static const struct net_device_ops fwnet_netdev_ops = {
	.ndo_open       = fwnet_open,
	.ndo_stop	= fwnet_stop,
	.ndo_start_xmit = fwnet_tx,
	.ndo_change_mtu = fwnet_change_mtu,
J
Jay Fenlason 已提交
1398 1399
};

S
Stefan Richter 已提交
1400 1401 1402 1403
static void fwnet_init_dev(struct net_device *net)
{
	net->header_ops		= &fwnet_header_ops;
	net->netdev_ops		= &fwnet_netdev_ops;
1404
	net->watchdog_timeo	= 2 * HZ;
S
Stefan Richter 已提交
1405 1406 1407 1408 1409
	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;
1410
	net->tx_queue_len	= 10;
J
Jay Fenlason 已提交
1411 1412
}

1413 1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433
/* 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;

1434 1435
	dev_set_drvdata(&unit->device, peer);

1436 1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455
	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 已提交
1456 1457 1458 1459 1460 1461
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;
1462
	bool allocated_netdev = false;
S
Stefan Richter 已提交
1463
	struct fwnet_device *dev;
J
Jay Fenlason 已提交
1464
	unsigned max_mtu;
1465
	int ret;
J
Jay Fenlason 已提交
1466

1467
	mutex_lock(&fwnet_device_mutex);
J
Jay Fenlason 已提交
1468

1469 1470 1471 1472
	dev = fwnet_dev_find(card);
	if (dev) {
		net = dev->netdev;
		goto have_dev;
J
Jay Fenlason 已提交
1473
	}
1474

S
Stefan Richter 已提交
1475 1476
	net = alloc_netdev(sizeof(*dev), "firewire%d", fwnet_init_dev);
	if (net == NULL) {
1477
		ret = -ENOMEM;
J
Jay Fenlason 已提交
1478 1479 1480
		goto out;
	}

1481
	allocated_netdev = true;
S
Stefan Richter 已提交
1482 1483
	SET_NETDEV_DEV(net, card->device);
	dev = netdev_priv(net);
J
Jay Fenlason 已提交
1484

S
Stefan Richter 已提交
1485 1486 1487 1488 1489
	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 已提交
1490

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

S
Stefan Richter 已提交
1493 1494 1495
	INIT_LIST_HEAD(&dev->packet_list);
	INIT_LIST_HEAD(&dev->broadcasted_list);
	INIT_LIST_HEAD(&dev->sent_list);
1496
	INIT_LIST_HEAD(&dev->peer_list);
J
Jay Fenlason 已提交
1497

S
Stefan Richter 已提交
1498
	dev->card = card;
1499
	dev->netdev = net;
J
Jay Fenlason 已提交
1500 1501 1502 1503 1504 1505

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

	/* Set our hardware address while we're at it */
S
Stefan Richter 已提交
1510 1511
	put_unaligned_be64(card->guid, net->dev_addr);
	put_unaligned_be64(~0ULL, net->broadcast);
1512 1513
	ret = register_netdev(net);
	if (ret) {
S
Stefan Richter 已提交
1514
		fw_error("Cannot register the driver\n");
J
Jay Fenlason 已提交
1515 1516 1517
		goto out;
	}

1518
	list_add_tail(&dev->dev_link, &fwnet_device_list);
S
Stefan Richter 已提交
1519 1520
	fw_notify("%s: IPv4 over FireWire on device %016llx\n",
		  net->name, (unsigned long long)card->guid);
1521 1522
 have_dev:
	ret = fwnet_add_peer(dev, unit, device);
1523
	if (ret && allocated_netdev) {
1524 1525 1526
		unregister_netdev(net);
		list_del(&dev->dev_link);
	}
J
Jay Fenlason 已提交
1527
 out:
1528
	if (ret && allocated_netdev)
S
Stefan Richter 已提交
1529 1530
		free_netdev(net);

1531 1532 1533 1534 1535 1536 1537 1538 1539 1540 1541 1542 1543 1544 1545 1546 1547
	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 已提交
1548 1549
}

S
Stefan Richter 已提交
1550 1551
static int fwnet_remove(struct device *_dev)
{
1552
	struct fwnet_peer *peer = dev_get_drvdata(_dev);
1553
	struct fwnet_device *dev = peer->dev;
S
Stefan Richter 已提交
1554 1555 1556
	struct net_device *net;
	struct fwnet_packet_task *ptask, *pt_next;

1557
	mutex_lock(&fwnet_device_mutex);
J
Jay Fenlason 已提交
1558

1559
	fwnet_remove_peer(peer);
S
Stefan Richter 已提交
1560

1561 1562
	if (list_empty(&dev->peer_list)) {
		net = dev->netdev;
S
Stefan Richter 已提交
1563 1564 1565 1566 1567 1568 1569 1570 1571
		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 已提交
1572
		}
S
Stefan Richter 已提交
1573 1574 1575 1576
		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 已提交
1577
		}
S
Stefan Richter 已提交
1578 1579 1580 1581
		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 已提交
1582
		}
S
Stefan Richter 已提交
1583 1584 1585 1586
		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 已提交
1587
		}
1588 1589
		list_del(&dev->dev_link);

S
Stefan Richter 已提交
1590
		free_netdev(net);
J
Jay Fenlason 已提交
1591
	}
S
Stefan Richter 已提交
1592

1593 1594
	mutex_unlock(&fwnet_device_mutex);

J
Jay Fenlason 已提交
1595 1596 1597
	return 0;
}

S
Stefan Richter 已提交
1598 1599 1600 1601 1602 1603 1604
/*
 * 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);
1605
	struct fwnet_peer *peer = dev_get_drvdata(&unit->device);
1606
	int generation;
J
Jay Fenlason 已提交
1607

1608 1609 1610 1611 1612 1613
	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 已提交
1614 1615
}

S
Stefan Richter 已提交
1616 1617 1618 1619 1620 1621 1622 1623 1624 1625 1626
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 已提交
1627
	.driver = {
S
Stefan Richter 已提交
1628 1629 1630 1631 1632
		.owner  = THIS_MODULE,
		.name   = "net",
		.bus    = &fw_bus_type,
		.probe  = fwnet_probe,
		.remove = fwnet_remove,
J
Jay Fenlason 已提交
1633
	},
S
Stefan Richter 已提交
1634 1635 1636 1637 1638 1639 1640 1641 1642 1643 1644 1645 1646 1647 1648 1649 1650 1651 1652 1653 1654 1655 1656 1657
	.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 已提交
1658 1659
};

S
Stefan Richter 已提交
1660 1661 1662 1663 1664 1665 1666
static int __init fwnet_init(void)
{
	int err;

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

S
Stefan Richter 已提交
1668 1669 1670 1671 1672 1673 1674 1675 1676 1677 1678 1679 1680 1681 1682 1683
	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 已提交
1684
}
S
Stefan Richter 已提交
1685
module_init(fwnet_init);
J
Jay Fenlason 已提交
1686

S
Stefan Richter 已提交
1687 1688 1689 1690 1691
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 已提交
1692
}
S
Stefan Richter 已提交
1693
module_exit(fwnet_cleanup);
J
Jay Fenlason 已提交
1694

S
Stefan Richter 已提交
1695 1696 1697 1698
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);