net.c 42.0 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 {
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 658
	return 0;

 failed_proto:
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 666 667
	if (netif_queue_stopped(net))
		netif_wake_queue(net);

	net->last_rx = jiffies;

J
Jay Fenlason 已提交
668 669 670
	return 0;
}

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

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

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

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

727 728 729 730 731
	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 已提交
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 745 746
			retval = -ENOMEM;
			goto bad_proto;
		}
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
			fwnet_pd_delete(pd);
			pd = fwnet_pd_new(net, peer, datagram_label,
					  dg_size, buf, fg_off, len);
			if (pd == NULL) {
J
Jay Fenlason 已提交
759
				retval = -ENOMEM;
S
Stefan Richter 已提交
760
				peer->pdg_size--;
J
Jay Fenlason 已提交
761 762 763
				goto bad_proto;
			}
		} 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--;
J
Jay Fenlason 已提交
772 773 774 775 776
				goto bad_proto;
			}
		}
	} /* new datagram or add to existing one */

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

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

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

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

J
Jay Fenlason 已提交
797 798 799
	return 0;

 bad_proto:
800
	spin_unlock_irqrestore(&dev->lock, flags);
S
Stefan Richter 已提交
801 802 803 804

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

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

S
Stefan Richter 已提交
808 809 810 811 812 813
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 已提交
814 815
	int status;

S
Stefan Richter 已提交
816 817 818 819 820
	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 已提交
821
		fw_send_response(card, r, RCODE_CONFLICT_ERROR);
S
Stefan Richter 已提交
822

J
Jay Fenlason 已提交
823 824
		return;
	}
S
Stefan Richter 已提交
825

826 827
	status = fwnet_incoming_packet(dev, payload, length,
				       source, generation, false);
S
Stefan Richter 已提交
828 829 830 831
	if (status != 0) {
		fw_error("Incoming packet failure\n");
		fw_send_response(card, r, RCODE_CONFLICT_ERROR);

J
Jay Fenlason 已提交
832 833
		return;
	}
S
Stefan Richter 已提交
834 835

	fw_send_response(card, r, RCODE_COMPLETE);
J
Jay Fenlason 已提交
836 837
}

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

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

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

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

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

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

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

S
Stefan Richter 已提交
898 899 900 901 902 903 904
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 已提交
905 906
	unsigned long flags;

S
Stefan Richter 已提交
907 908 909 910 911 912 913 914 915
	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 已提交
916 917 918 919 920 921 922
		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 已提交
923
		lf = fwnet_get_hdr_lf(&ptask->hdr);
J
Jay Fenlason 已提交
924
		switch (lf) {
S
Stefan Richter 已提交
925 926
		case RFC2374_HDR_LASTFRAG:
		case RFC2374_HDR_UNFRAG:
J
Jay Fenlason 已提交
927
		default:
S
Stefan Richter 已提交
928 929 930
			fw_error("Outstanding packet %x lf %x, header %x,%x\n",
				 ptask->outstanding_pkts, lf, ptask->hdr.w0,
				 ptask->hdr.w1);
J
Jay Fenlason 已提交
931 932
			BUG();

S
Stefan Richter 已提交
933
		case RFC2374_HDR_FIRSTFRAG:
J
Jay Fenlason 已提交
934
			/* Set frag type here for future interior fragments */
S
Stefan Richter 已提交
935 936 937
			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 已提交
938 939
			break;

S
Stefan Richter 已提交
940 941 942 943 944
		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 已提交
945 946 947
			break;
		}
		skb = ptask->skb;
S
Stefan Richter 已提交
948 949 950 951
		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 已提交
952
		} else {
S
Stefan Richter 已提交
953 954 955
			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 已提交
956
		}
S
Stefan Richter 已提交
957
		fwnet_send_packet(ptask);
J
Jay Fenlason 已提交
958
	} else {
S
Stefan Richter 已提交
959 960
		dev_kfree_skb_any(ptask->skb);
		kmem_cache_free(fwnet_packet_task_cache, ptask);
J
Jay Fenlason 已提交
961 962 963
	}
}

S
Stefan Richter 已提交
964 965 966 967
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 已提交
968 969 970

	ptask = data;

