static int __init init_binder_device(const char *name)
{
        int ret;
        struct binder_device *binder_device;
        binder_device = kzalloc(sizeof(*binder_device), GFP_KERNEL);
        if (!binder_device)
                return -ENOMEM;
        binder_device->miscdev.fops = &binder_fops;
        binder_device->miscdev.minor = MISC_DYNAMIC_MINOR;
        binder_device->miscdev.name = name;
        binder_device->context.binder_context_mgr_uid = INVALID_UID;
        binder_device->context.name = name;
        ret = misc_register(&binder_device->miscdev); 
        if (ret < 0) {
                kfree(binder_device);
                return ret;
        }
        hlist_add_head(&binder_device->hlist, &binder_devices); 
        return ret;
}
static int __init binder_init(void)
{
        int ret;
        char *device_name, *device_names;
        struct binder_device *device;
        struct hlist_node *tmp;
        binder_deferred_workqueue = create_singlethread_workqueue("binder"); 
        if (!binder_deferred_workqueue)
                return -ENOMEM;
        binder_debugfs_dir_entry_root = debugfs_create_dir("binder", NULL); 
        if (binder_debugfs_dir_entry_root) 
                binder_debugfs_dir_entry_proc = debugfs_create_dir("proc",
                                                 binder_debugfs_dir_entry_root);
        if (binder_debugfs_dir_entry_root) {
                debugfs_create_file("state",
                                    S_IRUGO,
                                    binder_debugfs_dir_entry_root,
                                    NULL,
                                    &binder_state_fops);
                debugfs_create_file("stats",
                                    S_IRUGO,
                                    binder_debugfs_dir_entry_root,
                                    NULL,
                                    &binder_stats_fops);
                debugfs_create_file("transactions",
                                    S_IRUGO,
                                    binder_debugfs_dir_entry_root,
                                    NULL,
                                    &binder_transactions_fops);
                debugfs_create_file("transaction_log",
                                    S_IRUGO,
                                    binder_debugfs_dir_entry_root,
                                    &binder_transaction_log,
                                    &binder_transaction_log_fops);
                debugfs_create_file("failed_transaction_log",
                                    S_IRUGO,
                                    binder_debugfs_dir_entry_root,
                                    &binder_transaction_log_failed,
                                    &binder_transaction_log_fops);
        }
        
        device_names = kzalloc(strlen(binder_devices_param) + 1, GFP_KERNEL);
        if (!device_names) {
                ret = -ENOMEM;
                goto err_alloc_device_names_failed;
        }
        strcpy(device_names, binder_devices_param);
        while ((device_name = strsep(&device_names, ","))) {
                ret = init_binder_device(device_name); 
                if (ret)
                        goto err_init_binder_device_failed;
        }
        return ret;
err_init_binder_device_failed:
        hlist_for_each_entry_safe(device, tmp, &binder_devices, hlist) {
                misc_deregister(&device->miscdev);
                hlist_del(&device->hlist);
                kfree(device);
        }
err_alloc_device_names_failed:
        debugfs_remove_recursive(binder_debugfs_dir_entry_root); 
        destroy_workqueue(binder_deferred_workqueue);
        return ret;
}