Blame view

net/nfc/core.c 13.2 KB
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
  /*
   * Copyright (C) 2011 Instituto Nokia de Tecnologia
   *
   * Authors:
   *    Lauro Ramos Venancio <lauro.venancio@openbossa.org>
   *    Aloisio Almeida Jr <aloisio.almeida@openbossa.org>
   *
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License as published by
   * the Free Software Foundation; either version 2 of the License, or
   * (at your option) any later version.
   *
   * This program is distributed in the hope that it will be useful,
   * but WITHOUT ANY WARRANTY; without even the implied warranty of
   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
   * GNU General Public License for more details.
   *
   * You should have received a copy of the GNU General Public License
   * along with this program; if not, write to the
   * Free Software Foundation, Inc.,
   * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
   */
52858b51b   Samuel Ortiz   NFC: Add function...
23
  #define pr_fmt(fmt) KBUILD_MODNAME ": %s: " fmt, __func__
ed1e0ad88   Joe Perches   nfc: Use standard...
24

3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
25
26
27
28
  #include <linux/init.h>
  #include <linux/kernel.h>
  #include <linux/module.h>
  #include <linux/slab.h>
7c7cd3bfe   Samuel Ortiz   NFC: Add tx skb a...
29
  #include <linux/nfc.h>
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
30
31
32
33
34
35
36
  
  #include "nfc.h"
  
  #define VERSION "0.1"
  
  int nfc_devlist_generation;
  DEFINE_MUTEX(nfc_devlist_mutex);
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
37
  /**
8b3fe7b59   Ilan Elias   NFC: Add dev_up a...
38
39
40
41
42
43
44
45
46
   * nfc_dev_up - turn on the NFC device
   *
   * @dev: The nfc device to be turned on
   *
   * The device remains up until the nfc_dev_down function is called.
   */
  int nfc_dev_up(struct nfc_dev *dev)
  {
  	int rc = 0;
20c239c13   Joe Perches   nfc: Convert nfc_...
47
48
  	pr_debug("dev_name=%s
  ", dev_name(&dev->dev));
8b3fe7b59   Ilan Elias   NFC: Add dev_up a...
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
  
  	device_lock(&dev->dev);
  
  	if (!device_is_registered(&dev->dev)) {
  		rc = -ENODEV;
  		goto error;
  	}
  
  	if (dev->dev_up) {
  		rc = -EALREADY;
  		goto error;
  	}
  
  	if (dev->ops->dev_up)
  		rc = dev->ops->dev_up(dev);
  
  	if (!rc)
  		dev->dev_up = true;
  
  error:
  	device_unlock(&dev->dev);
  	return rc;
  }
  
  /**
   * nfc_dev_down - turn off the NFC device
   *
   * @dev: The nfc device to be turned off
   */
  int nfc_dev_down(struct nfc_dev *dev)
  {
  	int rc = 0;
20c239c13   Joe Perches   nfc: Convert nfc_...
81
82
  	pr_debug("dev_name=%s
  ", dev_name(&dev->dev));
8b3fe7b59   Ilan Elias   NFC: Add dev_up a...
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
  
  	device_lock(&dev->dev);
  
  	if (!device_is_registered(&dev->dev)) {
  		rc = -ENODEV;
  		goto error;
  	}
  
  	if (!dev->dev_up) {
  		rc = -EALREADY;
  		goto error;
  	}
  
  	if (dev->polling || dev->remote_activated) {
  		rc = -EBUSY;
  		goto error;
  	}
  
  	if (dev->ops->dev_down)
  		dev->ops->dev_down(dev);
  
  	dev->dev_up = false;
  
  error:
  	device_unlock(&dev->dev);
  	return rc;
  }
  
  /**
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
112
113
114
115
116
117
118
119
120
121
122
   * nfc_start_poll - start polling for nfc targets
   *
   * @dev: The nfc device that must start polling
   * @protocols: bitset of nfc protocols that must be used for polling
   *
   * The device remains polling for targets until a target is found or
   * the nfc_stop_poll function is called.
   */
  int nfc_start_poll(struct nfc_dev *dev, u32 protocols)
  {
  	int rc;
20c239c13   Joe Perches   nfc: Convert nfc_...
123
124
125
  	pr_debug("dev_name=%s protocols=0x%x
  ",
  		 dev_name(&dev->dev), protocols);
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
  
  	if (!protocols)
  		return -EINVAL;
  
  	device_lock(&dev->dev);
  
  	if (!device_is_registered(&dev->dev)) {
  		rc = -ENODEV;
  		goto error;
  	}
  
  	if (dev->polling) {
  		rc = -EBUSY;
  		goto error;
  	}
  
  	rc = dev->ops->start_poll(dev, protocols);
  	if (!rc)
  		dev->polling = true;
  
  error:
  	device_unlock(&dev->dev);
  	return rc;
  }
  
  /**
   * nfc_stop_poll - stop polling for nfc targets
   *
   * @dev: The nfc device that must stop polling
   */
  int nfc_stop_poll(struct nfc_dev *dev)
  {
  	int rc = 0;
20c239c13   Joe Perches   nfc: Convert nfc_...
159
160
  	pr_debug("dev_name=%s
  ", dev_name(&dev->dev));
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
  
  	device_lock(&dev->dev);
  
  	if (!device_is_registered(&dev->dev)) {
  		rc = -ENODEV;
  		goto error;
  	}
  
  	if (!dev->polling) {
  		rc = -EINVAL;
  		goto error;
  	}
  
  	dev->ops->stop_poll(dev);
  	dev->polling = false;
  
  error:
  	device_unlock(&dev->dev);
  	return rc;
  }
1ed28f610   Samuel Ortiz   NFC: Add a DEP li...
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
  int nfc_dep_link_up(struct nfc_dev *dev, int target_index,
  					u8 comm_mode, u8 rf_mode)
  {
  	int rc = 0;
  
  	pr_debug("dev_name=%s comm:%d rf:%d
  ",
  			dev_name(&dev->dev), comm_mode, rf_mode);
  
  	if (!dev->ops->dep_link_up)
  		return -EOPNOTSUPP;
  
  	device_lock(&dev->dev);
  
  	if (!device_is_registered(&dev->dev)) {
  		rc = -ENODEV;
  		goto error;
  	}
  
  	if (dev->dep_link_up == true) {
  		rc = -EALREADY;
  		goto error;
  	}
  
  	rc = dev->ops->dep_link_up(dev, target_index, comm_mode, rf_mode);
  
  error:
  	device_unlock(&dev->dev);
  	return rc;
  }
  
  int nfc_dep_link_down(struct nfc_dev *dev)
  {
  	int rc = 0;
  
  	pr_debug("dev_name=%s
  ", dev_name(&dev->dev));
  
  	if (!dev->ops->dep_link_down)
  		return -EOPNOTSUPP;
  
  	device_lock(&dev->dev);
  
  	if (!device_is_registered(&dev->dev)) {
  		rc = -ENODEV;
  		goto error;
  	}
  
  	if (dev->dep_link_up == false) {
  		rc = -EALREADY;
  		goto error;
  	}
  
  	if (dev->dep_rf_mode == NFC_RF_TARGET) {
  		rc = -EOPNOTSUPP;
  		goto error;
  	}
  
  	rc = dev->ops->dep_link_down(dev);
  	if (!rc) {
  		dev->dep_link_up = false;
d646960f7   Samuel Ortiz   NFC: Initial LLCP...
242
  		nfc_llcp_mac_is_down(dev);
1ed28f610   Samuel Ortiz   NFC: Add a DEP li...
243
244
245
246
247
248
249
250
251
252
253
254
255
  		nfc_genl_dep_link_down_event(dev);
  	}
  
  error:
  	device_unlock(&dev->dev);
  	return rc;
  }
  
  int nfc_dep_link_is_up(struct nfc_dev *dev, u32 target_idx,
  					u8 comm_mode, u8 rf_mode)
  {
  	dev->dep_link_up = true;
  	dev->dep_rf_mode = rf_mode;
d646960f7   Samuel Ortiz   NFC: Initial LLCP...
256
  	nfc_llcp_mac_is_up(dev, target_idx, comm_mode, rf_mode);
1ed28f610   Samuel Ortiz   NFC: Add a DEP li...
257
258
259
  	return nfc_genl_dep_link_up_event(dev, target_idx, comm_mode, rf_mode);
  }
  EXPORT_SYMBOL(nfc_dep_link_is_up);
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
260
261
262
263
264
265
266
267
268
269
  /**
   * nfc_activate_target - prepare the target for data exchange
   *
   * @dev: The nfc device that found the target
   * @target_idx: index of the target that must be activated
   * @protocol: nfc protocol that will be used for data exchange
   */
  int nfc_activate_target(struct nfc_dev *dev, u32 target_idx, u32 protocol)
  {
  	int rc;
20c239c13   Joe Perches   nfc: Convert nfc_...
270
271
272
  	pr_debug("dev_name=%s target_idx=%u protocol=%u
  ",
  		 dev_name(&dev->dev), target_idx, protocol);
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
273
274
275
276
277
278
279
280
281
  
  	device_lock(&dev->dev);
  
  	if (!device_is_registered(&dev->dev)) {
  		rc = -ENODEV;
  		goto error;
  	}
  
  	rc = dev->ops->activate_target(dev, target_idx, protocol);
8b3fe7b59   Ilan Elias   NFC: Add dev_up a...
282
283
  	if (!rc)
  		dev->remote_activated = true;
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
  
  error:
  	device_unlock(&dev->dev);
  	return rc;
  }
  
  /**
   * nfc_deactivate_target - deactivate a nfc target
   *
   * @dev: The nfc device that found the target
   * @target_idx: index of the target that must be deactivated
   */
  int nfc_deactivate_target(struct nfc_dev *dev, u32 target_idx)
  {
  	int rc = 0;
20c239c13   Joe Perches   nfc: Convert nfc_...
299
300
301
  	pr_debug("dev_name=%s target_idx=%u
  ",
  		 dev_name(&dev->dev), target_idx);
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
302
303
304
305
306
307
308
309
310
  
  	device_lock(&dev->dev);
  
  	if (!device_is_registered(&dev->dev)) {
  		rc = -ENODEV;
  		goto error;
  	}
  
  	dev->ops->deactivate_target(dev, target_idx);
8b3fe7b59   Ilan Elias   NFC: Add dev_up a...
311
  	dev->remote_activated = false;
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
  
  error:
  	device_unlock(&dev->dev);
  	return rc;
  }
  
  /**
   * nfc_data_exchange - transceive data
   *
   * @dev: The nfc device that found the target
   * @target_idx: index of the target
   * @skb: data to be sent
   * @cb: callback called when the response is received
   * @cb_context: parameter for the callback function
   *
   * The user must wait for the callback before calling this function again.
   */
  int nfc_data_exchange(struct nfc_dev *dev, u32 target_idx,
  					struct sk_buff *skb,
  					data_exchange_cb_t cb,
  					void *cb_context)
  {
  	int rc;
20c239c13   Joe Perches   nfc: Convert nfc_...
335
336
337
  	pr_debug("dev_name=%s target_idx=%u skb->len=%u
  ",
  		 dev_name(&dev->dev), target_idx, skb->len);
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
  
  	device_lock(&dev->dev);
  
  	if (!device_is_registered(&dev->dev)) {
  		rc = -ENODEV;
  		kfree_skb(skb);
  		goto error;
  	}
  
  	rc = dev->ops->data_exchange(dev, target_idx, skb, cb, cb_context);
  
  error:
  	device_unlock(&dev->dev);
  	return rc;
  }
541d920b0   Samuel Ortiz   NFC: Set and get ...
353
354
355
356
357
358
359
360
  int nfc_set_remote_general_bytes(struct nfc_dev *dev, u8 *gb, u8 gb_len)
  {
  	pr_debug("dev_name=%s gb_len=%d
  ",
  			dev_name(&dev->dev), gb_len);
  
  	if (gb_len > NFC_MAX_GT_LEN)
  		return -EINVAL;
d646960f7   Samuel Ortiz   NFC: Initial LLCP...
361
  	return nfc_llcp_set_remote_gb(dev, gb, gb_len);
541d920b0   Samuel Ortiz   NFC: Set and get ...
362
363
364
365
366
  }
  EXPORT_SYMBOL(nfc_set_remote_general_bytes);
  
  u8 *nfc_get_local_general_bytes(struct nfc_dev *dev, u8 *gt_len)
  {
d646960f7   Samuel Ortiz   NFC: Initial LLCP...
367
  	return nfc_llcp_general_bytes(dev, gt_len);
541d920b0   Samuel Ortiz   NFC: Set and get ...
368
369
  }
  EXPORT_SYMBOL(nfc_get_local_general_bytes);
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
370
  /**
7c7cd3bfe   Samuel Ortiz   NFC: Add tx skb a...
371
   * nfc_alloc_send_skb - allocate a skb for data exchange responses
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
372
373
374
375
   *
   * @size: size to allocate
   * @gfp: gfp flags
   */
7c7cd3bfe   Samuel Ortiz   NFC: Add tx skb a...
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
  struct sk_buff *nfc_alloc_send_skb(struct nfc_dev *dev, struct sock *sk,
  					unsigned int flags, unsigned int size,
  					unsigned int *err)
  {
  	struct sk_buff *skb;
  	unsigned int total_size;
  
  	total_size = size +
  		dev->tx_headroom + dev->tx_tailroom + NFC_HEADER_SIZE;
  
  	skb = sock_alloc_send_skb(sk, total_size, flags & MSG_DONTWAIT, err);
  	if (skb)
  		skb_reserve(skb, dev->tx_headroom + NFC_HEADER_SIZE);
  
  	return skb;
  }
  
  /**
   * nfc_alloc_recv_skb - allocate a skb for data exchange responses
   *
   * @size: size to allocate
   * @gfp: gfp flags
   */
  struct sk_buff *nfc_alloc_recv_skb(unsigned int size, gfp_t gfp)
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
400
401
402
403
404
405
406
407
408
409
410
411
  {
  	struct sk_buff *skb;
  	unsigned int total_size;
  
  	total_size = size + 1;
  	skb = alloc_skb(total_size, gfp);
  
  	if (skb)
  		skb_reserve(skb, 1);
  
  	return skb;
  }
7c7cd3bfe   Samuel Ortiz   NFC: Add tx skb a...
412
  EXPORT_SYMBOL(nfc_alloc_recv_skb);
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
413

4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
  /**
   * nfc_targets_found - inform that targets were found
   *
   * @dev: The nfc device that found the targets
   * @targets: array of nfc targets found
   * @ntargets: targets array size
   *
   * The device driver must call this function when one or many nfc targets
   * are found. After calling this function, the device driver must stop
   * polling for targets.
   */
  int nfc_targets_found(struct nfc_dev *dev, struct nfc_target *targets,
  							int n_targets)
  {
  	int i;
20c239c13   Joe Perches   nfc: Convert nfc_...
429
430
  	pr_debug("dev_name=%s n_targets=%d
  ", dev_name(&dev->dev), n_targets);
4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
  
  	dev->polling = false;
  
  	for (i = 0; i < n_targets; i++)
  		targets[i].idx = dev->target_idx++;
  
  	spin_lock_bh(&dev->targets_lock);
  
  	dev->targets_generation++;
  
  	kfree(dev->targets);
  	dev->targets = kmemdup(targets, n_targets * sizeof(struct nfc_target),
  								GFP_ATOMIC);
  
  	if (!dev->targets) {
  		dev->n_targets = 0;
  		spin_unlock_bh(&dev->targets_lock);
  		return -ENOMEM;
  	}
  
  	dev->n_targets = n_targets;
  	spin_unlock_bh(&dev->targets_lock);
  
  	nfc_genl_targets_found(dev);
  
  	return 0;
  }
  EXPORT_SYMBOL(nfc_targets_found);
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
459
460
461
  static void nfc_release(struct device *d)
  {
  	struct nfc_dev *dev = to_nfc_dev(d);
20c239c13   Joe Perches   nfc: Convert nfc_...
462
463
  	pr_debug("dev_name=%s
  ", dev_name(&dev->dev));
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
464

4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
465
466
  	nfc_genl_data_exit(&dev->genl_data);
  	kfree(dev->targets);
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
  	kfree(dev);
  }
  
  struct class nfc_class = {
  	.name = "nfc",
  	.dev_release = nfc_release,
  };
  EXPORT_SYMBOL(nfc_class);
  
  static int match_idx(struct device *d, void *data)
  {
  	struct nfc_dev *dev = to_nfc_dev(d);
  	unsigned *idx = data;
  
  	return dev->idx == *idx;
  }
  
  struct nfc_dev *nfc_get_device(unsigned idx)
  {
  	struct device *d;
  
  	d = class_find_device(&nfc_class, NULL, &idx, match_idx);
  	if (!d)
  		return NULL;
  
  	return to_nfc_dev(d);
  }
  
  /**
   * nfc_allocate_device - allocate a new nfc device
   *
   * @ops: device operations
   * @supported_protocols: NFC protocols supported by the device
   */
  struct nfc_dev *nfc_allocate_device(struct nfc_ops *ops,
e8753043f   Samuel Ortiz   NFC: Reserve tx h...
502
503
504
  					u32 supported_protocols,
  					int tx_headroom,
  					int tx_tailroom)
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
  {
  	static atomic_t dev_no = ATOMIC_INIT(0);
  	struct nfc_dev *dev;
  
  	if (!ops->start_poll || !ops->stop_poll || !ops->activate_target ||
  		!ops->deactivate_target || !ops->data_exchange)
  		return NULL;
  
  	if (!supported_protocols)
  		return NULL;
  
  	dev = kzalloc(sizeof(struct nfc_dev), GFP_KERNEL);
  	if (!dev)
  		return NULL;
  
  	dev->dev.class = &nfc_class;
  	dev->idx = atomic_inc_return(&dev_no) - 1;
  	dev_set_name(&dev->dev, "nfc%d", dev->idx);
  	device_initialize(&dev->dev);
  
  	dev->ops = ops;
  	dev->supported_protocols = supported_protocols;
e8753043f   Samuel Ortiz   NFC: Reserve tx h...
527
528
  	dev->tx_headroom = tx_headroom;
  	dev->tx_tailroom = tx_tailroom;
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
529

4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
530
531
532
533
534
  	spin_lock_init(&dev->targets_lock);
  	nfc_genl_data_init(&dev->genl_data);
  
  	/* first generation must not be 0 */
  	dev->targets_generation = 1;
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
535
536
537
538
539
540
541
542
543
544
545
546
  	return dev;
  }
  EXPORT_SYMBOL(nfc_allocate_device);
  
  /**
   * nfc_register_device - register a nfc device in the nfc subsystem
   *
   * @dev: The nfc device to register
   */
  int nfc_register_device(struct nfc_dev *dev)
  {
  	int rc;
20c239c13   Joe Perches   nfc: Convert nfc_...
547
548
  	pr_debug("dev_name=%s
  ", dev_name(&dev->dev));
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
549
550
551
552
553
  
  	mutex_lock(&nfc_devlist_mutex);
  	nfc_devlist_generation++;
  	rc = device_add(&dev->dev);
  	mutex_unlock(&nfc_devlist_mutex);
4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
554
555
  	if (rc < 0)
  		return rc;
d646960f7   Samuel Ortiz   NFC: Initial LLCP...
556
557
558
559
  	rc = nfc_llcp_register_device(dev);
  	if (rc)
  		pr_err("Could not register llcp device
  ");
4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
560
561
  	rc = nfc_genl_device_added(dev);
  	if (rc)
20c239c13   Joe Perches   nfc: Convert nfc_...
562
563
564
  		pr_debug("The userspace won't be notified that the device %s was added
  ",
  			 dev_name(&dev->dev));
4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
565
566
  
  	return 0;
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
567
568
569
570
571
572
573
574
575
576
  }
  EXPORT_SYMBOL(nfc_register_device);
  
  /**
   * nfc_unregister_device - unregister a nfc device in the nfc subsystem
   *
   * @dev: The nfc device to unregister
   */
  void nfc_unregister_device(struct nfc_dev *dev)
  {
4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
577
  	int rc;
20c239c13   Joe Perches   nfc: Convert nfc_...
578
579
  	pr_debug("dev_name=%s
  ", dev_name(&dev->dev));
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
580
581
582
583
584
585
586
587
588
589
590
  
  	mutex_lock(&nfc_devlist_mutex);
  	nfc_devlist_generation++;
  
  	/* lock to avoid unregistering a device while an operation
  	   is in progress */
  	device_lock(&dev->dev);
  	device_del(&dev->dev);
  	device_unlock(&dev->dev);
  
  	mutex_unlock(&nfc_devlist_mutex);
4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
591

d646960f7   Samuel Ortiz   NFC: Initial LLCP...
592
  	nfc_llcp_unregister_device(dev);
4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
593
594
  	rc = nfc_genl_device_removed(dev);
  	if (rc)
20c239c13   Joe Perches   nfc: Convert nfc_...
595
596
597
  		pr_debug("The userspace won't be notified that the device %s was removed
  ",
  			 dev_name(&dev->dev));
4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
598

3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
599
600
601
602
603
  }
  EXPORT_SYMBOL(nfc_unregister_device);
  
  static int __init nfc_init(void)
  {
4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
604
  	int rc;
ed1e0ad88   Joe Perches   nfc: Use standard...
605
606
  	pr_info("NFC Core ver %s
  ", VERSION);
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
607

4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
608
609
610
611
612
613
614
615
616
617
  	rc = class_register(&nfc_class);
  	if (rc)
  		return rc;
  
  	rc = nfc_genl_init();
  	if (rc)
  		goto err_genl;
  
  	/* the first generation must not be 0 */
  	nfc_devlist_generation = 1;
23b7869c0   Lauro Ramos Venancio   NFC: add the NFC ...
618
619
620
  	rc = rawsock_init();
  	if (rc)
  		goto err_rawsock;
d646960f7   Samuel Ortiz   NFC: Initial LLCP...
621
622
623
  	rc = nfc_llcp_init();
  	if (rc)
  		goto err_llcp_sock;
c7fe3b52c   Aloisio Almeida Jr   NFC: add NFC sock...
624
625
626
  	rc = af_nfc_init();
  	if (rc)
  		goto err_af_nfc;
4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
627
  	return 0;
c7fe3b52c   Aloisio Almeida Jr   NFC: add NFC sock...
628
  err_af_nfc:
d646960f7   Samuel Ortiz   NFC: Initial LLCP...
629
630
  	nfc_llcp_exit();
  err_llcp_sock:
23b7869c0   Lauro Ramos Venancio   NFC: add the NFC ...
631
632
  	rawsock_exit();
  err_rawsock:
c7fe3b52c   Aloisio Almeida Jr   NFC: add NFC sock...
633
  	nfc_genl_exit();
4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
634
635
636
  err_genl:
  	class_unregister(&nfc_class);
  	return rc;
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
637
638
639
640
  }
  
  static void __exit nfc_exit(void)
  {
c7fe3b52c   Aloisio Almeida Jr   NFC: add NFC sock...
641
  	af_nfc_exit();
d646960f7   Samuel Ortiz   NFC: Initial LLCP...
642
  	nfc_llcp_exit();
23b7869c0   Lauro Ramos Venancio   NFC: add the NFC ...
643
  	rawsock_exit();
4d12b8b12   Lauro Ramos Venancio   NFC: add nfc gene...
644
  	nfc_genl_exit();
3e256b8f8   Lauro Ramos Venancio   NFC: add nfc subs...
645
646
647
648
649
650
651
652
653
654
  	class_unregister(&nfc_class);
  }
  
  subsys_initcall(nfc_init);
  module_exit(nfc_exit);
  
  MODULE_AUTHOR("Lauro Ramos Venancio <lauro.venancio@openbossa.org>");
  MODULE_DESCRIPTION("NFC Core ver " VERSION);
  MODULE_VERSION(VERSION);
  MODULE_LICENSE("GPL");