S
Stefan Richter 已提交
971 972 973 974
	if (rcode == RCODE_COMPLETE)
		fwnet_transmit_packet_done(ptask);
	else
		fw_error("fwnet_write_complete: failed: %x\n", rcode);
J
Jay Fenlason 已提交
975 976 977
		/* ??? error recovery */
}

S
Stefan Richter 已提交
978 979 980
static int fwnet_send_packet(struct fwnet_packet_task *ptask)
{
	struct fwnet_device *dev;
J
Jay Fenlason 已提交
981
	unsigned tx_len;
S
Stefan Richter 已提交
982
	struct rfc2734_header *bufhdr;
J
Jay Fenlason 已提交
983 984
	unsigned long flags;

S
Stefan Richter 已提交
985
	dev = ptask->dev;
J
Jay Fenlason 已提交
986
	tx_len = ptask->max_payload;
S
Stefan Richter 已提交
987 988 989 990 991
	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 已提交
992 993
		break;

S
Stefan Richter 已提交
994 995 996 997 998 999 1000
	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 已提交
1001 1002 1003 1004 1005
		break;

	default:
		BUG();
	}
S
Stefan Richter 已提交
1006 1007
	if (ptask->dest_node == IEEE1394_ALL_NODES) {
		u8 *p;
J
Jay Fenlason 已提交
1008
		int generation;
S
Stefan Richter 已提交
1009
		int node_id;
J
Jay Fenlason 已提交
1010 1011

		/* ptask->generation may not have been set yet */
S
Stefan Richter 已提交
1012
		generation = dev->card->generation;
J
Jay Fenlason 已提交
1013
		smp_rmb();
S
Stefan Richter 已提交
1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033
		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 已提交
1034 1035
		return 0;
	}
S
Stefan Richter 已提交
1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046

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

1047
	dev->netdev->trans_start = jiffies;
S
Stefan Richter 已提交
1048

J
Jay Fenlason 已提交
1049 1050 1051
	return 0;
}

S
Stefan Richter 已提交
1052 1053
static int fwnet_broadcast_start(struct fwnet_device *dev)
{
J
Jay Fenlason 已提交
1054 1055 1056 1057 1058 1059 1060 1061
	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 已提交
1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074
	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 已提交
1075
			goto failed_initial;
S
Stefan Richter 已提交
1076 1077

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

S
Stefan Richter 已提交
1080 1081 1082 1083
	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 已提交
1084 1085
		void **ptrptr;

S
Stefan Richter 已提交
1086 1087 1088
		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 已提交
1089 1090 1091 1092
		if (IS_ERR(context)) {
			retval = PTR_ERR(context);
			goto failed_context_create;
		}
S
Stefan Richter 已提交
1093 1094 1095 1096

		retval = fw_iso_buffer_init(&dev->broadcast_rcv_buffer,
		    dev->card, FWNET_ISO_PAGE_COUNT, DMA_FROM_DEVICE);
		if (retval < 0)
J
Jay Fenlason 已提交
1097
			goto failed_buffer_init;
S
Stefan Richter 已提交
1098 1099 1100

		ptrptr = kmalloc(sizeof(void *) * num_packets, GFP_KERNEL);
		if (!ptrptr) {
J
Jay Fenlason 已提交
1101 1102 1103
			retval = -ENOMEM;
			goto failed_ptrs_alloc;
		}
S
Stefan Richter 已提交
1104 1105 1106

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

S
Stefan Richter 已提交
1110 1111 1112 1113
			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 已提交
1114
		}
S
Stefan Richter 已提交
1115 1116 1117 1118
		dev->broadcast_rcv_context = context;
	} else {
		context = dev->broadcast_rcv_context;
	}
J
Jay Fenlason 已提交
1119 1120 1121 1122 1123 1124

	packet.payload_length = max_receive;
	packet.interrupt = 1;
	packet.skip = 0;
	packet.tag = 3;
	packet.sy = 0;
S
Stefan Richter 已提交
1125
	packet.header_length = IEEE1394_GASP_HDR_SIZE;
J
Jay Fenlason 已提交
1126
	offset = 0;
S
Stefan Richter 已提交
1127 1128 1129 1130 1131

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

J
Jay Fenlason 已提交
1134 1135
		offset += max_receive;
	}
S
Stefan Richter 已提交
1136 1137 1138 1139 1140 1141
	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 已提交
1142
		goto failed_rcv_queue;
