📄 w1.c
字号:
if (sl->family->fops && sl->family->fops->add_slave && ((err = sl->family->fops->add_slave(sl)) < 0)) { dev_err(&sl->dev, "sysfs file creation for [%s] failed. err=%d\n", sl->dev.bus_id, err); goto out_rem2; } list_add_tail(&sl->w1_slave_entry, &sl->master->slist); return 0;out_rem2: sysfs_remove_bin_file(&sl->dev.kobj, &w1_slave_attr_bin_id);out_rem1: device_remove_file(&sl->dev, &w1_slave_attr_name);out_unreg: device_unregister(&sl->dev); return err;}static int w1_attach_slave_device(struct w1_master *dev, struct w1_reg_num *rn){ struct w1_slave *sl; struct w1_family *f; int err; struct w1_netlink_msg msg; sl = kzalloc(sizeof(struct w1_slave), GFP_KERNEL); if (!sl) { dev_err(&dev->dev, "%s: failed to allocate new slave device.\n", __func__); return -ENOMEM; } sl->owner = THIS_MODULE; sl->master = dev; set_bit(W1_SLAVE_ACTIVE, (long *)&sl->flags); memset(&msg, 0, sizeof(msg)); memcpy(&sl->reg_num, rn, sizeof(sl->reg_num)); atomic_set(&sl->refcnt, 0); init_completion(&sl->released); spin_lock(&w1_flock); f = w1_family_registered(rn->family); if (!f) { f= &w1_default_family; dev_info(&dev->dev, "Family %x for %02x.%012llx.%02x is not registered.\n", rn->family, rn->family, (unsigned long long)rn->id, rn->crc); } __w1_family_get(f); spin_unlock(&w1_flock); sl->family = f; err = __w1_attach_slave_device(sl); if (err < 0) { dev_err(&dev->dev, "%s: Attaching %s failed.\n", __func__, sl->name); w1_family_put(sl->family); kfree(sl); return err; } sl->ttl = dev->slave_ttl; dev->slave_count++; memcpy(msg.id.id, rn, sizeof(msg.id)); msg.type = W1_SLAVE_ADD; w1_netlink_send(dev, &msg); return 0;}static void w1_slave_detach(struct w1_slave *sl){ struct w1_netlink_msg msg; dev_dbg(&sl->dev, "%s: detaching %s [%p].\n", __func__, sl->name, sl); list_del(&sl->w1_slave_entry); if (sl->family->fops && sl->family->fops->remove_slave) sl->family->fops->remove_slave(sl); memset(&msg, 0, sizeof(msg)); memcpy(msg.id.id, &sl->reg_num, sizeof(msg.id)); msg.type = W1_SLAVE_REMOVE; w1_netlink_send(sl->master, &msg); sysfs_remove_bin_file(&sl->dev.kobj, &w1_slave_attr_bin_id); device_remove_file(&sl->dev, &w1_slave_attr_name); device_unregister(&sl->dev); wait_for_completion(&sl->released); kfree(sl);}static struct w1_master *w1_search_master(void *data){ struct w1_master *dev; int found = 0; mutex_lock(&w1_mlock); list_for_each_entry(dev, &w1_masters, w1_master_entry) { if (dev->bus_master->data == data) { found = 1; atomic_inc(&dev->refcnt); break; } } mutex_unlock(&w1_mlock); return (found)?dev:NULL;}struct w1_master *w1_search_master_id(u32 id){ struct w1_master *dev; int found = 0; mutex_lock(&w1_mlock); list_for_each_entry(dev, &w1_masters, w1_master_entry) { if (dev->id == id) { found = 1; atomic_inc(&dev->refcnt); break; } } mutex_unlock(&w1_mlock); return (found)?dev:NULL;}struct w1_slave *w1_search_slave(struct w1_reg_num *id){ struct w1_master *dev; struct w1_slave *sl = NULL; int found = 0; mutex_lock(&w1_mlock); list_for_each_entry(dev, &w1_masters, w1_master_entry) { mutex_lock(&dev->mutex); list_for_each_entry(sl, &dev->slist, w1_slave_entry) { if (sl->reg_num.family == id->family && sl->reg_num.id == id->id && sl->reg_num.crc == id->crc) { found = 1; atomic_inc(&dev->refcnt); atomic_inc(&sl->refcnt); break; } } mutex_unlock(&dev->mutex); if (found) break; } mutex_unlock(&w1_mlock); return (found)?sl:NULL;}void w1_reconnect_slaves(struct w1_family *f){ struct w1_master *dev; mutex_lock(&w1_mlock); list_for_each_entry(dev, &w1_masters, w1_master_entry) { dev_dbg(&dev->dev, "Reconnecting slaves in %s into new family %02x.\n", dev->name, f->fid); set_bit(W1_MASTER_NEED_RECONNECT, &dev->flags); } mutex_unlock(&w1_mlock);}static void w1_slave_found(void *data, u64 rn){ int slave_count; struct w1_slave *sl; struct list_head *ent; struct w1_reg_num *tmp; int family_found = 0; struct w1_master *dev; u64 rn_le = cpu_to_le64(rn); dev = w1_search_master(data); if (!dev) { printk(KERN_ERR "Failed to find w1 master device for data %p, " "it is impossible.\n", data); return; } tmp = (struct w1_reg_num *) &rn; slave_count = 0; list_for_each(ent, &dev->slist) { sl = list_entry(ent, struct w1_slave, w1_slave_entry); if (sl->reg_num.family == tmp->family && sl->reg_num.id == tmp->id && sl->reg_num.crc == tmp->crc) { set_bit(W1_SLAVE_ACTIVE, (long *)&sl->flags); break; } else if (sl->reg_num.family == tmp->family) { family_found = 1; break; } slave_count++; } if (slave_count == dev->slave_count && rn && ((rn >> 56) & 0xff) == w1_calc_crc8((u8 *)&rn_le, 7)) { w1_attach_slave_device(dev, tmp); } atomic_dec(&dev->refcnt);}/** * Performs a ROM Search & registers any devices found. * The 1-wire search is a simple binary tree search. * For each bit of the address, we read two bits and write one bit. * The bit written will put to sleep all devies that don't match that bit. * When the two reads differ, the direction choice is obvious. * When both bits are 0, we must choose a path to take. * When we can scan all 64 bits without having to choose a path, we are done. * * See "Application note 187 1-wire search algorithm" at www.maxim-ic.com * * @dev The master device to search * @cb Function to call when a device is found */void w1_search(struct w1_master *dev, u8 search_type, w1_slave_found_callback cb){ u64 last_rn, rn, tmp64; int i, slave_count = 0; int last_zero, last_device; int search_bit, desc_bit; u8 triplet_ret = 0; search_bit = 0; rn = last_rn = 0; last_device = 0; last_zero = -1; desc_bit = 64; while ( !last_device && (slave_count++ < dev->max_slave_count) ) { last_rn = rn; rn = 0; /* * Reset bus and all 1-wire device state machines * so they can respond to our requests. * * Return 0 - device(s) present, 1 - no devices present. */ if (w1_reset_bus(dev)) { dev_dbg(&dev->dev, "No devices present on the wire.\n"); break; } /* Start the search */ w1_write_8(dev, search_type); for (i = 0; i < 64; ++i) { /* Determine the direction/search bit */ if (i == desc_bit) search_bit = 1; /* took the 0 path last time, so take the 1 path */ else if (i > desc_bit) search_bit = 0; /* take the 0 path on the next branch */ else search_bit = ((last_rn >> i) & 0x1); /** Read two bits and write one bit */ triplet_ret = w1_triplet(dev, search_bit); /* quit if no device responded */ if ( (triplet_ret & 0x03) == 0x03 ) break; /* If both directions were valid, and we took the 0 path... */ if (triplet_ret == 0) last_zero = i; /* extract the direction taken & update the device number */ tmp64 = (triplet_ret >> 2); rn |= (tmp64 << i); } if ( (triplet_ret & 0x03) != 0x03 ) { if ( (desc_bit == last_zero) || (last_zero < 0)) last_device = 1; desc_bit = last_zero; cb(dev->bus_master->data, rn); } }}static int w1_control(void *data){ struct w1_slave *sl, *sln; struct w1_master *dev, *n; int have_to_wait = 0; set_freezable(); while (!kthread_should_stop() || have_to_wait) { have_to_wait = 0; try_to_freeze(); msleep_interruptible(w1_control_timeout * 1000); list_for_each_entry_safe(dev, n, &w1_masters, w1_master_entry) { if (!kthread_should_stop() && !dev->flags) continue; /* * Little race: we can create thread but not set the flag. * Get a chance for external process to set flag up. */ if (!dev->initialized) { have_to_wait = 1; continue; } if (kthread_should_stop() || test_bit(W1_MASTER_NEED_EXIT, &dev->flags)) { set_bit(W1_MASTER_NEED_EXIT, &dev->flags); mutex_lock(&w1_mlock); list_del(&dev->w1_master_entry); mutex_unlock(&w1_mlock); mutex_lock(&dev->mutex); list_for_each_entry_safe(sl, sln, &dev->slist, w1_slave_entry) { w1_slave_detach(sl); } w1_destroy_master_attributes(dev); mutex_unlock(&dev->mutex); atomic_dec(&dev->refcnt); continue; } if (test_bit(W1_MASTER_NEED_RECONNECT, &dev->flags)) { dev_dbg(&dev->dev, "Reconnecting slaves in device %s.\n", dev->name); mutex_lock(&dev->mutex); list_for_each_entry_safe(sl, sln, &dev->slist, w1_slave_entry) { if (sl->family->fid == W1_FAMILY_DEFAULT) { struct w1_reg_num rn; memcpy(&rn, &sl->reg_num, sizeof(rn)); w1_slave_detach(sl); w1_attach_slave_device(dev, &rn); } } dev_dbg(&dev->dev, "Reconnecting slaves in device %s has been finished.\n", dev->name); clear_bit(W1_MASTER_NEED_RECONNECT, &dev->flags); mutex_unlock(&dev->mutex); } } } return 0;}void w1_search_process(struct w1_master *dev, u8 search_type){ struct w1_slave *sl, *sln; list_for_each_entry(sl, &dev->slist, w1_slave_entry) clear_bit(W1_SLAVE_ACTIVE, (long *)&sl->flags); w1_search_devices(dev, search_type, w1_slave_found); list_for_each_entry_safe(sl, sln, &dev->slist, w1_slave_entry) { if (!test_bit(W1_SLAVE_ACTIVE, (unsigned long *)&sl->flags) && !--sl->ttl) w1_slave_detach(sl); else if (test_bit(W1_SLAVE_ACTIVE, (unsigned long *)&sl->flags)) sl->ttl = dev->slave_ttl; } if (dev->search_count > 0) dev->search_count--;}int w1_process(void *data){ struct w1_master *dev = (struct w1_master *) data; while (!kthread_should_stop() && !test_bit(W1_MASTER_NEED_EXIT, &dev->flags)) { try_to_freeze(); msleep_interruptible(w1_timeout * 1000); if (kthread_should_stop() || test_bit(W1_MASTER_NEED_EXIT, &dev->flags)) break; if (!dev->initialized) continue; if (dev->search_count == 0) continue; mutex_lock(&dev->mutex); w1_search_process(dev, W1_SEARCH); mutex_unlock(&dev->mutex); } atomic_dec(&dev->refcnt); return 0;}static int w1_init(void){ int retval; printk(KERN_INFO "Driver for 1-wire Dallas network protocol.\n"); w1_init_netlink(); retval = bus_register(&w1_bus_type); if (retval) { printk(KERN_ERR "Failed to register bus. err=%d.\n", retval); goto err_out_exit_init; } retval = driver_register(&w1_master_driver); if (retval) { printk(KERN_ERR "Failed to register master driver. err=%d.\n", retval); goto err_out_bus_unregister; } retval = driver_register(&w1_slave_driver); if (retval) { printk(KERN_ERR "Failed to register master driver. err=%d.\n", retval); goto err_out_master_unregister; } w1_control_thread = kthread_run(w1_control, NULL, "w1_control"); if (IS_ERR(w1_control_thread)) { retval = PTR_ERR(w1_control_thread); printk(KERN_ERR "Failed to create control thread. err=%d\n", retval); goto err_out_slave_unregister; } return 0;err_out_slave_unregister: driver_unregister(&w1_slave_driver);err_out_master_unregister: driver_unregister(&w1_master_driver);err_out_bus_unregister: bus_unregister(&w1_bus_type);err_out_exit_init: return retval;}static void w1_fini(void){ struct w1_master *dev; list_for_each_entry(dev, &w1_masters, w1_master_entry) __w1_remove_master_device(dev); w1_fini_netlink(); kthread_stop(w1_control_thread); driver_unregister(&w1_slave_driver); driver_unregister(&w1_master_driver); bus_unregister(&w1_bus_type);}module_init(w1_init);module_exit(w1_fini);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -