Example #1
0
static int ixp4xx_flash_probe(struct platform_device *dev)
{
    struct flash_platform_data *plat = dev->dev.platform_data;
    struct ixp4xx_flash_info *info;
    const char *part_type = NULL;
    int nr_parts = 0;
    int err = -1;

    if (!plat)
        return -ENODEV;

    if (plat->init) {
        err = plat->init();
        if (err)
            return err;
    }

    info = kzalloc(sizeof(struct ixp4xx_flash_info), GFP_KERNEL);
    if(!info) {
        err = -ENOMEM;
        goto Error;
    }

    platform_set_drvdata(dev, info);

    /*
     * Tell the MTD layer we're not 1:1 mapped so that it does
     * not attempt to do a direct access on us.
     */
    info->map.phys = NO_XIP;
    info->map.size = resource_size(dev->resource);

    /*
     * We only support 16-bit accesses for now. If and when
     * any board use 8-bit access, we'll fixup the driver to
     * handle that.
     */
    info->map.bankwidth = 2;
    info->map.name = dev_name(&dev->dev);
    info->map.read = ixp4xx_read16;
    info->map.write = ixp4xx_probe_write16;
    info->map.copy_from = ixp4xx_copy_from;

    info->res = request_mem_region(dev->resource->start,
                                   resource_size(dev->resource),
                                   "IXP4XXFlash");
    if (!info->res) {
        printk(KERN_ERR "IXP4XXFlash: Could not reserve memory region\n");
        err = -ENOMEM;
        goto Error;
    }

    info->map.virt = ioremap(dev->resource->start,
                             resource_size(dev->resource));
    if (!info->map.virt) {
        printk(KERN_ERR "IXP4XXFlash: Failed to ioremap region\n");
        err = -EIO;
        goto Error;
    }

    info->mtd = do_map_probe(plat->map_name, &info->map);
    if (!info->mtd) {
        printk(KERN_ERR "IXP4XXFlash: map_probe failed\n");
        err = -ENXIO;
        goto Error;
    }
    info->mtd->owner = THIS_MODULE;

    /* Use the fast version */
    info->map.write = ixp4xx_write16;

    nr_parts = parse_mtd_partitions(info->mtd, probes, &info->partitions,
                                    dev->resource->start);
    if (nr_parts > 0) {
        part_type = "dynamic";
    } else {
        info->partitions = plat->parts;
        nr_parts = plat->nr_parts;
        part_type = "static";
    }
    if (nr_parts == 0)
        printk(KERN_NOTICE "IXP4xx flash: no partition info "
               "available, registering whole flash\n");
    else
        printk(KERN_NOTICE "IXP4xx flash: using %s partition "
               "definition\n", part_type);

    err = mtd_device_register(info->mtd, info->partitions, nr_parts);
    if (err)
        printk(KERN_ERR "Could not parse partitions\n");

    if (err)
        goto Error;

    return 0;

Error:
    ixp4xx_flash_remove(dev);
    return err;
}
Example #2
0
static int ixp4xx_flash_probe(struct device *_dev)
{
	struct platform_device *dev = to_platform_device(_dev);
	struct flash_platform_data *plat = dev->dev.platform_data;
	struct ixp4xx_flash_info *info;
	int err = -1;

	if (!plat)
		return -ENODEV;

	if (plat->init) {
		err = plat->init();
		if (err)
			return err;
	}

	info = kmalloc(sizeof(struct ixp4xx_flash_info), GFP_KERNEL);
	if(!info) {
		err = -ENOMEM;
		goto Error;
	}	
	memzero(info, sizeof(struct ixp4xx_flash_info));

	dev_set_drvdata(&dev->dev, info);

	/* 
	 * Enable flash write 
	 * TODO: Move this out to board specific code
	 */
	*IXP4XX_EXP_CS0 |= IXP4XX_FLASH_WRITABLE;

	/*
	 * Tell the MTD layer we're not 1:1 mapped so that it does
	 * not attempt to do a direct access on us.
	 */
	info->map.phys = NO_XIP;
	info->map.size = dev->resource->end - dev->resource->start + 1;

	/*
	 * We only support 16-bit accesses for now. If and when
	 * any board use 8-bit access, we'll fixup the driver to
	 * handle that.
	 */
	info->map.bankwidth = 2;
	info->map.name = dev->dev.bus_id;
	info->map.read = ixp4xx_read16,
	info->map.write = ixp4xx_probe_write16,
	info->map.copy_from = ixp4xx_copy_from,

	info->res = request_mem_region(dev->resource->start, 
			dev->resource->end - dev->resource->start + 1, 
			"IXP4XXFlash");
	if (!info->res) {
		printk(KERN_ERR "IXP4XXFlash: Could not reserve memory region\n");
		err = -ENOMEM;
		goto Error;
	}

	info->map.map_priv_1 = ioremap(dev->resource->start,
			    dev->resource->end - dev->resource->start + 1);
	if (!info->map.map_priv_1) {
		printk(KERN_ERR "IXP4XXFlash: Failed to ioremap region\n");
		err = -EIO;
		goto Error;
	}

	info->mtd = do_map_probe(plat->map_name, &info->map);
	if (!info->mtd) {
		printk(KERN_ERR "IXP4XXFlash: map_probe failed\n");
		err = -ENXIO;
		goto Error;
	}
	info->mtd->owner = THIS_MODULE;
	
	/* Use the fast version */
	info->map.write = ixp4xx_write16,

	err = parse_mtd_partitions(info->mtd, probes, &info->partitions, 0);
	if (err > 0) {
		err = add_mtd_partitions(info->mtd, info->partitions, err);
		if(err)
			printk(KERN_ERR "Could not parse partitions\n");
	}

	if (err)
		goto Error;

	return 0;

Error:
	ixp4xx_flash_remove(_dev);
	return err;
}
Example #3
0
File: ixp4xx.c Project: 3null/linux
static int ixp4xx_flash_probe(struct platform_device *dev)
{
	struct flash_platform_data *plat = dev_get_platdata(&dev->dev);
	struct ixp4xx_flash_info *info;
	struct mtd_part_parser_data ppdata = {
		.origin = dev->resource->start,
	};
	int err = -1;

	if (!plat)
		return -ENODEV;

	if (plat->init) {
		err = plat->init();
		if (err)
			return err;
	}

	info = devm_kzalloc(&dev->dev, sizeof(struct ixp4xx_flash_info),
			    GFP_KERNEL);
	if(!info) {
		err = -ENOMEM;
		goto Error;
	}

	platform_set_drvdata(dev, info);

	/*
	 * Tell the MTD layer we're not 1:1 mapped so that it does
	 * not attempt to do a direct access on us.
	 */
	info->map.phys = NO_XIP;
	info->map.size = resource_size(dev->resource);

	/*
	 * We only support 16-bit accesses for now. If and when
	 * any board use 8-bit access, we'll fixup the driver to
	 * handle that.
	 */
	info->map.bankwidth = 2;
	info->map.name = dev_name(&dev->dev);
	info->map.read = ixp4xx_read16;
	info->map.write = ixp4xx_probe_write16;
	info->map.copy_from = ixp4xx_copy_from;

	info->map.virt = devm_ioremap_resource(&dev->dev, dev->resource);
	if (IS_ERR(info->map.virt)) {
		err = PTR_ERR(info->map.virt);
		goto Error;
	}

	info->mtd = do_map_probe(plat->map_name, &info->map);
	if (!info->mtd) {
		printk(KERN_ERR "IXP4XXFlash: map_probe failed\n");
		err = -ENXIO;
		goto Error;
	}
	info->mtd->owner = THIS_MODULE;

	/* Use the fast version */
	info->map.write = ixp4xx_write16;

	err = mtd_device_parse_register(info->mtd, probes, &ppdata,
			plat->parts, plat->nr_parts);
	if (err) {
		printk(KERN_ERR "Could not parse partitions\n");
		goto Error;
	}

	return 0;

Error:
	ixp4xx_flash_remove(dev);
	return err;
}

static struct platform_driver ixp4xx_flash_driver = {
	.probe		= ixp4xx_flash_probe,
	.remove		= ixp4xx_flash_remove,
	.driver		= {
		.name	= "IXP4XX-Flash",
		.owner	= THIS_MODULE,
	},
};

module_platform_driver(ixp4xx_flash_driver);

MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("MTD map driver for Intel IXP4xx systems");
MODULE_AUTHOR("Deepak Saxena");
MODULE_ALIAS("platform:IXP4XX-Flash");