S
Stefan Richter 已提交
1143 1144 1145 1146 1147 1148

	/* 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 已提交
1149 1150 1151
	return 0;

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

J
Jay Fenlason 已提交
1164 1165 1166
	return retval;
}

S
Stefan Richter 已提交
1167 1168 1169 1170
/* ifup */
static int fwnet_open(struct net_device *net)
{
	struct fwnet_device *dev = netdev_priv(net);
J
Jay Fenlason 已提交
1171 1172
	int ret;

S
Stefan Richter 已提交
1173 1174
	if (dev->broadcast_state == FWNET_BROADCAST_ERROR) {
		ret = fwnet_broadcast_start(dev);
J
Jay Fenlason 已提交
1175 1176 1177
		if (ret)
			return ret;
	}
S
Stefan Richter 已提交
1178 1179
	netif_start_queue(net);

J
Jay Fenlason 已提交
1180 1181 1182
	return 0;
}

S
Stefan Richter 已提交
1183 1184
/* ifdown */
static int fwnet_stop(struct net_device *net)
J
Jay Fenlason 已提交
1185
{
S
Stefan Richter 已提交
1186 1187 1188
	netif_stop_queue(net);

	/* Deallocate iso context for use by other applications? */
J
Jay Fenlason 已提交
1189 1190 1191 1192

	return 0;
}

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

S
Stefan Richter 已提交
1206
	ptask = kmem_cache_alloc(fwnet_packet_task_cache, GFP_ATOMIC);
J
Jay Fenlason 已提交
1207 1208 1209 1210 1211 1212 1213 1214
	if (ptask == NULL)
		goto fail;

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

	/*
S
Stefan Richter 已提交
1215
	 * Make a copy of the driver-specific header.
J
Jay Fenlason 已提交
1216 1217 1218 1219 1220 1221 1222 1223
	 * 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;

1224 1225 1226
	/* serialize access to peer, including peer->datagram_label */
	spin_lock_irqsave(&dev->lock, flags);

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

1238 1239 1240 1241
		ptask->fifo_addr   = FWNET_NO_FIFO_ADDR;
		ptask->generation  = 0;
		ptask->dest_node   = IEEE1394_ALL_NODES;
		ptask->speed       = SCODE_100;
J
Jay Fenlason 已提交
1242
	} else {
S
Stefan Richter 已提交
1243
		__be64 guid = get_unaligned((__be64 *)hdr_buf.h_dest);
J
Jay Fenlason 已提交
1244 1245
		u8 generation;

S
Stefan Richter 已提交
1246
		peer = fwnet_peer_find_by_guid(dev, be64_to_cpu(guid));
1247 1248
		if (!peer || peer->fifo == FWNET_NO_FIFO_ADDR)
			goto fail_unlock;
J
Jay Fenlason 已提交
1249

1250 1251 1252
		generation         = peer->generation;
		dest_node          = peer->node_id;
		max_payload        = peer->max_payload;
S
Stefan Richter 已提交
1253
		datagram_label_ptr = &peer->datagram_label;
J
Jay Fenlason 已提交
1254

1255 1256 1257 1258
		ptask->fifo_addr   = peer->fifo;
		ptask->generation  = generation;
		ptask->dest_node   = dest_node;
		ptask->speed       = peer->speed;
J
Jay Fenlason 已提交
1259 1260 1261 1262 1263 1264
	}

	/* 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 已提交
1265 1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278
		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 已提交
1279 1280 1281 1282 1283
	}

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

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

S
Stefan Richter 已提交
1294
		max_payload -= RFC2374_FRAG_OVERHEAD;
J
Jay Fenlason 已提交
1295
		datagram_label = (*datagram_label_ptr)++;
S
Stefan Richter 已提交
1296 1297
		fwnet_make_ff_hdr(&ptask->hdr, ntohs(proto), dg_size,
				  datagram_label);
J
Jay Fenlason 已提交
1298
		ptask->outstanding_pkts = DIV_ROUND_UP(dg_size, max_payload);
S
Stefan Richter 已提交
1299
		max_payload += RFC2374_FRAG_HDR_SIZE;
J
Jay Fenlason 已提交
1300
	}
1301 1302 1303

	spin_unlock_irqrestore(&dev->lock, flags);

J
Jay Fenlason 已提交
1304
	ptask->max_payload = max_payload;
S
Stefan Richter 已提交
1305 1306
	fwnet_send_packet(ptask);

J
Jay Fenlason 已提交
1307 1308
	return NETDEV_TX_OK;

1309 1310
 fail_unlock:
	spin_unlock_irqrestore(&dev->lock, flags);
J
Jay Fenlason 已提交
1311 1312
 fail:
	if (ptask)
S
Stefan Richter 已提交
1313
		kmem_cache_free(fwnet_packet_task_cache, ptask);
J
Jay Fenlason 已提交
1314 1315 1316 1317

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

S
Stefan Richter 已提交
1318 1319
	net->stats.tx_dropped++;
	net->stats.tx_errors++;
J
Jay Fenlason 已提交
1320 1321 1322 1323 1324 1325 1326 1327 1328 1329 1330

	/*
	 * 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 已提交
1331 1332
static int fwnet_change_mtu(struct net_device *net, int new_mtu)
{
J
Jay Fenlason 已提交
1333 1334 1335
	if (new_mtu < 68)
		return -EINVAL;

S
Stefan Richter 已提交
1336
	net->mtu = new_mtu;
J
Jay Fenlason 已提交
1337 1338 1339
	return 0;
}

S
Stefan Richter 已提交
1340 1341 1342 1343 1344
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 已提交
1345 1346
}

S
Stefan Richter 已提交
1347 1348
static struct ethtool_ops fwnet_ethtool_ops = {
	.get_drvinfo = fwnet_get_drvinfo,
J
Jay Fenlason 已提交
1349 1350
};

S
Stefan Richter 已提交
1351 1352 1353 1354 1355
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 已提交
1356 1357
};

S
Stefan Richter 已提交
1358 1359 1360 1361
static void fwnet_init_dev(struct net_device *net)
{
	net->header_ops		= &fwnet_header_ops;
	net->netdev_ops		= &fwnet_netdev_ops;
1362
	net->watchdog_timeo	= 2 * HZ;
S
Stefan Richter 已提交
1363 1364 1365 1366 1367
	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;
1368
	net->tx_queue_len	= 10;
S
Stefan Richter 已提交
1369
	SET_ETHTOOL_OPS(net, &fwnet_ethtool_ops);
J
Jay Fenlason 已提交
1370 1371
}

1372 1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 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
/* 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 已提交
1414 1415 1416 1417 1418 1419 1420
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 已提交
1421
	unsigned max_mtu;
1422 1423
	bool new_netdev;
	int ret;
J
Jay Fenlason 已提交
1424

1425
	mutex_lock(&fwnet_device_mutex);
J
Jay Fenlason 已提交
1426

1427 1428 1429 1430 1431
	dev = fwnet_dev_find(card);
	if (dev) {
		new_netdev = false;
		net = dev->netdev;
		goto have_dev;
J
Jay Fenlason 已提交
1432
	}
1433 1434

	new_netdev = true;
S
Stefan Richter 已提交
1435 1436
	net = alloc_netdev(sizeof(*dev), "firewire%d", fwnet_init_dev);
	if (net == NULL) {
1437
		ret = -ENOMEM;
J
Jay Fenlason 已提交
1438 1439 1440
		goto out;
	}

S
Stefan Richter 已提交
1441 1442
	SET_NETDEV_DEV(net, card->device);
	dev = netdev_priv(net);
J
Jay Fenlason 已提交
1443

S
Stefan Richter 已提交
1444 1445 1446 1447 1448
	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 已提交
1449

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

S
Stefan Richter 已提交
1452 1453 1454
	INIT_LIST_HEAD(&dev->packet_list);
	INIT_LIST_HEAD(&dev->broadcasted_list);
	INIT_LIST_HEAD(&dev->sent_list);
1455
	INIT_LIST_HEAD(&dev->peer_list);
J
Jay Fenlason 已提交
1456

S
Stefan Richter 已提交
1457
	dev->card = card;
1458
	dev->netdev = net;
J
Jay Fenlason 已提交
1459 1460 1461 1462 1463 1464

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

	/* Set our hardware address while we're at it */
S
Stefan Richter 已提交
1469 1470
	put_unaligned_be64(card->guid, net->dev_addr);
	put_unaligned_be64(~0ULL, net->broadcast);
1471 1472
	ret = register_netdev(net);
	if (ret) {
S
Stefan Richter 已提交
1473
		fw_error("Cannot register the driver\n");
J
Jay Fenlason 已提交
1474 1475 1476
		goto out;
	}

1477
	list_add_tail(&dev->dev_link, &fwnet_device_list);
S
Stefan Richter 已提交
1478 1479
	fw_notify("%s: IPv4 over FireWire on device %016llx\n",
		  net->name, (unsigned long long)card->guid);
1480 1481 1482 1483 1484 1485
 have_dev:
	ret = fwnet_add_peer(dev, unit, device);
	if (ret && new_netdev) {
		unregister_netdev(net);
		list_del(&dev->dev_link);
	}
J
Jay Fenlason 已提交
1486
 out:
1487
	if (ret && new_netdev)
S
Stefan Richter 已提交
1488 1489
		free_netdev(net);

1490 1491 1492 1493 1494 1495 1496 1497 1498 1499 1500 1501 1502 1503 1504 1505 1506
	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 已提交
1507 1508
}

S
Stefan Richter 已提交
1509 1510
static int fwnet_remove(struct device *_dev)
{
1511 1512
	struct fwnet_peer *peer = _dev->driver_data;
	struct fwnet_device *dev = peer->dev;
S
Stefan Richter 已提交
1513 1514 1515
	struct net_device *net;
	struct fwnet_packet_task *ptask, *pt_next;

1516
	mutex_lock(&fwnet_device_mutex);
J
Jay Fenlason 已提交
1517

1518
	fwnet_remove_peer(peer);
S
Stefan Richter 已提交
1519

1520 1521
	if (list_empty(&dev->peer_list)) {
		net = dev->netdev;
S
Stefan Richter 已提交
1522 1523 1524 1525 1526 1527 1528 1529 1530
		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 已提交
1531
		}
S
Stefan Richter 已提交
1532 1533 1534 1535
		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 已提交
1536
		}
S
Stefan Richter 已提交
1537 1538 1539 1540
		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 已提交
1541
		}
S
Stefan Richter 已提交
1542 1543 1544 1545
		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 已提交
1546
		}
S
Stefan Richter 已提交
1547
		free_netdev(net);
J
Jay Fenlason 已提交
1548
	}
S
Stefan Richter 已提交
1549

1550 1551
	mutex_unlock(&fwnet_device_mutex);

J
Jay Fenlason 已提交
1552 1553 1554
	return 0;
}

S
Stefan Richter 已提交
1555 1556 1557 1558 1559 1560 1561
/*
 * 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);
1562 1563
	struct fwnet_peer *peer = unit->device.driver_data;
	int generation;
J
Jay Fenlason 已提交
1564

1565 1566 1567 1568 1569 1570
	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 已提交
1571 1572
}

S
Stefan Richter 已提交
1573 1574 1575 1576 1577 1578 1579 1580 1581 1582 1583
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 已提交
1584
	.driver = {
S
Stefan Richter 已提交
1585 1586 1587 1588 1589
		.owner  = THIS_MODULE,
		.name   = "net",
		.bus    = &fw_bus_type,
		.probe  = fwnet_probe,
		.remove = fwnet_remove,
J
Jay Fenlason 已提交
1590
	},
S
Stefan Richter 已提交
1591 1592 1593 1594 1595 1596 1597 1598 1599 1600 1601 1602 1603 1604 1605 1606 1607 1608 1609 1610 1611 1612 1613 1614
	.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 已提交
1615 1616
};

S
Stefan Richter 已提交
1617 1618 1619 1620 1621 1622 1623
static int __init fwnet_init(void)
{
	int err;

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

S
Stefan Richter 已提交
1625 1626 1627 1628 1629 1630 1631 1632 1633 1634 1635 1636 1637 1638 1639 1640
	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 已提交
1641
}
S
Stefan Richter 已提交
1642
module_init(fwnet_init);
J
Jay Fenlason 已提交
1643

S
Stefan Richter 已提交
1644 1645 1646 1647 1648
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 已提交
1649
}
S
Stefan Richter 已提交
1650
module_exit(fwnet_cleanup);
J
Jay Fenlason 已提交
1651

S
Stefan Richter 已提交
1652 1653 1654 1655
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);