diff options
Diffstat (limited to 'drivers')
144 files changed, 2034 insertions, 848 deletions
diff --git a/drivers/acpi/apei/cper.c b/drivers/acpi/apei/cper.c index 1e5d8a40101e..fefc2ca7cc3e 100644 --- a/drivers/acpi/apei/cper.c +++ b/drivers/acpi/apei/cper.c | |||
| @@ -405,7 +405,7 @@ int apei_estatus_check(const struct acpi_hest_generic_status *estatus) | |||
| 405 | return rc; | 405 | return rc; |
| 406 | data_len = estatus->data_length; | 406 | data_len = estatus->data_length; |
| 407 | gdata = (struct acpi_hest_generic_data *)(estatus + 1); | 407 | gdata = (struct acpi_hest_generic_data *)(estatus + 1); |
| 408 | while (data_len > sizeof(*gdata)) { | 408 | while (data_len >= sizeof(*gdata)) { |
| 409 | gedata_len = gdata->error_data_length; | 409 | gedata_len = gdata->error_data_length; |
| 410 | if (gedata_len > data_len - sizeof(*gdata)) | 410 | if (gedata_len > data_len - sizeof(*gdata)) |
| 411 | return -EINVAL; | 411 | return -EINVAL; |
diff --git a/drivers/acpi/pci_root.c b/drivers/acpi/pci_root.c index 0ac546d5e53f..5ff173066127 100644 --- a/drivers/acpi/pci_root.c +++ b/drivers/acpi/pci_root.c | |||
| @@ -646,6 +646,7 @@ static void handle_root_bridge_insertion(acpi_handle handle) | |||
| 646 | 646 | ||
| 647 | static void handle_root_bridge_removal(struct acpi_device *device) | 647 | static void handle_root_bridge_removal(struct acpi_device *device) |
| 648 | { | 648 | { |
| 649 | acpi_status status; | ||
| 649 | struct acpi_eject_event *ej_event; | 650 | struct acpi_eject_event *ej_event; |
| 650 | 651 | ||
| 651 | ej_event = kmalloc(sizeof(*ej_event), GFP_KERNEL); | 652 | ej_event = kmalloc(sizeof(*ej_event), GFP_KERNEL); |
| @@ -661,7 +662,9 @@ static void handle_root_bridge_removal(struct acpi_device *device) | |||
| 661 | ej_event->device = device; | 662 | ej_event->device = device; |
| 662 | ej_event->event = ACPI_NOTIFY_EJECT_REQUEST; | 663 | ej_event->event = ACPI_NOTIFY_EJECT_REQUEST; |
| 663 | 664 | ||
| 664 | acpi_bus_hot_remove_device(ej_event); | 665 | status = acpi_os_hotplug_execute(acpi_bus_hot_remove_device, ej_event); |
| 666 | if (ACPI_FAILURE(status)) | ||
| 667 | kfree(ej_event); | ||
| 665 | } | 668 | } |
| 666 | 669 | ||
| 667 | static void _handle_hotplug_event_root(struct work_struct *work) | 670 | static void _handle_hotplug_event_root(struct work_struct *work) |
| @@ -676,8 +679,9 @@ static void _handle_hotplug_event_root(struct work_struct *work) | |||
| 676 | handle = hp_work->handle; | 679 | handle = hp_work->handle; |
| 677 | type = hp_work->type; | 680 | type = hp_work->type; |
| 678 | 681 | ||
| 679 | root = acpi_pci_find_root(handle); | 682 | acpi_scan_lock_acquire(); |
| 680 | 683 | ||
| 684 | root = acpi_pci_find_root(handle); | ||
| 681 | acpi_get_name(handle, ACPI_FULL_PATHNAME, &buffer); | 685 | acpi_get_name(handle, ACPI_FULL_PATHNAME, &buffer); |
| 682 | 686 | ||
| 683 | switch (type) { | 687 | switch (type) { |
| @@ -711,6 +715,7 @@ static void _handle_hotplug_event_root(struct work_struct *work) | |||
| 711 | break; | 715 | break; |
| 712 | } | 716 | } |
| 713 | 717 | ||
| 718 | acpi_scan_lock_release(); | ||
| 714 | kfree(hp_work); /* allocated in handle_hotplug_event_bridge */ | 719 | kfree(hp_work); /* allocated in handle_hotplug_event_bridge */ |
| 715 | kfree(buffer.pointer); | 720 | kfree(buffer.pointer); |
| 716 | } | 721 | } |
diff --git a/drivers/acpi/sleep.c b/drivers/acpi/sleep.c index 24213033fbae..9c1a435d10e6 100644 --- a/drivers/acpi/sleep.c +++ b/drivers/acpi/sleep.c | |||
| @@ -193,6 +193,14 @@ static struct dmi_system_id __initdata acpisleep_dmi_table[] = { | |||
| 193 | }, | 193 | }, |
| 194 | { | 194 | { |
| 195 | .callback = init_nvs_nosave, | 195 | .callback = init_nvs_nosave, |
| 196 | .ident = "Sony Vaio VGN-FW21M", | ||
| 197 | .matches = { | ||
| 198 | DMI_MATCH(DMI_SYS_VENDOR, "Sony Corporation"), | ||
| 199 | DMI_MATCH(DMI_PRODUCT_NAME, "VGN-FW21M"), | ||
| 200 | }, | ||
| 201 | }, | ||
| 202 | { | ||
| 203 | .callback = init_nvs_nosave, | ||
| 196 | .ident = "Sony Vaio VPCEB17FX", | 204 | .ident = "Sony Vaio VPCEB17FX", |
| 197 | .matches = { | 205 | .matches = { |
| 198 | DMI_MATCH(DMI_SYS_VENDOR, "Sony Corporation"), | 206 | DMI_MATCH(DMI_SYS_VENDOR, "Sony Corporation"), |
diff --git a/drivers/block/Kconfig b/drivers/block/Kconfig index 5dc0daed8fac..b81ddfea1da0 100644 --- a/drivers/block/Kconfig +++ b/drivers/block/Kconfig | |||
| @@ -532,11 +532,11 @@ config BLK_DEV_RBD | |||
| 532 | If unsure, say N. | 532 | If unsure, say N. |
| 533 | 533 | ||
| 534 | config BLK_DEV_RSXX | 534 | config BLK_DEV_RSXX |
| 535 | tristate "RamSam PCIe Flash SSD Device Driver" | 535 | tristate "IBM FlashSystem 70/80 PCIe SSD Device Driver" |
| 536 | depends on PCI | 536 | depends on PCI |
| 537 | help | 537 | help |
| 538 | Device driver for IBM's high speed PCIe SSD | 538 | Device driver for IBM's high speed PCIe SSD |
| 539 | storage devices: RamSan-70 and RamSan-80. | 539 | storage devices: FlashSystem-70 and FlashSystem-80. |
| 540 | 540 | ||
| 541 | To compile this driver as a module, choose M here: the | 541 | To compile this driver as a module, choose M here: the |
| 542 | module will be called rsxx. | 542 | module will be called rsxx. |
diff --git a/drivers/block/cciss.c b/drivers/block/cciss.c index ade58bc8f3c4..1c1b8e544aa2 100644 --- a/drivers/block/cciss.c +++ b/drivers/block/cciss.c | |||
| @@ -4206,7 +4206,7 @@ static int cciss_find_cfgtables(ctlr_info_t *h) | |||
| 4206 | if (rc) | 4206 | if (rc) |
| 4207 | return rc; | 4207 | return rc; |
| 4208 | h->cfgtable = remap_pci_mem(pci_resource_start(h->pdev, | 4208 | h->cfgtable = remap_pci_mem(pci_resource_start(h->pdev, |
| 4209 | cfg_base_addr_index) + cfg_offset, sizeof(h->cfgtable)); | 4209 | cfg_base_addr_index) + cfg_offset, sizeof(*h->cfgtable)); |
| 4210 | if (!h->cfgtable) | 4210 | if (!h->cfgtable) |
| 4211 | return -ENOMEM; | 4211 | return -ENOMEM; |
| 4212 | rc = write_driver_ver_to_cfgtable(h->cfgtable); | 4212 | rc = write_driver_ver_to_cfgtable(h->cfgtable); |
diff --git a/drivers/block/loop.c b/drivers/block/loop.c index 747bb2af69dc..fe5f6403417f 100644 --- a/drivers/block/loop.c +++ b/drivers/block/loop.c | |||
| @@ -1044,12 +1044,29 @@ static int loop_clr_fd(struct loop_device *lo) | |||
| 1044 | lo->lo_state = Lo_unbound; | 1044 | lo->lo_state = Lo_unbound; |
| 1045 | /* This is safe: open() is still holding a reference. */ | 1045 | /* This is safe: open() is still holding a reference. */ |
| 1046 | module_put(THIS_MODULE); | 1046 | module_put(THIS_MODULE); |
| 1047 | if (lo->lo_flags & LO_FLAGS_PARTSCAN && bdev) | ||
| 1048 | ioctl_by_bdev(bdev, BLKRRPART, 0); | ||
| 1049 | lo->lo_flags = 0; | 1047 | lo->lo_flags = 0; |
| 1050 | if (!part_shift) | 1048 | if (!part_shift) |
| 1051 | lo->lo_disk->flags |= GENHD_FL_NO_PART_SCAN; | 1049 | lo->lo_disk->flags |= GENHD_FL_NO_PART_SCAN; |
| 1052 | mutex_unlock(&lo->lo_ctl_mutex); | 1050 | mutex_unlock(&lo->lo_ctl_mutex); |
| 1051 | |||
| 1052 | /* | ||
| 1053 | * Remove all partitions, since BLKRRPART won't remove user | ||
| 1054 | * added partitions when max_part=0 | ||
| 1055 | */ | ||
| 1056 | if (bdev) { | ||
| 1057 | struct disk_part_iter piter; | ||
| 1058 | struct hd_struct *part; | ||
| 1059 | |||
| 1060 | mutex_lock_nested(&bdev->bd_mutex, 1); | ||
| 1061 | invalidate_partition(bdev->bd_disk, 0); | ||
| 1062 | disk_part_iter_init(&piter, bdev->bd_disk, | ||
| 1063 | DISK_PITER_INCL_EMPTY); | ||
| 1064 | while ((part = disk_part_iter_next(&piter))) | ||
| 1065 | delete_partition(bdev->bd_disk, part->partno); | ||
| 1066 | disk_part_iter_exit(&piter); | ||
| 1067 | mutex_unlock(&bdev->bd_mutex); | ||
| 1068 | } | ||
| 1069 | |||
| 1053 | /* | 1070 | /* |
| 1054 | * Need not hold lo_ctl_mutex to fput backing file. | 1071 | * Need not hold lo_ctl_mutex to fput backing file. |
| 1055 | * Calling fput holding lo_ctl_mutex triggers a circular | 1072 | * Calling fput holding lo_ctl_mutex triggers a circular |
| @@ -1623,6 +1640,7 @@ static int loop_add(struct loop_device **l, int i) | |||
| 1623 | goto out_free_dev; | 1640 | goto out_free_dev; |
| 1624 | i = err; | 1641 | i = err; |
| 1625 | 1642 | ||
| 1643 | err = -ENOMEM; | ||
| 1626 | lo->lo_queue = blk_alloc_queue(GFP_KERNEL); | 1644 | lo->lo_queue = blk_alloc_queue(GFP_KERNEL); |
| 1627 | if (!lo->lo_queue) | 1645 | if (!lo->lo_queue) |
| 1628 | goto out_free_dev; | 1646 | goto out_free_dev; |
diff --git a/drivers/block/mg_disk.c b/drivers/block/mg_disk.c index 1788f491e0fb..076ae7f1b781 100644 --- a/drivers/block/mg_disk.c +++ b/drivers/block/mg_disk.c | |||
| @@ -890,8 +890,10 @@ static int mg_probe(struct platform_device *plat_dev) | |||
| 890 | gpio_direction_output(host->rst, 1); | 890 | gpio_direction_output(host->rst, 1); |
| 891 | 891 | ||
| 892 | /* reset out pin */ | 892 | /* reset out pin */ |
| 893 | if (!(prv_data->dev_attr & MG_DEV_MASK)) | 893 | if (!(prv_data->dev_attr & MG_DEV_MASK)) { |
| 894 | err = -EINVAL; | ||
| 894 | goto probe_err_3a; | 895 | goto probe_err_3a; |
| 896 | } | ||
| 895 | 897 | ||
| 896 | if (prv_data->dev_attr != MG_BOOT_DEV) { | 898 | if (prv_data->dev_attr != MG_BOOT_DEV) { |
| 897 | rsc = platform_get_resource_byname(plat_dev, IORESOURCE_IO, | 899 | rsc = platform_get_resource_byname(plat_dev, IORESOURCE_IO, |
diff --git a/drivers/block/mtip32xx/mtip32xx.c b/drivers/block/mtip32xx/mtip32xx.c index 11cc9522cdd4..92250af84e7d 100644 --- a/drivers/block/mtip32xx/mtip32xx.c +++ b/drivers/block/mtip32xx/mtip32xx.c | |||
| @@ -4224,6 +4224,7 @@ static int mtip_pci_probe(struct pci_dev *pdev, | |||
| 4224 | dd->isr_workq = create_workqueue(dd->workq_name); | 4224 | dd->isr_workq = create_workqueue(dd->workq_name); |
| 4225 | if (!dd->isr_workq) { | 4225 | if (!dd->isr_workq) { |
| 4226 | dev_warn(&pdev->dev, "Can't create wq %d\n", dd->instance); | 4226 | dev_warn(&pdev->dev, "Can't create wq %d\n", dd->instance); |
| 4227 | rv = -ENOMEM; | ||
| 4227 | goto block_initialize_err; | 4228 | goto block_initialize_err; |
| 4228 | } | 4229 | } |
| 4229 | 4230 | ||
| @@ -4282,7 +4283,8 @@ static int mtip_pci_probe(struct pci_dev *pdev, | |||
| 4282 | INIT_WORK(&dd->work[7].work, mtip_workq_sdbf7); | 4283 | INIT_WORK(&dd->work[7].work, mtip_workq_sdbf7); |
| 4283 | 4284 | ||
| 4284 | pci_set_master(pdev); | 4285 | pci_set_master(pdev); |
| 4285 | if (pci_enable_msi(pdev)) { | 4286 | rv = pci_enable_msi(pdev); |
| 4287 | if (rv) { | ||
| 4286 | dev_warn(&pdev->dev, | 4288 | dev_warn(&pdev->dev, |
| 4287 | "Unable to enable MSI interrupt.\n"); | 4289 | "Unable to enable MSI interrupt.\n"); |
| 4288 | goto block_initialize_err; | 4290 | goto block_initialize_err; |
diff --git a/drivers/block/rbd.c b/drivers/block/rbd.c index 6c81a4c040b9..f556f8a8b3f9 100644 --- a/drivers/block/rbd.c +++ b/drivers/block/rbd.c | |||
| @@ -1264,6 +1264,32 @@ static bool obj_request_done_test(struct rbd_obj_request *obj_request) | |||
| 1264 | return atomic_read(&obj_request->done) != 0; | 1264 | return atomic_read(&obj_request->done) != 0; |
| 1265 | } | 1265 | } |
| 1266 | 1266 | ||
| 1267 | static void | ||
| 1268 | rbd_img_obj_request_read_callback(struct rbd_obj_request *obj_request) | ||
| 1269 | { | ||
| 1270 | dout("%s: obj %p img %p result %d %llu/%llu\n", __func__, | ||
| 1271 | obj_request, obj_request->img_request, obj_request->result, | ||
| 1272 | obj_request->xferred, obj_request->length); | ||
| 1273 | /* | ||
| 1274 | * ENOENT means a hole in the image. We zero-fill the | ||
| 1275 | * entire length of the request. A short read also implies | ||
| 1276 | * zero-fill to the end of the request. Either way we | ||
| 1277 | * update the xferred count to indicate the whole request | ||
| 1278 | * was satisfied. | ||
| 1279 | */ | ||
| 1280 | BUG_ON(obj_request->type != OBJ_REQUEST_BIO); | ||
| 1281 | if (obj_request->result == -ENOENT) { | ||
| 1282 | zero_bio_chain(obj_request->bio_list, 0); | ||
| 1283 | obj_request->result = 0; | ||
| 1284 | obj_request->xferred = obj_request->length; | ||
| 1285 | } else if (obj_request->xferred < obj_request->length && | ||
| 1286 | !obj_request->result) { | ||
| 1287 | zero_bio_chain(obj_request->bio_list, obj_request->xferred); | ||
| 1288 | obj_request->xferred = obj_request->length; | ||
| 1289 | } | ||
| 1290 | obj_request_done_set(obj_request); | ||
| 1291 | } | ||
| 1292 | |||
| 1267 | static void rbd_obj_request_complete(struct rbd_obj_request *obj_request) | 1293 | static void rbd_obj_request_complete(struct rbd_obj_request *obj_request) |
| 1268 | { | 1294 | { |
| 1269 | dout("%s: obj %p cb %p\n", __func__, obj_request, | 1295 | dout("%s: obj %p cb %p\n", __func__, obj_request, |
| @@ -1284,23 +1310,10 @@ static void rbd_osd_read_callback(struct rbd_obj_request *obj_request) | |||
| 1284 | { | 1310 | { |
| 1285 | dout("%s: obj %p result %d %llu/%llu\n", __func__, obj_request, | 1311 | dout("%s: obj %p result %d %llu/%llu\n", __func__, obj_request, |
| 1286 | obj_request->result, obj_request->xferred, obj_request->length); | 1312 | obj_request->result, obj_request->xferred, obj_request->length); |
| 1287 | /* | 1313 | if (obj_request->img_request) |
| 1288 | * ENOENT means a hole in the object. We zero-fill the | 1314 | rbd_img_obj_request_read_callback(obj_request); |
| 1289 | * entire length of the request. A short read also implies | 1315 | else |
| 1290 | * zero-fill to the end of the request. Either way we | 1316 | obj_request_done_set(obj_request); |
| 1291 | * update the xferred count to indicate the whole request | ||
| 1292 | * was satisfied. | ||
| 1293 | */ | ||
| 1294 | if (obj_request->result == -ENOENT) { | ||
| 1295 | zero_bio_chain(obj_request->bio_list, 0); | ||
| 1296 | obj_request->result = 0; | ||
| 1297 | obj_request->xferred = obj_request->length; | ||
| 1298 | } else if (obj_request->xferred < obj_request->length && | ||
| 1299 | !obj_request->result) { | ||
| 1300 | zero_bio_chain(obj_request->bio_list, obj_request->xferred); | ||
| 1301 | obj_request->xferred = obj_request->length; | ||
| 1302 | } | ||
| 1303 | obj_request_done_set(obj_request); | ||
| 1304 | } | 1317 | } |
| 1305 | 1318 | ||
| 1306 | static void rbd_osd_write_callback(struct rbd_obj_request *obj_request) | 1319 | static void rbd_osd_write_callback(struct rbd_obj_request *obj_request) |
diff --git a/drivers/block/rsxx/Makefile b/drivers/block/rsxx/Makefile index f35cd0b71f7b..b1c53c0aa450 100644 --- a/drivers/block/rsxx/Makefile +++ b/drivers/block/rsxx/Makefile | |||
| @@ -1,2 +1,2 @@ | |||
| 1 | obj-$(CONFIG_BLK_DEV_RSXX) += rsxx.o | 1 | obj-$(CONFIG_BLK_DEV_RSXX) += rsxx.o |
| 2 | rsxx-y := config.o core.o cregs.o dev.o dma.o | 2 | rsxx-objs := config.o core.o cregs.o dev.o dma.o |
diff --git a/drivers/block/rsxx/config.c b/drivers/block/rsxx/config.c index a295e7e9ee41..10cd530d3e10 100644 --- a/drivers/block/rsxx/config.c +++ b/drivers/block/rsxx/config.c | |||
| @@ -29,15 +29,13 @@ | |||
| 29 | #include "rsxx_priv.h" | 29 | #include "rsxx_priv.h" |
| 30 | #include "rsxx_cfg.h" | 30 | #include "rsxx_cfg.h" |
| 31 | 31 | ||
| 32 | static void initialize_config(void *config) | 32 | static void initialize_config(struct rsxx_card_cfg *cfg) |
| 33 | { | 33 | { |
| 34 | struct rsxx_card_cfg *cfg = config; | ||
| 35 | |||
| 36 | cfg->hdr.version = RSXX_CFG_VERSION; | 34 | cfg->hdr.version = RSXX_CFG_VERSION; |
| 37 | 35 | ||
| 38 | cfg->data.block_size = RSXX_HW_BLK_SIZE; | 36 | cfg->data.block_size = RSXX_HW_BLK_SIZE; |
| 39 | cfg->data.stripe_size = RSXX_HW_BLK_SIZE; | 37 | cfg->data.stripe_size = RSXX_HW_BLK_SIZE; |
| 40 | cfg->data.vendor_id = RSXX_VENDOR_ID_TMS_IBM; | 38 | cfg->data.vendor_id = RSXX_VENDOR_ID_IBM; |
| 41 | cfg->data.cache_order = (-1); | 39 | cfg->data.cache_order = (-1); |
| 42 | cfg->data.intr_coal.mode = RSXX_INTR_COAL_DISABLED; | 40 | cfg->data.intr_coal.mode = RSXX_INTR_COAL_DISABLED; |
| 43 | cfg->data.intr_coal.count = 0; | 41 | cfg->data.intr_coal.count = 0; |
| @@ -181,7 +179,7 @@ int rsxx_load_config(struct rsxx_cardinfo *card) | |||
| 181 | } else { | 179 | } else { |
| 182 | dev_info(CARD_TO_DEV(card), | 180 | dev_info(CARD_TO_DEV(card), |
| 183 | "Initializing card configuration.\n"); | 181 | "Initializing card configuration.\n"); |
| 184 | initialize_config(card); | 182 | initialize_config(&card->config); |
| 185 | st = rsxx_save_config(card); | 183 | st = rsxx_save_config(card); |
| 186 | if (st) | 184 | if (st) |
| 187 | return st; | 185 | return st; |
diff --git a/drivers/block/rsxx/core.c b/drivers/block/rsxx/core.c index e5162487686a..5af21f2db29c 100644 --- a/drivers/block/rsxx/core.c +++ b/drivers/block/rsxx/core.c | |||
| @@ -30,6 +30,7 @@ | |||
| 30 | #include <linux/reboot.h> | 30 | #include <linux/reboot.h> |
| 31 | #include <linux/slab.h> | 31 | #include <linux/slab.h> |
| 32 | #include <linux/bitops.h> | 32 | #include <linux/bitops.h> |
| 33 | #include <linux/delay.h> | ||
| 33 | 34 | ||
| 34 | #include <linux/genhd.h> | 35 | #include <linux/genhd.h> |
| 35 | #include <linux/idr.h> | 36 | #include <linux/idr.h> |
| @@ -39,8 +40,8 @@ | |||
| 39 | 40 | ||
| 40 | #define NO_LEGACY 0 | 41 | #define NO_LEGACY 0 |
| 41 | 42 | ||
| 42 | MODULE_DESCRIPTION("IBM RamSan PCIe Flash SSD Device Driver"); | 43 | MODULE_DESCRIPTION("IBM FlashSystem 70/80 PCIe SSD Device Driver"); |
| 43 | MODULE_AUTHOR("IBM <support@ramsan.com>"); | 44 | MODULE_AUTHOR("Joshua Morris/Philip Kelleher, IBM"); |
| 44 | MODULE_LICENSE("GPL"); | 45 | MODULE_LICENSE("GPL"); |
| 45 | MODULE_VERSION(DRIVER_VERSION); | 46 | MODULE_VERSION(DRIVER_VERSION); |
| 46 | 47 | ||
| @@ -52,6 +53,13 @@ static DEFINE_IDA(rsxx_disk_ida); | |||
| 52 | static DEFINE_SPINLOCK(rsxx_ida_lock); | 53 | static DEFINE_SPINLOCK(rsxx_ida_lock); |
| 53 | 54 | ||
| 54 | /*----------------- Interrupt Control & Handling -------------------*/ | 55 | /*----------------- Interrupt Control & Handling -------------------*/ |
| 56 | |||
| 57 | static void rsxx_mask_interrupts(struct rsxx_cardinfo *card) | ||
| 58 | { | ||
| 59 | card->isr_mask = 0; | ||
| 60 | card->ier_mask = 0; | ||
| 61 | } | ||
| 62 | |||
| 55 | static void __enable_intr(unsigned int *mask, unsigned int intr) | 63 | static void __enable_intr(unsigned int *mask, unsigned int intr) |
| 56 | { | 64 | { |
| 57 | *mask |= intr; | 65 | *mask |= intr; |
| @@ -71,7 +79,8 @@ static void __disable_intr(unsigned int *mask, unsigned int intr) | |||
| 71 | */ | 79 | */ |
| 72 | void rsxx_enable_ier(struct rsxx_cardinfo *card, unsigned int intr) | 80 | void rsxx_enable_ier(struct rsxx_cardinfo *card, unsigned int intr) |
| 73 | { | 81 | { |
| 74 | if (unlikely(card->halt)) | 82 | if (unlikely(card->halt) || |
| 83 | unlikely(card->eeh_state)) | ||
| 75 | return; | 84 | return; |
| 76 | 85 | ||
| 77 | __enable_intr(&card->ier_mask, intr); | 86 | __enable_intr(&card->ier_mask, intr); |
| @@ -80,6 +89,9 @@ void rsxx_enable_ier(struct rsxx_cardinfo *card, unsigned int intr) | |||
| 80 | 89 | ||
| 81 | void rsxx_disable_ier(struct rsxx_cardinfo *card, unsigned int intr) | 90 | void rsxx_disable_ier(struct rsxx_cardinfo *card, unsigned int intr) |
| 82 | { | 91 | { |
| 92 | if (unlikely(card->eeh_state)) | ||
| 93 | return; | ||
| 94 | |||
| 83 | __disable_intr(&card->ier_mask, intr); | 95 | __disable_intr(&card->ier_mask, intr); |
| 84 | iowrite32(card->ier_mask, card->regmap + IER); | 96 | iowrite32(card->ier_mask, card->regmap + IER); |
| 85 | } | 97 | } |
| @@ -87,7 +99,8 @@ void rsxx_disable_ier(struct rsxx_cardinfo *card, unsigned int intr) | |||
| 87 | void rsxx_enable_ier_and_isr(struct rsxx_cardinfo *card, | 99 | void rsxx_enable_ier_and_isr(struct rsxx_cardinfo *card, |
| 88 | unsigned int intr) | 100 | unsigned int intr) |
| 89 | { | 101 | { |
| 90 | if (unlikely(card->halt)) | 102 | if (unlikely(card->halt) || |
| 103 | unlikely(card->eeh_state)) | ||
| 91 | return; | 104 | return; |
| 92 | 105 | ||
| 93 | __enable_intr(&card->isr_mask, intr); | 106 | __enable_intr(&card->isr_mask, intr); |
| @@ -97,6 +110,9 @@ void rsxx_enable_ier_and_isr(struct rsxx_cardinfo *card, | |||
| 97 | void rsxx_disable_ier_and_isr(struct rsxx_cardinfo *card, | 110 | void rsxx_disable_ier_and_isr(struct rsxx_cardinfo *card, |
| 98 | unsigned int intr) | 111 | unsigned int intr) |
| 99 | { | 112 | { |
| 113 | if (unlikely(card->eeh_state)) | ||
| 114 | return; | ||
| 115 | |||
| 100 | __disable_intr(&card->isr_mask, intr); | 116 | __disable_intr(&card->isr_mask, intr); |
| 101 | __disable_intr(&card->ier_mask, intr); | 117 | __disable_intr(&card->ier_mask, intr); |
| 102 | iowrite32(card->ier_mask, card->regmap + IER); | 118 | iowrite32(card->ier_mask, card->regmap + IER); |
| @@ -115,6 +131,9 @@ static irqreturn_t rsxx_isr(int irq, void *pdata) | |||
| 115 | do { | 131 | do { |
| 116 | reread_isr = 0; | 132 | reread_isr = 0; |
| 117 | 133 | ||
| 134 | if (unlikely(card->eeh_state)) | ||
| 135 | break; | ||
| 136 | |||
| 118 | isr = ioread32(card->regmap + ISR); | 137 | isr = ioread32(card->regmap + ISR); |
| 119 | if (isr == 0xffffffff) { | 138 | if (isr == 0xffffffff) { |
| 120 | /* | 139 | /* |
| @@ -161,9 +180,9 @@ static irqreturn_t rsxx_isr(int irq, void *pdata) | |||
| 161 | } | 180 | } |
| 162 | 181 | ||
| 163 | /*----------------- Card Event Handler -------------------*/ | 182 | /*----------------- Card Event Handler -------------------*/ |
| 164 | static char *rsxx_card_state_to_str(unsigned int state) | 183 | static const char * const rsxx_card_state_to_str(unsigned int state) |
| 165 | { | 184 | { |
| 166 | static char *state_strings[] = { | 185 | static const char * const state_strings[] = { |
| 167 | "Unknown", "Shutdown", "Starting", "Formatting", | 186 | "Unknown", "Shutdown", "Starting", "Formatting", |
| 168 | "Uninitialized", "Good", "Shutting Down", | 187 | "Uninitialized", "Good", "Shutting Down", |
| 169 | "Fault", "Read Only Fault", "dStroying" | 188 | "Fault", "Read Only Fault", "dStroying" |
| @@ -304,6 +323,192 @@ static int card_shutdown(struct rsxx_cardinfo *card) | |||
| 304 | return 0; | 323 | return 0; |
| 305 | } | 324 | } |
| 306 | 325 | ||
| 326 | static int rsxx_eeh_frozen(struct pci_dev *dev) | ||
| 327 | { | ||
| 328 | struct rsxx_cardinfo *card = pci_get_drvdata(dev); | ||
| 329 | int i; | ||
| 330 | int st; | ||
| 331 | |||
| 332 | dev_warn(&dev->dev, "IBM FlashSystem PCI: preparing for slot reset.\n"); | ||
| 333 | |||
| 334 | card->eeh_state = 1; | ||
| 335 | rsxx_mask_interrupts(card); | ||
| 336 | |||
| 337 | /* | ||
| 338 | * We need to guarantee that the write for eeh_state and masking | ||
| 339 | * interrupts does not become reordered. This will prevent a possible | ||
| 340 | * race condition with the EEH code. | ||
| 341 | */ | ||
| 342 | wmb(); | ||
| 343 | |||
| 344 | pci_disable_device(dev); | ||
| 345 | |||
| 346 | st = rsxx_eeh_save_issued_dmas(card); | ||
| 347 | if (st) | ||
| 348 | return st; | ||
| 349 | |||
| 350 | rsxx_eeh_save_issued_creg(card); | ||
| 351 | |||
| 352 | for (i = 0; i < card->n_targets; i++) { | ||
| 353 | if (card->ctrl[i].status.buf) | ||
| 354 | pci_free_consistent(card->dev, STATUS_BUFFER_SIZE8, | ||
| 355 | card->ctrl[i].status.buf, | ||
| 356 | card->ctrl[i].status.dma_addr); | ||
| 357 | if (card->ctrl[i].cmd.buf) | ||
| 358 | pci_free_consistent(card->dev, COMMAND_BUFFER_SIZE8, | ||
| 359 | card->ctrl[i].cmd.buf, | ||
| 360 | card->ctrl[i].cmd.dma_addr); | ||
| 361 | } | ||
| 362 | |||
| 363 | return 0; | ||
| 364 | } | ||
| 365 | |||
| 366 | static void rsxx_eeh_failure(struct pci_dev *dev) | ||
| 367 | { | ||
| 368 | struct rsxx_cardinfo *card = pci_get_drvdata(dev); | ||
| 369 | int i; | ||
| 370 | |||
| 371 | dev_err(&dev->dev, "IBM FlashSystem PCI: disabling failed card.\n"); | ||
| 372 | |||
| 373 | card->eeh_state = 1; | ||
| 374 | |||
| 375 | for (i = 0; i < card->n_targets; i++) | ||
| 376 | del_timer_sync(&card->ctrl[i].activity_timer); | ||
| 377 | |||
| 378 | rsxx_eeh_cancel_dmas(card); | ||
| 379 | } | ||
| 380 | |||
| 381 | static int rsxx_eeh_fifo_flush_poll(struct rsxx_cardinfo *card) | ||
| 382 | { | ||
| 383 | unsigned int status; | ||
| 384 | int iter = 0; | ||
| 385 | |||
| 386 | /* We need to wait for the hardware to reset */ | ||
| 387 | while (iter++ < 10) { | ||
| 388 | status = ioread32(card->regmap + PCI_RECONFIG); | ||
| 389 | |||
| 390 | if (status & RSXX_FLUSH_BUSY) { | ||
| 391 | ssleep(1); | ||
| 392 | continue; | ||
| 393 | } | ||
| 394 | |||
| 395 | if (status & RSXX_FLUSH_TIMEOUT) | ||
| 396 | dev_warn(CARD_TO_DEV(card), "HW: flash controller timeout\n"); | ||
| 397 | return 0; | ||
| 398 | } | ||
| 399 | |||
| 400 | /* Hardware failed resetting itself. */ | ||
| 401 | return -1; | ||
| 402 | } | ||
| 403 | |||
| 404 | static pci_ers_result_t rsxx_error_detected(struct pci_dev *dev, | ||
| 405 | enum pci_channel_state error) | ||
| 406 | { | ||
| 407 | int st; | ||
| 408 | |||
| 409 | if (dev->revision < RSXX_EEH_SUPPORT) | ||
| 410 | return PCI_ERS_RESULT_NONE; | ||
| 411 | |||
| 412 | if (error == pci_channel_io_perm_failure) { | ||
| 413 | rsxx_eeh_failure(dev); | ||
| 414 | return PCI_ERS_RESULT_DISCONNECT; | ||
| 415 | } | ||
| 416 | |||
| 417 | st = rsxx_eeh_frozen(dev); | ||
| 418 | if (st) { | ||
| 419 | dev_err(&dev->dev, "Slot reset setup failed\n"); | ||
| 420 | rsxx_eeh_failure(dev); | ||
| 421 | return PCI_ERS_RESULT_DISCONNECT; | ||
| 422 | } | ||
| 423 | |||
| 424 | return PCI_ERS_RESULT_NEED_RESET; | ||
| 425 | } | ||
| 426 | |||
| 427 | static pci_ers_result_t rsxx_slot_reset(struct pci_dev *dev) | ||
| 428 | { | ||
| 429 | struct rsxx_cardinfo *card = pci_get_drvdata(dev); | ||
| 430 | unsigned long flags; | ||
| 431 | int i; | ||
| 432 | int st; | ||
| 433 | |||
| 434 | dev_warn(&dev->dev, | ||
| 435 | "IBM FlashSystem PCI: recovering from slot reset.\n"); | ||
| 436 | |||
| 437 | st = pci_enable_device(dev); | ||
| 438 | if (st) | ||
| 439 | goto failed_hw_setup; | ||
| 440 | |||
| 441 | pci_set_master(dev); | ||
| 442 | |||
| 443 | st = rsxx_eeh_fifo_flush_poll(card); | ||
| 444 | if (st) | ||
| 445 | goto failed_hw_setup; | ||
| 446 | |||
| 447 | rsxx_dma_queue_reset(card); | ||
| 448 | |||
| 449 | for (i = 0; i < card->n_targets; i++) { | ||
| 450 | st = rsxx_hw_buffers_init(dev, &card->ctrl[i]); | ||
| 451 | if (st) | ||
| 452 | goto failed_hw_buffers_init; | ||
| 453 | } | ||
| 454 | |||
| 455 | if (card->config_valid) | ||
| 456 | rsxx_dma_configure(card); | ||
| 457 | |||
| 458 | /* Clears the ISR register from spurious interrupts */ | ||
| 459 | st = ioread32(card->regmap + ISR); | ||
| 460 | |||
| 461 | card->eeh_state = 0; | ||
| 462 | |||
| 463 | st = rsxx_eeh_remap_dmas(card); | ||
| 464 | if (st) | ||
| 465 | goto failed_remap_dmas; | ||
| 466 | |||
| 467 | spin_lock_irqsave(&card->irq_lock, flags); | ||
| 468 | if (card->n_targets & RSXX_MAX_TARGETS) | ||
| 469 | rsxx_enable_ier_and_isr(card, CR_INTR_ALL_G); | ||
| 470 | else | ||
| 471 | rsxx_enable_ier_and_isr(card, CR_INTR_ALL_C); | ||
| 472 | spin_unlock_irqrestore(&card->irq_lock, flags); | ||
| 473 | |||
| 474 | rsxx_kick_creg_queue(card); | ||
| 475 | |||
| 476 | for (i = 0; i < card->n_targets; i++) { | ||
| 477 | spin_lock(&card->ctrl[i].queue_lock); | ||
| 478 | if (list_empty(&card->ctrl[i].queue)) { | ||
| 479 | spin_unlock(&card->ctrl[i].queue_lock); | ||
| 480 | continue; | ||
| 481 | } | ||
| 482 | spin_unlock(&card->ctrl[i].queue_lock); | ||
| 483 | |||
| 484 | queue_work(card->ctrl[i].issue_wq, | ||
| 485 | &card->ctrl[i].issue_dma_work); | ||
| 486 | } | ||
| 487 | |||
| 488 | dev_info(&dev->dev, "IBM FlashSystem PCI: recovery complete.\n"); | ||
| 489 | |||
| 490 | return PCI_ERS_RESULT_RECOVERED; | ||
| 491 | |||
| 492 | failed_hw_buffers_init: | ||
| 493 | failed_remap_dmas: | ||
| 494 | for (i = 0; i < card->n_targets; i++) { | ||
| 495 | if (card->ctrl[i].status.buf) | ||
| 496 | pci_free_consistent(card->dev, | ||
| 497 | STATUS_BUFFER_SIZE8, | ||
| 498 | card->ctrl[i].status.buf, | ||
| 499 | card->ctrl[i].status.dma_addr); | ||
| 500 | if (card->ctrl[i].cmd.buf) | ||
| 501 | pci_free_consistent(card->dev, | ||
| 502 | COMMAND_BUFFER_SIZE8, | ||
| 503 | card->ctrl[i].cmd.buf, | ||
| 504 | card->ctrl[i].cmd.dma_addr); | ||
| 505 | } | ||
| 506 | failed_hw_setup: | ||
| 507 | rsxx_eeh_failure(dev); | ||
| 508 | return PCI_ERS_RESULT_DISCONNECT; | ||
| 509 | |||
| 510 | } | ||
| 511 | |||
| 307 | /*----------------- Driver Initialization & Setup -------------------*/ | 512 | /*----------------- Driver Initialization & Setup -------------------*/ |
| 308 | /* Returns: 0 if the driver is compatible with the device | 513 | /* Returns: 0 if the driver is compatible with the device |
| 309 | -1 if the driver is NOT compatible with the device */ | 514 | -1 if the driver is NOT compatible with the device */ |
| @@ -383,6 +588,7 @@ static int rsxx_pci_probe(struct pci_dev *dev, | |||
| 383 | 588 | ||
| 384 | spin_lock_init(&card->irq_lock); | 589 | spin_lock_init(&card->irq_lock); |
| 385 | card->halt = 0; | 590 | card->halt = 0; |
| 591 | card->eeh_state = 0; | ||
| 386 | 592 | ||
| 387 | spin_lock_irq(&card->irq_lock); | 593 | spin_lock_irq(&card->irq_lock); |
| 388 | rsxx_disable_ier_and_isr(card, CR_INTR_ALL); | 594 | rsxx_disable_ier_and_isr(card, CR_INTR_ALL); |
| @@ -538,9 +744,6 @@ static void rsxx_pci_remove(struct pci_dev *dev) | |||
| 538 | rsxx_disable_ier_and_isr(card, CR_INTR_EVENT); | 744 | rsxx_disable_ier_and_isr(card, CR_INTR_EVENT); |
| 539 | spin_unlock_irqrestore(&card->irq_lock, flags); | 745 | spin_unlock_irqrestore(&card->irq_lock, flags); |
| 540 | 746 | ||
| 541 | /* Prevent work_structs from re-queuing themselves. */ | ||
| 542 | card->halt = 1; | ||
| 543 | |||
| 544 | cancel_work_sync(&card->event_work); | 747 | cancel_work_sync(&card->event_work); |
| 545 | 748 | ||
| 546 | rsxx_destroy_dev(card); | 749 | rsxx_destroy_dev(card); |
| @@ -549,6 +752,10 @@ static void rsxx_pci_remove(struct pci_dev *dev) | |||
| 549 | spin_lock_irqsave(&card->irq_lock, flags); | 752 | spin_lock_irqsave(&card->irq_lock, flags); |
| 550 | rsxx_disable_ier_and_isr(card, CR_INTR_ALL); | 753 | rsxx_disable_ier_and_isr(card, CR_INTR_ALL); |
| 551 | spin_unlock_irqrestore(&card->irq_lock, flags); | 754 | spin_unlock_irqrestore(&card->irq_lock, flags); |
| 755 | |||
| 756 | /* Prevent work_structs from re-queuing themselves. */ | ||
| 757 | card->halt = 1; | ||
| 758 | |||
| 552 | free_irq(dev->irq, card); | 759 | free_irq(dev->irq, card); |
| 553 | 760 | ||
| 554 | if (!force_legacy) | 761 | if (!force_legacy) |
| @@ -592,11 +799,14 @@ static void rsxx_pci_shutdown(struct pci_dev *dev) | |||
| 592 | card_shutdown(card); | 799 | card_shutdown(card); |
| 593 | } | 800 | } |
| 594 | 801 | ||
| 802 | static const struct pci_error_handlers rsxx_err_handler = { | ||
| 803 | .error_detected = rsxx_error_detected, | ||
| 804 | .slot_reset = rsxx_slot_reset, | ||
| 805 | }; | ||
| 806 | |||
| 595 | static DEFINE_PCI_DEVICE_TABLE(rsxx_pci_ids) = { | 807 | static DEFINE_PCI_DEVICE_TABLE(rsxx_pci_ids) = { |
| 596 | {PCI_DEVICE(PCI_VENDOR_ID_TMS_IBM, PCI_DEVICE_ID_RS70_FLASH)}, | 808 | {PCI_DEVICE(PCI_VENDOR_ID_IBM, PCI_DEVICE_ID_FS70_FLASH)}, |
| 597 | {PCI_DEVICE(PCI_VENDOR_ID_TMS_IBM, PCI_DEVICE_ID_RS70D_FLASH)}, | 809 | {PCI_DEVICE(PCI_VENDOR_ID_IBM, PCI_DEVICE_ID_FS80_FLASH)}, |
| 598 | {PCI_DEVICE(PCI_VENDOR_ID_TMS_IBM, PCI_DEVICE_ID_RS80_FLASH)}, | ||
| 599 | {PCI_DEVICE(PCI_VENDOR_ID_TMS_IBM, PCI_DEVICE_ID_RS81_FLASH)}, | ||
| 600 | {0,}, | 810 | {0,}, |
| 601 | }; | 811 | }; |
| 602 | 812 | ||
| @@ -609,6 +819,7 @@ static struct pci_driver rsxx_pci_driver = { | |||
| 609 | .remove = rsxx_pci_remove, | 819 | .remove = rsxx_pci_remove, |
| 610 | .suspend = rsxx_pci_suspend, | 820 | .suspend = rsxx_pci_suspend, |
| 611 | .shutdown = rsxx_pci_shutdown, | 821 | .shutdown = rsxx_pci_shutdown, |
| 822 | .err_handler = &rsxx_err_handler, | ||
| 612 | }; | 823 | }; |
| 613 | 824 | ||
| 614 | static int __init rsxx_core_init(void) | 825 | static int __init rsxx_core_init(void) |
diff --git a/drivers/block/rsxx/cregs.c b/drivers/block/rsxx/cregs.c index 80bbe639fccd..4b5c020a0a65 100644 --- a/drivers/block/rsxx/cregs.c +++ b/drivers/block/rsxx/cregs.c | |||
| @@ -58,7 +58,7 @@ static struct kmem_cache *creg_cmd_pool; | |||
| 58 | #error Unknown endianess!!! Aborting... | 58 | #error Unknown endianess!!! Aborting... |
| 59 | #endif | 59 | #endif |
| 60 | 60 | ||
| 61 | static void copy_to_creg_data(struct rsxx_cardinfo *card, | 61 | static int copy_to_creg_data(struct rsxx_cardinfo *card, |
| 62 | int cnt8, | 62 | int cnt8, |
| 63 | void *buf, | 63 | void *buf, |
| 64 | unsigned int stream) | 64 | unsigned int stream) |
| @@ -66,6 +66,9 @@ static void copy_to_creg_data(struct rsxx_cardinfo *card, | |||
| 66 | int i = 0; | 66 | int i = 0; |
| 67 | u32 *data = buf; | 67 | u32 *data = buf; |
| 68 | 68 | ||
| 69 | if (unlikely(card->eeh_state)) | ||
| 70 | return -EIO; | ||
| 71 | |||
| 69 | for (i = 0; cnt8 > 0; i++, cnt8 -= 4) { | 72 | for (i = 0; cnt8 > 0; i++, cnt8 -= 4) { |
| 70 | /* | 73 | /* |
| 71 | * Firmware implementation makes it necessary to byte swap on | 74 | * Firmware implementation makes it necessary to byte swap on |
| @@ -76,10 +79,12 @@ static void copy_to_creg_data(struct rsxx_cardinfo *card, | |||
| 76 | else | 79 | else |
| 77 | iowrite32(data[i], card->regmap + CREG_DATA(i)); | 80 | iowrite32(data[i], card->regmap + CREG_DATA(i)); |
| 78 | } | 81 | } |
| 82 | |||
| 83 | return 0; | ||
| 79 | } | 84 | } |
| 80 | 85 | ||
| 81 | 86 | ||
| 82 | static void copy_from_creg_data(struct rsxx_cardinfo *card, | 87 | static int copy_from_creg_data(struct rsxx_cardinfo *card, |
| 83 | int cnt8, | 88 | int cnt8, |
| 84 | void *buf, | 89 | void *buf, |
| 85 | unsigned int stream) | 90 | unsigned int stream) |
| @@ -87,6 +92,9 @@ static void copy_from_creg_data(struct rsxx_cardinfo *card, | |||
| 87 | int i = 0; | 92 | int i = 0; |
| 88 | u32 *data = buf; | 93 | u32 *data = buf; |
| 89 | 94 | ||
| 95 | if (unlikely(card->eeh_state)) | ||
| 96 | return -EIO; | ||
| 97 | |||
| 90 | for (i = 0; cnt8 > 0; i++, cnt8 -= 4) { | 98 | for (i = 0; cnt8 > 0; i++, cnt8 -= 4) { |
| 91 | /* | 99 | /* |
| 92 | * Firmware implementation makes it necessary to byte swap on | 100 | * Firmware implementation makes it necessary to byte swap on |
| @@ -97,41 +105,31 @@ static void copy_from_creg_data(struct rsxx_cardinfo *card, | |||
| 97 | else | 105 | else |
| 98 | data[i] = ioread32(card->regmap + CREG_DATA(i)); | 106 | data[i] = ioread32(card->regmap + CREG_DATA(i)); |
| 99 | } | 107 | } |
| 100 | } | ||
| 101 | |||
| 102 | static struct creg_cmd *pop_active_cmd(struct rsxx_cardinfo *card) | ||
| 103 | { | ||
| 104 | struct creg_cmd *cmd; | ||
| 105 | 108 | ||
| 106 | /* | 109 | return 0; |
| 107 | * Spin lock is needed because this can be called in atomic/interrupt | ||
| 108 | * context. | ||
| 109 | */ | ||
| 110 | spin_lock_bh(&card->creg_ctrl.lock); | ||
| 111 | cmd = card->creg_ctrl.active_cmd; | ||
| 112 | card->creg_ctrl.active_cmd = NULL; | ||
| 113 | spin_unlock_bh(&card->creg_ctrl.lock); | ||
| 114 | |||
| 115 | return cmd; | ||
| 116 | } | 110 | } |
| 117 | 111 | ||
| 118 | static void creg_issue_cmd(struct rsxx_cardinfo *card, struct creg_cmd *cmd) | 112 | static void creg_issue_cmd(struct rsxx_cardinfo *card, struct creg_cmd *cmd) |
| 119 | { | 113 | { |
| 114 | int st; | ||
| 115 | |||
| 116 | if (unlikely(card->eeh_state)) | ||
| 117 | return; | ||
| 118 | |||
| 120 | iowrite32(cmd->addr, card->regmap + CREG_ADD); | 119 | iowrite32(cmd->addr, card->regmap + CREG_ADD); |
| 121 | iowrite32(cmd->cnt8, card->regmap + CREG_CNT); | 120 | iowrite32(cmd->cnt8, card->regmap + CREG_CNT); |
| 122 | 121 | ||
| 123 | if (cmd->op == CREG_OP_WRITE) { | 122 | if (cmd->op == CREG_OP_WRITE) { |
| 124 | if (cmd->buf) | 123 | if (cmd->buf) { |
| 125 | copy_to_creg_data(card, cmd->cnt8, | 124 | st = copy_to_creg_data(card, cmd->cnt8, |
| 126 | cmd->buf, cmd->stream); | 125 | cmd->buf, cmd->stream); |
| 126 | if (st) | ||
| 127 | return; | ||
| 128 | } | ||
| 127 | } | 129 | } |
| 128 | 130 | ||
| 129 | /* | 131 | if (unlikely(card->eeh_state)) |
| 130 | * Data copy must complete before initiating the command. This is | 132 | return; |
| 131 | * needed for weakly ordered processors (i.e. PowerPC), so that all | ||
| 132 | * neccessary registers are written before we kick the hardware. | ||
| 133 | */ | ||
| 134 | wmb(); | ||
| 135 | 133 | ||
| 136 | /* Setting the valid bit will kick off the command. */ | 134 | /* Setting the valid bit will kick off the command. */ |
| 137 | iowrite32(cmd->op, card->regmap + CREG_CMD); | 135 | iowrite32(cmd->op, card->regmap + CREG_CMD); |
| @@ -196,11 +194,11 @@ static int creg_queue_cmd(struct rsxx_cardinfo *card, | |||
| 196 | cmd->cb_private = cb_private; | 194 | cmd->cb_private = cb_private; |
| 197 | cmd->status = 0; | 195 | cmd->status = 0; |
| 198 | 196 | ||
| 199 | spin_lock(&card->creg_ctrl.lock); | 197 | spin_lock_bh(&card->creg_ctrl.lock); |
| 200 | list_add_tail(&cmd->list, &card->creg_ctrl.queue); | 198 | list_add_tail(&cmd->list, &card->creg_ctrl.queue); |
| 201 | card->creg_ctrl.q_depth++; | 199 | card->creg_ctrl.q_depth++; |
| 202 | creg_kick_queue(card); | 200 | creg_kick_queue(card); |
| 203 | spin_unlock(&card->creg_ctrl.lock); | 201 | spin_unlock_bh(&card->creg_ctrl.lock); |
| 204 | 202 | ||
| 205 | return 0; | 203 | return 0; |
| 206 | } | 204 | } |
| @@ -210,7 +208,11 @@ static void creg_cmd_timed_out(unsigned long data) | |||
| 210 | struct rsxx_cardinfo *card = (struct rsxx_cardinfo *) data; | 208 | struct rsxx_cardinfo *card = (struct rsxx_cardinfo *) data; |
| 211 | struct creg_cmd *cmd; | 209 | struct creg_cmd *cmd; |
| 212 | 210 | ||
| 213 | cmd = pop_active_cmd(card); | 211 | spin_lock(&card->creg_ctrl.lock); |
| 212 | cmd = card->creg_ctrl.active_cmd; | ||
| 213 | card->creg_ctrl.active_cmd = NULL; | ||
| 214 | spin_unlock(&card->creg_ctrl.lock); | ||
| 215 | |||
| 214 | if (cmd == NULL) { | 216 | if (cmd == NULL) { |
| 215 | card->creg_ctrl.creg_stats.creg_timeout++; | 217 | card->creg_ctrl.creg_stats.creg_timeout++; |
| 216 | dev_warn(CARD_TO_DEV(card), | 218 | dev_warn(CARD_TO_DEV(card), |
| @@ -247,7 +249,11 @@ static void creg_cmd_done(struct work_struct *work) | |||
| 247 | if (del_timer_sync(&card->creg_ctrl.cmd_timer) == 0) | 249 | if (del_timer_sync(&card->creg_ctrl.cmd_timer) == 0) |
| 248 | card->creg_ctrl.creg_stats.failed_cancel_timer++; | 250 | card->creg_ctrl.creg_stats.failed_cancel_timer++; |
| 249 | 251 | ||
| 250 | cmd = pop_active_cmd(card); | 252 | spin_lock_bh(&card->creg_ctrl.lock); |
| 253 | cmd = card->creg_ctrl.active_cmd; | ||
| 254 | card->creg_ctrl.active_cmd = NULL; | ||
| 255 | spin_unlock_bh(&card->creg_ctrl.lock); | ||
| 256 | |||
| 251 | if (cmd == NULL) { | 257 | if (cmd == NULL) { |
| 252 | dev_err(CARD_TO_DEV(card), | 258 | dev_err(CARD_TO_DEV(card), |
| 253 | "Spurious creg interrupt!\n"); | 259 | "Spurious creg interrupt!\n"); |
| @@ -287,7 +293,7 @@ static void creg_cmd_done(struct work_struct *work) | |||
| 287 | goto creg_done; | 293 | goto creg_done; |
| 288 | } | 294 | } |
| 289 | 295 | ||
| 290 | copy_from_creg_data(card, cnt8, cmd->buf, cmd->stream); | 296 | st = copy_from_creg_data(card, cnt8, cmd->buf, cmd->stream); |
| 291 | } | 297 | } |
| 292 | 298 | ||
| 293 | creg_done: | 299 | creg_done: |
| @@ -296,10 +302,10 @@ creg_done: | |||
| 296 | 302 | ||
| 297 | kmem_cache_free(creg_cmd_pool, cmd); | 303 | kmem_cache_free(creg_cmd_pool, cmd); |
| 298 | 304 | ||
| 299 | spin_lock(&card->creg_ctrl.lock); | 305 | spin_lock_bh(&card->creg_ctrl.lock); |
| 300 | card->creg_ctrl.active = 0; | 306 | card->creg_ctrl.active = 0; |
| 301 | creg_kick_queue(card); | 307 | creg_kick_queue(card); |
| 302 | spin_unlock(&card->creg_ctrl.lock); | 308 | spin_unlock_bh(&card->creg_ctrl.lock); |
| 303 | } | 309 | } |
| 304 | 310 | ||
| 305 | static void creg_reset(struct rsxx_cardinfo *card) | 311 | static void creg_reset(struct rsxx_cardinfo *card) |
| @@ -324,7 +330,7 @@ static void creg_reset(struct rsxx_cardinfo *card) | |||
| 324 | "Resetting creg interface for recovery\n"); | 330 | "Resetting creg interface for recovery\n"); |
| 325 | 331 | ||
| 326 | /* Cancel outstanding commands */ | 332 | /* Cancel outstanding commands */ |
| 327 | spin_lock(&card->creg_ctrl.lock); | 333 | spin_lock_bh(&card->creg_ctrl.lock); |
| 328 | list_for_each_entry_safe(cmd, tmp, &card->creg_ctrl.queue, list) { | 334 | list_for_each_entry_safe(cmd, tmp, &card->creg_ctrl.queue, list) { |
| 329 | list_del(&cmd->list); | 335 | list_del(&cmd->list); |
| 330 | card->creg_ctrl.q_depth--; | 336 | card->creg_ctrl.q_depth--; |
| @@ -345,7 +351,7 @@ static void creg_reset(struct rsxx_cardinfo *card) | |||
| 345 | 351 | ||
| 346 | card->creg_ctrl.active = 0; | 352 | card->creg_ctrl.active = 0; |
| 347 | } | 353 | } |
| 348 | spin_unlock(&card->creg_ctrl.lock); | 354 | spin_unlock_bh(&card->creg_ctrl.lock); |
| 349 | 355 | ||
| 350 | card->creg_ctrl.reset = 0; | 356 | card->creg_ctrl.reset = 0; |
| 351 | spin_lock_irqsave(&card->irq_lock, flags); | 357 | spin_lock_irqsave(&card->irq_lock, flags); |
| @@ -399,12 +405,12 @@ static int __issue_creg_rw(struct rsxx_cardinfo *card, | |||
| 399 | return st; | 405 | return st; |
| 400 | 406 | ||
| 401 | /* | 407 | /* |
| 402 | * This timeout is neccessary for unresponsive hardware. The additional | 408 | * This timeout is necessary for unresponsive hardware. The additional |
| 403 | * 20 seconds to used to guarantee that each cregs requests has time to | 409 | * 20 seconds to used to guarantee that each cregs requests has time to |
| 404 | * complete. | 410 | * complete. |
| 405 | */ | 411 | */ |
| 406 | timeout = msecs_to_jiffies((CREG_TIMEOUT_MSEC * | 412 | timeout = msecs_to_jiffies(CREG_TIMEOUT_MSEC * |
| 407 | card->creg_ctrl.q_depth) + 20000); | 413 | card->creg_ctrl.q_depth + 20000); |
| 408 | 414 | ||
| 409 | /* | 415 | /* |
| 410 | * The creg interface is guaranteed to complete. It has a timeout | 416 | * The creg interface is guaranteed to complete. It has a timeout |
| @@ -690,6 +696,32 @@ int rsxx_reg_access(struct rsxx_cardinfo *card, | |||
| 690 | return 0; | 696 | return 0; |
| 691 | } | 697 | } |
| 692 | 698 | ||
| 699 | void rsxx_eeh_save_issued_creg(struct rsxx_cardinfo *card) | ||
| 700 | { | ||
| 701 | struct creg_cmd *cmd = NULL; | ||
| 702 | |||
| 703 | cmd = card->creg_ctrl.active_cmd; | ||
| 704 | card->creg_ctrl.active_cmd = NULL; | ||
| 705 | |||
| 706 | if (cmd) { | ||
| 707 | del_timer_sync(&card->creg_ctrl.cmd_timer); | ||
| 708 | |||
| 709 | spin_lock_bh(&card->creg_ctrl.lock); | ||
| 710 | list_add(&cmd->list, &card->creg_ctrl.queue); | ||
| 711 | card->creg_ctrl.q_depth++; | ||
| 712 | card->creg_ctrl.active = 0; | ||
| 713 | spin_unlock_bh(&card->creg_ctrl.lock); | ||
| 714 | } | ||
| 715 | } | ||
| 716 | |||
| 717 | void rsxx_kick_creg_queue(struct rsxx_cardinfo *card) | ||
| 718 | { | ||
| 719 | spin_lock_bh(&card->creg_ctrl.lock); | ||
| 720 | if (!list_empty(&card->creg_ctrl.queue)) | ||
| 721 | creg_kick_queue(card); | ||
| 722 | spin_unlock_bh(&card->creg_ctrl.lock); | ||
| 723 | } | ||
| 724 | |||
| 693 | /*------------ Initialization & Setup --------------*/ | 725 | /*------------ Initialization & Setup --------------*/ |
| 694 | int rsxx_creg_setup(struct rsxx_cardinfo *card) | 726 | int rsxx_creg_setup(struct rsxx_cardinfo *card) |
| 695 | { | 727 | { |
| @@ -712,7 +744,7 @@ void rsxx_creg_destroy(struct rsxx_cardinfo *card) | |||
| 712 | int cnt = 0; | 744 | int cnt = 0; |
| 713 | 745 | ||
| 714 | /* Cancel outstanding commands */ | 746 | /* Cancel outstanding commands */ |
| 715 | spin_lock(&card->creg_ctrl.lock); | 747 | spin_lock_bh(&card->creg_ctrl.lock); |
| 716 | list_for_each_entry_safe(cmd, tmp, &card->creg_ctrl.queue, list) { | 748 | list_for_each_entry_safe(cmd, tmp, &card->creg_ctrl.queue, list) { |
| 717 | list_del(&cmd->list); | 749 | list_del(&cmd->list); |
| 718 | if (cmd->cb) | 750 | if (cmd->cb) |
| @@ -737,7 +769,7 @@ void rsxx_creg_destroy(struct rsxx_cardinfo *card) | |||
| 737 | "Canceled active creg command\n"); | 769 | "Canceled active creg command\n"); |
| 738 | kmem_cache_free(creg_cmd_pool, cmd); | 770 | kmem_cache_free(creg_cmd_pool, cmd); |
| 739 | } | 771 | } |
| 740 | spin_unlock(&card->creg_ctrl.lock); | 772 | spin_unlock_bh(&card->creg_ctrl.lock); |
| 741 | 773 | ||
| 742 | cancel_work_sync(&card->creg_ctrl.done_work); | 774 | cancel_work_sync(&card->creg_ctrl.done_work); |
| 743 | } | 775 | } |
diff --git a/drivers/block/rsxx/dma.c b/drivers/block/rsxx/dma.c index 63176e67662f..0607513cfb41 100644 --- a/drivers/block/rsxx/dma.c +++ b/drivers/block/rsxx/dma.c | |||
| @@ -28,7 +28,7 @@ | |||
| 28 | struct rsxx_dma { | 28 | struct rsxx_dma { |
| 29 | struct list_head list; | 29 | struct list_head list; |
| 30 | u8 cmd; | 30 | u8 cmd; |
| 31 | unsigned int laddr; /* Logical address on the ramsan */ | 31 | unsigned int laddr; /* Logical address */ |
| 32 | struct { | 32 | struct { |
| 33 | u32 off; | 33 | u32 off; |
| 34 | u32 cnt; | 34 | u32 cnt; |
| @@ -81,9 +81,6 @@ enum rsxx_hw_status { | |||
| 81 | HW_STATUS_FAULT = 0x08, | 81 | HW_STATUS_FAULT = 0x08, |
| 82 | }; | 82 | }; |
| 83 | 83 | ||
| 84 | #define STATUS_BUFFER_SIZE8 4096 | ||
| 85 | #define COMMAND_BUFFER_SIZE8 4096 | ||
| 86 | |||
| 87 | static struct kmem_cache *rsxx_dma_pool; | 84 | static struct kmem_cache *rsxx_dma_pool; |
| 88 | 85 | ||
| 89 | struct dma_tracker { | 86 | struct dma_tracker { |
| @@ -122,7 +119,7 @@ static unsigned int rsxx_get_dma_tgt(struct rsxx_cardinfo *card, u64 addr8) | |||
| 122 | return tgt; | 119 | return tgt; |
| 123 | } | 120 | } |
| 124 | 121 | ||
| 125 | static void rsxx_dma_queue_reset(struct rsxx_cardinfo *card) | 122 | void rsxx_dma_queue_reset(struct rsxx_cardinfo *card) |
| 126 | { | 123 | { |
| 127 | /* Reset all DMA Command/Status Queues */ | 124 | /* Reset all DMA Command/Status Queues */ |
| 128 | iowrite32(DMA_QUEUE_RESET, card->regmap + RESET); | 125 | iowrite32(DMA_QUEUE_RESET, card->regmap + RESET); |
| @@ -210,7 +207,8 @@ static void dma_intr_coal_auto_tune(struct rsxx_cardinfo *card) | |||
| 210 | u32 q_depth = 0; | 207 | u32 q_depth = 0; |
| 211 | u32 intr_coal; | 208 | u32 intr_coal; |
| 212 | 209 | ||
| 213 | if (card->config.data.intr_coal.mode != RSXX_INTR_COAL_AUTO_TUNE) | 210 | if (card->config.data.intr_coal.mode != RSXX_INTR_COAL_AUTO_TUNE || |
| 211 | unlikely(card->eeh_state)) | ||
| 214 | return; | 212 | return; |
| 215 | 213 | ||
| 216 | for (i = 0; i < card->n_targets; i++) | 214 | for (i = 0; i < card->n_targets; i++) |
| @@ -223,31 +221,26 @@ static void dma_intr_coal_auto_tune(struct rsxx_cardinfo *card) | |||
| 223 | } | 221 | } |
| 224 | 222 | ||
| 225 | /*----------------- RSXX DMA Handling -------------------*/ | 223 | /*----------------- RSXX DMA Handling -------------------*/ |
| 226 | static void rsxx_complete_dma(struct rsxx_cardinfo *card, | 224 | static void rsxx_complete_dma(struct rsxx_dma_ctrl *ctrl, |
| 227 | struct rsxx_dma *dma, | 225 | struct rsxx_dma *dma, |
| 228 | unsigned int status) | 226 | unsigned int status) |
| 229 | { | 227 | { |
| 230 | if (status & DMA_SW_ERR) | 228 | if (status & DMA_SW_ERR) |
| 231 | printk_ratelimited(KERN_ERR | 229 | ctrl->stats.dma_sw_err++; |
| 232 | "SW Error in DMA(cmd x%02x, laddr x%08x)\n", | ||
| 233 | dma->cmd, dma->laddr); | ||
| 234 | if (status & DMA_HW_FAULT) | 230 | if (status & DMA_HW_FAULT) |
| 235 | printk_ratelimited(KERN_ERR | 231 | ctrl->stats.dma_hw_fault++; |
| 236 | "HW Fault in DMA(cmd x%02x, laddr x%08x)\n", | ||
| 237 | dma->cmd, dma->laddr); | ||
| 238 | if (status & DMA_CANCELLED) | 232 | if (status & DMA_CANCELLED) |
| 239 | printk_ratelimited(KERN_ERR | 233 | ctrl->stats.dma_cancelled++; |
| 240 | "DMA Cancelled(cmd x%02x, laddr x%08x)\n", | ||
| 241 | dma->cmd, dma->laddr); | ||
| 242 | 234 | ||
| 243 | if (dma->dma_addr) | 235 | if (dma->dma_addr) |
| 244 | pci_unmap_page(card->dev, dma->dma_addr, get_dma_size(dma), | 236 | pci_unmap_page(ctrl->card->dev, dma->dma_addr, |
| 237 | get_dma_size(dma), | ||
| 245 | dma->cmd == HW_CMD_BLK_WRITE ? | 238 | dma->cmd == HW_CMD_BLK_WRITE ? |
| 246 | PCI_DMA_TODEVICE : | 239 | PCI_DMA_TODEVICE : |
| 247 | PCI_DMA_FROMDEVICE); | 240 | PCI_DMA_FROMDEVICE); |
| 248 | 241 | ||
| 249 | if (dma->cb) | 242 | if (dma->cb) |
| 250 | dma->cb(card, dma->cb_data, status ? 1 : 0); | 243 | dma->cb(ctrl->card, dma->cb_data, status ? 1 : 0); |
| 251 | 244 | ||
| 252 | kmem_cache_free(rsxx_dma_pool, dma); | 245 | kmem_cache_free(rsxx_dma_pool, dma); |
| 253 | } | 246 | } |
| @@ -330,14 +323,15 @@ static void rsxx_handle_dma_error(struct rsxx_dma_ctrl *ctrl, | |||
| 330 | if (requeue_cmd) | 323 | if (requeue_cmd) |
| 331 | rsxx_requeue_dma(ctrl, dma); | 324 | rsxx_requeue_dma(ctrl, dma); |
| 332 | else | 325 | else |
| 333 | rsxx_complete_dma(ctrl->card, dma, status); | 326 | rsxx_complete_dma(ctrl, dma, status); |
| 334 | } | 327 | } |
| 335 | 328 | ||
| 336 | static void dma_engine_stalled(unsigned long data) | 329 | static void dma_engine_stalled(unsigned long data) |
| 337 | { | 330 | { |
| 338 | struct rsxx_dma_ctrl *ctrl = (struct rsxx_dma_ctrl *)data; | 331 | struct rsxx_dma_ctrl *ctrl = (struct rsxx_dma_ctrl *)data; |
| 339 | 332 | ||
| 340 | if (atomic_read(&ctrl->stats.hw_q_depth) == 0) | 333 | if (atomic_read(&ctrl->stats.hw_q_depth) == 0 || |
| 334 | unlikely(ctrl->card->eeh_state)) | ||
| 341 | return; | 335 | return; |
| 342 | 336 | ||
| 343 | if (ctrl->cmd.idx != ioread32(ctrl->regmap + SW_CMD_IDX)) { | 337 | if (ctrl->cmd.idx != ioread32(ctrl->regmap + SW_CMD_IDX)) { |
| @@ -369,7 +363,8 @@ static void rsxx_issue_dmas(struct work_struct *work) | |||
| 369 | ctrl = container_of(work, struct rsxx_dma_ctrl, issue_dma_work); | 363 | ctrl = container_of(work, struct rsxx_dma_ctrl, issue_dma_work); |
| 370 | hw_cmd_buf = ctrl->cmd.buf; | 364 | hw_cmd_buf = ctrl->cmd.buf; |
| 371 | 365 | ||
| 372 | if (unlikely(ctrl->card->halt)) | 366 | if (unlikely(ctrl->card->halt) || |
| 367 | unlikely(ctrl->card->eeh_state)) | ||
| 373 | return; | 368 | return; |
| 374 | 369 | ||
| 375 | while (1) { | 370 | while (1) { |
| @@ -397,7 +392,7 @@ static void rsxx_issue_dmas(struct work_struct *work) | |||
| 397 | */ | 392 | */ |
| 398 | if (unlikely(ctrl->card->dma_fault)) { | 393 | if (unlikely(ctrl->card->dma_fault)) { |
| 399 | push_tracker(ctrl->trackers, tag); | 394 | push_tracker(ctrl->trackers, tag); |
| 400 | rsxx_complete_dma(ctrl->card, dma, DMA_CANCELLED); | 395 | rsxx_complete_dma(ctrl, dma, DMA_CANCELLED); |
| 401 | continue; | 396 | continue; |
| 402 | } | 397 | } |
| 403 | 398 | ||
| @@ -432,19 +427,15 @@ static void rsxx_issue_dmas(struct work_struct *work) | |||
| 432 | 427 | ||
| 433 | /* Let HW know we've queued commands. */ | 428 | /* Let HW know we've queued commands. */ |
| 434 | if (cmds_pending) { | 429 | if (cmds_pending) { |
| 435 | /* | ||
| 436 | * We must guarantee that the CPU writes to 'ctrl->cmd.buf' | ||
| 437 | * (which is in PCI-consistent system-memory) from the loop | ||
| 438 | * above make it into the coherency domain before the | ||
| 439 | * following PIO "trigger" updating the cmd.idx. A WMB is | ||
| 440 | * sufficient. We need not explicitly CPU cache-flush since | ||
| 441 | * the memory is a PCI-consistent (ie; coherent) mapping. | ||
| 442 | */ | ||
| 443 | wmb(); | ||
| 444 | |||
| 445 | atomic_add(cmds_pending, &ctrl->stats.hw_q_depth); | 430 | atomic_add(cmds_pending, &ctrl->stats.hw_q_depth); |
| 446 | mod_timer(&ctrl->activity_timer, | 431 | mod_timer(&ctrl->activity_timer, |
| 447 | jiffies + DMA_ACTIVITY_TIMEOUT); | 432 | jiffies + DMA_ACTIVITY_TIMEOUT); |
| 433 | |||
| 434 | if (unlikely(ctrl->card->eeh_state)) { | ||
| 435 | del_timer_sync(&ctrl->activity_timer); | ||
| 436 | return; | ||
| 437 | } | ||
| 438 | |||
| 448 | iowrite32(ctrl->cmd.idx, ctrl->regmap + SW_CMD_IDX); | 439 | iowrite32(ctrl->cmd.idx, ctrl->regmap + SW_CMD_IDX); |
| 449 | } | 440 | } |
| 450 | } | 441 | } |
| @@ -463,7 +454,8 @@ static void rsxx_dma_done(struct work_struct *work) | |||
| 463 | hw_st_buf = ctrl->status.buf; | 454 | hw_st_buf = ctrl->status.buf; |
| 464 | 455 | ||
| 465 | if (unlikely(ctrl->card->halt) || | 456 | if (unlikely(ctrl->card->halt) || |
| 466 | unlikely(ctrl->card->dma_fault)) | 457 | unlikely(ctrl->card->dma_fault) || |
| 458 | unlikely(ctrl->card->eeh_state)) | ||
| 467 | return; | 459 | return; |
| 468 | 460 | ||
| 469 | count = le16_to_cpu(hw_st_buf[ctrl->status.idx].count); | 461 | count = le16_to_cpu(hw_st_buf[ctrl->status.idx].count); |
| @@ -508,7 +500,7 @@ static void rsxx_dma_done(struct work_struct *work) | |||
| 508 | if (status) | 500 | if (status) |
| 509 | rsxx_handle_dma_error(ctrl, dma, status); | 501 | rsxx_handle_dma_error(ctrl, dma, status); |
| 510 | else | 502 | else |
| 511 | rsxx_complete_dma(ctrl->card, dma, 0); | 503 | rsxx_complete_dma(ctrl, dma, 0); |
| 512 | 504 | ||
| 513 | push_tracker(ctrl->trackers, tag); | 505 | push_tracker(ctrl->trackers, tag); |
| 514 | 506 | ||
| @@ -727,20 +719,54 @@ bvec_err: | |||
| 727 | 719 | ||
| 728 | 720 | ||
| 729 | /*----------------- DMA Engine Initialization & Setup -------------------*/ | 721 | /*----------------- DMA Engine Initialization & Setup -------------------*/ |
| 722 | int rsxx_hw_buffers_init(struct pci_dev *dev, struct rsxx_dma_ctrl *ctrl) | ||
| 723 | { | ||
| 724 | ctrl->status.buf = pci_alloc_consistent(dev, STATUS_BUFFER_SIZE8, | ||
| 725 | &ctrl->status.dma_addr); | ||
| 726 | ctrl->cmd.buf = pci_alloc_consistent(dev, COMMAND_BUFFER_SIZE8, | ||
| 727 | &ctrl->cmd.dma_addr); | ||
| 728 | if (ctrl->status.buf == NULL || ctrl->cmd.buf == NULL) | ||
| 729 | return -ENOMEM; | ||
| 730 | |||
| 731 | memset(ctrl->status.buf, 0xac, STATUS_BUFFER_SIZE8); | ||
| 732 | iowrite32(lower_32_bits(ctrl->status.dma_addr), | ||
| 733 | ctrl->regmap + SB_ADD_LO); | ||
| 734 | iowrite32(upper_32_bits(ctrl->status.dma_addr), | ||
| 735 | ctrl->regmap + SB_ADD_HI); | ||
| 736 | |||
| 737 | memset(ctrl->cmd.buf, 0x83, COMMAND_BUFFER_SIZE8); | ||
| 738 | iowrite32(lower_32_bits(ctrl->cmd.dma_addr), ctrl->regmap + CB_ADD_LO); | ||
| 739 | iowrite32(upper_32_bits(ctrl->cmd.dma_addr), ctrl->regmap + CB_ADD_HI); | ||
| 740 | |||
| 741 | ctrl->status.idx = ioread32(ctrl->regmap + HW_STATUS_CNT); | ||
| 742 | if (ctrl->status.idx > RSXX_MAX_OUTSTANDING_CMDS) { | ||
| 743 | dev_crit(&dev->dev, "Failed reading status cnt x%x\n", | ||
| 744 | ctrl->status.idx); | ||
| 745 | return -EINVAL; | ||
| 746 | } | ||
| 747 | iowrite32(ctrl->status.idx, ctrl->regmap + HW_STATUS_CNT); | ||
| 748 | iowrite32(ctrl->status.idx, ctrl->regmap + SW_STATUS_CNT); | ||
| 749 | |||
| 750 | ctrl->cmd.idx = ioread32(ctrl->regmap + HW_CMD_IDX); | ||
| 751 | if (ctrl->cmd.idx > RSXX_MAX_OUTSTANDING_CMDS) { | ||
| 752 | dev_crit(&dev->dev, "Failed reading cmd cnt x%x\n", | ||
| 753 | ctrl->status.idx); | ||
| 754 | return -EINVAL; | ||
| 755 | } | ||
| 756 | iowrite32(ctrl->cmd.idx, ctrl->regmap + HW_CMD_IDX); | ||
| 757 | iowrite32(ctrl->cmd.idx, ctrl->regmap + SW_CMD_IDX); | ||
| 758 | |||
| 759 | return 0; | ||
| 760 | } | ||
| 761 | |||
| 730 | static int rsxx_dma_ctrl_init(struct pci_dev *dev, | 762 | static int rsxx_dma_ctrl_init(struct pci_dev *dev, |
| 731 | struct rsxx_dma_ctrl *ctrl) | 763 | struct rsxx_dma_ctrl *ctrl) |
| 732 | { | 764 | { |
| 733 | int i; | 765 | int i; |
| 766 | int st; | ||
| 734 | 767 | ||
| 735 | memset(&ctrl->stats, 0, sizeof(ctrl->stats)); | 768 | memset(&ctrl->stats, 0, sizeof(ctrl->stats)); |
| 736 | 769 | ||
| 737 | ctrl->status.buf = pci_alloc_consistent(dev, STATUS_BUFFER_SIZE8, | ||
| 738 | &ctrl->status.dma_addr); | ||
| 739 | ctrl->cmd.buf = pci_alloc_consistent(dev, COMMAND_BUFFER_SIZE8, | ||
| 740 | &ctrl->cmd.dma_addr); | ||
| 741 | if (ctrl->status.buf == NULL || ctrl->cmd.buf == NULL) | ||
| 742 | return -ENOMEM; | ||
| 743 | |||
| 744 | ctrl->trackers = vmalloc(DMA_TRACKER_LIST_SIZE8); | 770 | ctrl->trackers = vmalloc(DMA_TRACKER_LIST_SIZE8); |
| 745 | if (!ctrl->trackers) | 771 | if (!ctrl->trackers) |
| 746 | return -ENOMEM; | 772 | return -ENOMEM; |
| @@ -770,35 +796,9 @@ static int rsxx_dma_ctrl_init(struct pci_dev *dev, | |||
| 770 | INIT_WORK(&ctrl->issue_dma_work, rsxx_issue_dmas); | 796 | INIT_WORK(&ctrl->issue_dma_work, rsxx_issue_dmas); |
| 771 | INIT_WORK(&ctrl->dma_done_work, rsxx_dma_done); | 797 | INIT_WORK(&ctrl->dma_done_work, rsxx_dma_done); |
| 772 | 798 | ||
| 773 | memset(ctrl->status.buf, 0xac, STATUS_BUFFER_SIZE8); | 799 | st = rsxx_hw_buffers_init(dev, ctrl); |
| 774 | iowrite32(lower_32_bits(ctrl->status.dma_addr), | 800 | if (st) |
| 775 | ctrl->regmap + SB_ADD_LO); | 801 | return st; |
| 776 | iowrite32(upper_32_bits(ctrl->status.dma_addr), | ||
| 777 | ctrl->regmap + SB_ADD_HI); | ||
| 778 | |||
| 779 | memset(ctrl->cmd.buf, 0x83, COMMAND_BUFFER_SIZE8); | ||
| 780 | iowrite32(lower_32_bits(ctrl->cmd.dma_addr), ctrl->regmap + CB_ADD_LO); | ||
| 781 | iowrite32(upper_32_bits(ctrl->cmd.dma_addr), ctrl->regmap + CB_ADD_HI); | ||
| 782 | |||
| 783 | ctrl->status.idx = ioread32(ctrl->regmap + HW_STATUS_CNT); | ||
| 784 | if (ctrl->status.idx > RSXX_MAX_OUTSTANDING_CMDS) { | ||
| 785 | dev_crit(&dev->dev, "Failed reading status cnt x%x\n", | ||
| 786 | ctrl->status.idx); | ||
| 787 | return -EINVAL; | ||
| 788 | } | ||
| 789 | iowrite32(ctrl->status.idx, ctrl->regmap + HW_STATUS_CNT); | ||
| 790 | iowrite32(ctrl->status.idx, ctrl->regmap + SW_STATUS_CNT); | ||
| 791 | |||
| 792 | ctrl->cmd.idx = ioread32(ctrl->regmap + HW_CMD_IDX); | ||
| 793 | if (ctrl->cmd.idx > RSXX_MAX_OUTSTANDING_CMDS) { | ||
| 794 | dev_crit(&dev->dev, "Failed reading cmd cnt x%x\n", | ||
| 795 | ctrl->status.idx); | ||
| 796 | return -EINVAL; | ||
| 797 | } | ||
| 798 | iowrite32(ctrl->cmd.idx, ctrl->regmap + HW_CMD_IDX); | ||
| 799 | iowrite32(ctrl->cmd.idx, ctrl->regmap + SW_CMD_IDX); | ||
| 800 | |||
| 801 | wmb(); | ||
| 802 | 802 | ||
| 803 | return 0; | 803 | return 0; |
| 804 | } | 804 | } |
| @@ -834,7 +834,7 @@ static int rsxx_dma_stripe_setup(struct rsxx_cardinfo *card, | |||
| 834 | return 0; | 834 | return 0; |
| 835 | } | 835 | } |
| 836 | 836 | ||
| 837 | static int rsxx_dma_configure(struct rsxx_cardinfo *card) | 837 | int rsxx_dma_configure(struct rsxx_cardinfo *card) |
| 838 | { | 838 | { |
| 839 | u32 intr_coal; | 839 | u32 intr_coal; |
| 840 | 840 | ||
| @@ -980,6 +980,103 @@ void rsxx_dma_destroy(struct rsxx_cardinfo *card) | |||
| 980 | } | 980 | } |
| 981 | } | 981 | } |
| 982 | 982 | ||
| 983 | int rsxx_eeh_save_issued_dmas(struct rsxx_cardinfo *card) | ||
| 984 | { | ||
| 985 | int i; | ||
| 986 | int j; | ||
| 987 | int cnt; | ||
| 988 | struct rsxx_dma *dma; | ||
| 989 | struct list_head *issued_dmas; | ||
| 990 | |||
| 991 | issued_dmas = kzalloc(sizeof(*issued_dmas) * card->n_targets, | ||
| 992 | GFP_KERNEL); | ||
| 993 | if (!issued_dmas) | ||
| 994 | return -ENOMEM; | ||
| 995 | |||
| 996 | for (i = 0; i < card->n_targets; i++) { | ||
| 997 | INIT_LIST_HEAD(&issued_dmas[i]); | ||
| 998 | cnt = 0; | ||
| 999 | for (j = 0; j < RSXX_MAX_OUTSTANDING_CMDS; j++) { | ||
| 1000 | dma = get_tracker_dma(card->ctrl[i].trackers, j); | ||
| 1001 | if (dma == NULL) | ||
| 1002 | continue; | ||
| 1003 | |||
| 1004 | if (dma->cmd == HW_CMD_BLK_WRITE) | ||
| 1005 | card->ctrl[i].stats.writes_issued--; | ||
| 1006 | else if (dma->cmd == HW_CMD_BLK_DISCARD) | ||
| 1007 | card->ctrl[i].stats.discards_issued--; | ||
| 1008 | else | ||
| 1009 | card->ctrl[i].stats.reads_issued--; | ||
| 1010 | |||
| 1011 | list_add_tail(&dma->list, &issued_dmas[i]); | ||
| 1012 | push_tracker(card->ctrl[i].trackers, j); | ||
| 1013 | cnt++; | ||
| 1014 | } | ||
| 1015 | |||
| 1016 | spin_lock(&card->ctrl[i].queue_lock); | ||
| 1017 | list_splice(&issued_dmas[i], &card->ctrl[i].queue); | ||
| 1018 | |||
| 1019 | atomic_sub(cnt, &card->ctrl[i].stats.hw_q_depth); | ||
| 1020 | card->ctrl[i].stats.sw_q_depth += cnt; | ||
| 1021 | card->ctrl[i].e_cnt = 0; | ||
| 1022 | |||
| 1023 | list_for_each_entry(dma, &card->ctrl[i].queue, list) { | ||
| 1024 | if (dma->dma_addr) | ||
| 1025 | pci_unmap_page(card->dev, dma->dma_addr, | ||
| 1026 | get_dma_size(dma), | ||
| 1027 | dma->cmd == HW_CMD_BLK_WRITE ? | ||
| 1028 | PCI_DMA_TODEVICE : | ||
| 1029 | PCI_DMA_FROMDEVICE); | ||
| 1030 | } | ||
| 1031 | spin_unlock(&card->ctrl[i].queue_lock); | ||
| 1032 | } | ||
| 1033 | |||
| 1034 | kfree(issued_dmas); | ||
| 1035 | |||
| 1036 | return 0; | ||
| 1037 | } | ||
| 1038 | |||
| 1039 | void rsxx_eeh_cancel_dmas(struct rsxx_cardinfo *card) | ||
| 1040 | { | ||
| 1041 | struct rsxx_dma *dma; | ||
| 1042 | struct rsxx_dma *tmp; | ||
| 1043 | int i; | ||
| 1044 | |||
| 1045 | for (i = 0; i < card->n_targets; i++) { | ||
| 1046 | spin_lock(&card->ctrl[i].queue_lock); | ||
| 1047 | list_for_each_entry_safe(dma, tmp, &card->ctrl[i].queue, list) { | ||
| 1048 | list_del(&dma->list); | ||
| 1049 | |||
| 1050 | rsxx_complete_dma(&card->ctrl[i], dma, DMA_CANCELLED); | ||
| 1051 | } | ||
| 1052 | spin_unlock(&card->ctrl[i].queue_lock); | ||
| 1053 | } | ||
| 1054 | } | ||
| 1055 | |||
| 1056 | int rsxx_eeh_remap_dmas(struct rsxx_cardinfo *card) | ||
| 1057 | { | ||
| 1058 | struct rsxx_dma *dma; | ||
| 1059 | int i; | ||
| 1060 | |||
| 1061 | for (i = 0; i < card->n_targets; i++) { | ||
| 1062 | spin_lock(&card->ctrl[i].queue_lock); | ||
| 1063 | list_for_each_entry(dma, &card->ctrl[i].queue, list) { | ||
| 1064 | dma->dma_addr = pci_map_page(card->dev, dma->page, | ||
| 1065 | dma->pg_off, get_dma_size(dma), | ||
| 1066 | dma->cmd == HW_CMD_BLK_WRITE ? | ||
| 1067 | PCI_DMA_TODEVICE : | ||
| 1068 | PCI_DMA_FROMDEVICE); | ||
| 1069 | if (!dma->dma_addr) { | ||
| 1070 | spin_unlock(&card->ctrl[i].queue_lock); | ||
| 1071 | kmem_cache_free(rsxx_dma_pool, dma); | ||
| 1072 | return -ENOMEM; | ||
| 1073 | } | ||
| 1074 | } | ||
| 1075 | spin_unlock(&card->ctrl[i].queue_lock); | ||
| 1076 | } | ||
| 1077 | |||
| 1078 | return 0; | ||
| 1079 | } | ||
| 983 | 1080 | ||
| 984 | int rsxx_dma_init(void) | 1081 | int rsxx_dma_init(void) |
| 985 | { | 1082 | { |
diff --git a/drivers/block/rsxx/rsxx.h b/drivers/block/rsxx/rsxx.h index 2e50b65902b7..24ba3642bd89 100644 --- a/drivers/block/rsxx/rsxx.h +++ b/drivers/block/rsxx/rsxx.h | |||
| @@ -27,15 +27,17 @@ | |||
| 27 | 27 | ||
| 28 | /*----------------- IOCTL Definitions -------------------*/ | 28 | /*----------------- IOCTL Definitions -------------------*/ |
| 29 | 29 | ||
| 30 | #define RSXX_MAX_DATA 8 | ||
| 31 | |||
| 30 | struct rsxx_reg_access { | 32 | struct rsxx_reg_access { |
| 31 | __u32 addr; | 33 | __u32 addr; |
| 32 | __u32 cnt; | 34 | __u32 cnt; |
| 33 | __u32 stat; | 35 | __u32 stat; |
| 34 | __u32 stream; | 36 | __u32 stream; |
| 35 | __u32 data[8]; | 37 | __u32 data[RSXX_MAX_DATA]; |
| 36 | }; | 38 | }; |
| 37 | 39 | ||
| 38 | #define RSXX_MAX_REG_CNT (8 * (sizeof(__u32))) | 40 | #define RSXX_MAX_REG_CNT (RSXX_MAX_DATA * (sizeof(__u32))) |
| 39 | 41 | ||
| 40 | #define RSXX_IOC_MAGIC 'r' | 42 | #define RSXX_IOC_MAGIC 'r' |
| 41 | 43 | ||
diff --git a/drivers/block/rsxx/rsxx_cfg.h b/drivers/block/rsxx/rsxx_cfg.h index c025fe5fdb70..f384c943846d 100644 --- a/drivers/block/rsxx/rsxx_cfg.h +++ b/drivers/block/rsxx/rsxx_cfg.h | |||
| @@ -58,7 +58,7 @@ struct rsxx_card_cfg { | |||
| 58 | }; | 58 | }; |
| 59 | 59 | ||
| 60 | /* Vendor ID Values */ | 60 | /* Vendor ID Values */ |
| 61 | #define RSXX_VENDOR_ID_TMS_IBM 0 | 61 | #define RSXX_VENDOR_ID_IBM 0 |
| 62 | #define RSXX_VENDOR_ID_DSI 1 | 62 | #define RSXX_VENDOR_ID_DSI 1 |
| 63 | #define RSXX_VENDOR_COUNT 2 | 63 | #define RSXX_VENDOR_COUNT 2 |
| 64 | 64 | ||
diff --git a/drivers/block/rsxx/rsxx_priv.h b/drivers/block/rsxx/rsxx_priv.h index a1ac907d8f4c..382e8bf5c03b 100644 --- a/drivers/block/rsxx/rsxx_priv.h +++ b/drivers/block/rsxx/rsxx_priv.h | |||
| @@ -45,16 +45,13 @@ | |||
| 45 | 45 | ||
| 46 | struct proc_cmd; | 46 | struct proc_cmd; |
| 47 | 47 | ||
| 48 | #define PCI_VENDOR_ID_TMS_IBM 0x15B6 | 48 | #define PCI_DEVICE_ID_FS70_FLASH 0x04A9 |
| 49 | #define PCI_DEVICE_ID_RS70_FLASH 0x0019 | 49 | #define PCI_DEVICE_ID_FS80_FLASH 0x04AA |
| 50 | #define PCI_DEVICE_ID_RS70D_FLASH 0x001A | ||
| 51 | #define PCI_DEVICE_ID_RS80_FLASH 0x001C | ||
| 52 | #define PCI_DEVICE_ID_RS81_FLASH 0x001E | ||
| 53 | 50 | ||
| 54 | #define RS70_PCI_REV_SUPPORTED 4 | 51 | #define RS70_PCI_REV_SUPPORTED 4 |
| 55 | 52 | ||
| 56 | #define DRIVER_NAME "rsxx" | 53 | #define DRIVER_NAME "rsxx" |
| 57 | #define DRIVER_VERSION "3.7" | 54 | #define DRIVER_VERSION "4.0" |
| 58 | 55 | ||
| 59 | /* Block size is 4096 */ | 56 | /* Block size is 4096 */ |
| 60 | #define RSXX_HW_BLK_SHIFT 12 | 57 | #define RSXX_HW_BLK_SHIFT 12 |
| @@ -67,6 +64,9 @@ struct proc_cmd; | |||
| 67 | #define RSXX_MAX_OUTSTANDING_CMDS 255 | 64 | #define RSXX_MAX_OUTSTANDING_CMDS 255 |
| 68 | #define RSXX_CS_IDX_MASK 0xff | 65 | #define RSXX_CS_IDX_MASK 0xff |
| 69 | 66 | ||
| 67 | #define STATUS_BUFFER_SIZE8 4096 | ||
| 68 | #define COMMAND_BUFFER_SIZE8 4096 | ||
| 69 | |||
| 70 | #define RSXX_MAX_TARGETS 8 | 70 | #define RSXX_MAX_TARGETS 8 |
| 71 | 71 | ||
| 72 | struct dma_tracker_list; | 72 | struct dma_tracker_list; |
| @@ -91,6 +91,9 @@ struct rsxx_dma_stats { | |||
| 91 | u32 discards_failed; | 91 | u32 discards_failed; |
| 92 | u32 done_rescheduled; | 92 | u32 done_rescheduled; |
| 93 | u32 issue_rescheduled; | 93 | u32 issue_rescheduled; |
| 94 | u32 dma_sw_err; | ||
| 95 | u32 dma_hw_fault; | ||
| 96 | u32 dma_cancelled; | ||
| 94 | u32 sw_q_depth; /* Number of DMAs on the SW queue. */ | 97 | u32 sw_q_depth; /* Number of DMAs on the SW queue. */ |
| 95 | atomic_t hw_q_depth; /* Number of DMAs queued to HW. */ | 98 | atomic_t hw_q_depth; /* Number of DMAs queued to HW. */ |
| 96 | }; | 99 | }; |
| @@ -116,6 +119,7 @@ struct rsxx_dma_ctrl { | |||
| 116 | struct rsxx_cardinfo { | 119 | struct rsxx_cardinfo { |
| 117 | struct pci_dev *dev; | 120 | struct pci_dev *dev; |
| 118 | unsigned int halt; | 121 | unsigned int halt; |
| 122 | unsigned int eeh_state; | ||
| 119 | 123 | ||
| 120 | void __iomem *regmap; | 124 | void __iomem *regmap; |
| 121 | spinlock_t irq_lock; | 125 | spinlock_t irq_lock; |
| @@ -224,6 +228,7 @@ enum rsxx_pci_regmap { | |||
| 224 | PERF_RD512_HI = 0xac, | 228 | PERF_RD512_HI = 0xac, |
| 225 | PERF_WR512_LO = 0xb0, | 229 | PERF_WR512_LO = 0xb0, |
| 226 | PERF_WR512_HI = 0xb4, | 230 | PERF_WR512_HI = 0xb4, |
| 231 | PCI_RECONFIG = 0xb8, | ||
| 227 | }; | 232 | }; |
| 228 | 233 | ||
| 229 | enum rsxx_intr { | 234 | enum rsxx_intr { |
| @@ -237,6 +242,8 @@ enum rsxx_intr { | |||
| 237 | CR_INTR_DMA5 = 0x00000080, | 242 | CR_INTR_DMA5 = 0x00000080, |
| 238 | CR_INTR_DMA6 = 0x00000100, | 243 | CR_INTR_DMA6 = 0x00000100, |
| 239 | CR_INTR_DMA7 = 0x00000200, | 244 | CR_INTR_DMA7 = 0x00000200, |
| 245 | CR_INTR_ALL_C = 0x0000003f, | ||
| 246 | CR_INTR_ALL_G = 0x000003ff, | ||
| 240 | CR_INTR_DMA_ALL = 0x000003f5, | 247 | CR_INTR_DMA_ALL = 0x000003f5, |
| 241 | CR_INTR_ALL = 0xffffffff, | 248 | CR_INTR_ALL = 0xffffffff, |
| 242 | }; | 249 | }; |
| @@ -253,8 +260,14 @@ enum rsxx_pci_reset { | |||
| 253 | DMA_QUEUE_RESET = 0x00000001, | 260 | DMA_QUEUE_RESET = 0x00000001, |
| 254 | }; | 261 | }; |
| 255 | 262 | ||
| 263 | enum rsxx_hw_fifo_flush { | ||
| 264 | RSXX_FLUSH_BUSY = 0x00000002, | ||
| 265 | RSXX_FLUSH_TIMEOUT = 0x00000004, | ||
| 266 | }; | ||
| 267 | |||
| 256 | enum rsxx_pci_revision { | 268 | enum rsxx_pci_revision { |
| 257 | RSXX_DISCARD_SUPPORT = 2, | 269 | RSXX_DISCARD_SUPPORT = 2, |
| 270 | RSXX_EEH_SUPPORT = 3, | ||
| 258 | }; | 271 | }; |
| 259 | 272 | ||
| 260 | enum rsxx_creg_cmd { | 273 | enum rsxx_creg_cmd { |
| @@ -360,11 +373,17 @@ int rsxx_dma_setup(struct rsxx_cardinfo *card); | |||
| 360 | void rsxx_dma_destroy(struct rsxx_cardinfo *card); | 373 | void rsxx_dma_destroy(struct rsxx_cardinfo *card); |
| 361 | int rsxx_dma_init(void); | 374 | int rsxx_dma_init(void); |
| 362 | void rsxx_dma_cleanup(void); | 375 | void rsxx_dma_cleanup(void); |
| 376 | void rsxx_dma_queue_reset(struct rsxx_cardinfo *card); | ||
| 377 | int rsxx_dma_configure(struct rsxx_cardinfo *card); | ||
| 363 | int rsxx_dma_queue_bio(struct rsxx_cardinfo *card, | 378 | int rsxx_dma_queue_bio(struct rsxx_cardinfo *card, |
| 364 | struct bio *bio, | 379 | struct bio *bio, |
| 365 | atomic_t *n_dmas, | 380 | atomic_t *n_dmas, |
| 366 | rsxx_dma_cb cb, | 381 | rsxx_dma_cb cb, |
| 367 | void *cb_data); | 382 | void *cb_data); |
| 383 | int rsxx_hw_buffers_init(struct pci_dev *dev, struct rsxx_dma_ctrl *ctrl); | ||
| 384 | int rsxx_eeh_save_issued_dmas(struct rsxx_cardinfo *card); | ||
| 385 | void rsxx_eeh_cancel_dmas(struct rsxx_cardinfo *card); | ||
| 386 | int rsxx_eeh_remap_dmas(struct rsxx_cardinfo *card); | ||
| 368 | 387 | ||
| 369 | /***** cregs.c *****/ | 388 | /***** cregs.c *****/ |
| 370 | int rsxx_creg_write(struct rsxx_cardinfo *card, u32 addr, | 389 | int rsxx_creg_write(struct rsxx_cardinfo *card, u32 addr, |
| @@ -389,10 +408,11 @@ int rsxx_creg_setup(struct rsxx_cardinfo *card); | |||
| 389 | void rsxx_creg_destroy(struct rsxx_cardinfo *card); | 408 | void rsxx_creg_destroy(struct rsxx_cardinfo *card); |
| 390 | int rsxx_creg_init(void); | 409 | int rsxx_creg_init(void); |
| 391 | void rsxx_creg_cleanup(void); | 410 | void rsxx_creg_cleanup(void); |
| 392 | |||
| 393 | int rsxx_reg_access(struct rsxx_cardinfo *card, | 411 | int rsxx_reg_access(struct rsxx_cardinfo *card, |
| 394 | struct rsxx_reg_access __user *ucmd, | 412 | struct rsxx_reg_access __user *ucmd, |
| 395 | int read); | 413 | int read); |
| 414 | void rsxx_eeh_save_issued_creg(struct rsxx_cardinfo *card); | ||
| 415 | void rsxx_kick_creg_queue(struct rsxx_cardinfo *card); | ||
| 396 | 416 | ||
| 397 | 417 | ||
| 398 | 418 | ||
diff --git a/drivers/block/xen-blkback/blkback.c b/drivers/block/xen-blkback/blkback.c index de1f319f7bd7..dd5b2fed97e9 100644 --- a/drivers/block/xen-blkback/blkback.c +++ b/drivers/block/xen-blkback/blkback.c | |||
| @@ -164,7 +164,7 @@ static void make_response(struct xen_blkif *blkif, u64 id, | |||
| 164 | 164 | ||
| 165 | #define foreach_grant_safe(pos, n, rbtree, node) \ | 165 | #define foreach_grant_safe(pos, n, rbtree, node) \ |
| 166 | for ((pos) = container_of(rb_first((rbtree)), typeof(*(pos)), node), \ | 166 | for ((pos) = container_of(rb_first((rbtree)), typeof(*(pos)), node), \ |
| 167 | (n) = rb_next(&(pos)->node); \ | 167 | (n) = (&(pos)->node != NULL) ? rb_next(&(pos)->node) : NULL; \ |
| 168 | &(pos)->node != NULL; \ | 168 | &(pos)->node != NULL; \ |
| 169 | (pos) = container_of(n, typeof(*(pos)), node), \ | 169 | (pos) = container_of(n, typeof(*(pos)), node), \ |
| 170 | (n) = (&(pos)->node != NULL) ? rb_next(&(pos)->node) : NULL) | 170 | (n) = (&(pos)->node != NULL) ? rb_next(&(pos)->node) : NULL) |
| @@ -381,8 +381,8 @@ irqreturn_t xen_blkif_be_int(int irq, void *dev_id) | |||
| 381 | 381 | ||
| 382 | static void print_stats(struct xen_blkif *blkif) | 382 | static void print_stats(struct xen_blkif *blkif) |
| 383 | { | 383 | { |
| 384 | pr_info("xen-blkback (%s): oo %3d | rd %4d | wr %4d | f %4d" | 384 | pr_info("xen-blkback (%s): oo %3llu | rd %4llu | wr %4llu | f %4llu" |
| 385 | " | ds %4d\n", | 385 | " | ds %4llu\n", |
| 386 | current->comm, blkif->st_oo_req, | 386 | current->comm, blkif->st_oo_req, |
| 387 | blkif->st_rd_req, blkif->st_wr_req, | 387 | blkif->st_rd_req, blkif->st_wr_req, |
| 388 | blkif->st_f_req, blkif->st_ds_req); | 388 | blkif->st_f_req, blkif->st_ds_req); |
| @@ -442,7 +442,7 @@ int xen_blkif_schedule(void *arg) | |||
| 442 | } | 442 | } |
| 443 | 443 | ||
| 444 | struct seg_buf { | 444 | struct seg_buf { |
| 445 | unsigned long buf; | 445 | unsigned int offset; |
| 446 | unsigned int nsec; | 446 | unsigned int nsec; |
| 447 | }; | 447 | }; |
| 448 | /* | 448 | /* |
| @@ -621,30 +621,21 @@ static int xen_blkbk_map(struct blkif_request *req, | |||
| 621 | * If this is a new persistent grant | 621 | * If this is a new persistent grant |
| 622 | * save the handler | 622 | * save the handler |
| 623 | */ | 623 | */ |
| 624 | persistent_gnts[i]->handle = map[j].handle; | 624 | persistent_gnts[i]->handle = map[j++].handle; |
| 625 | persistent_gnts[i]->dev_bus_addr = | ||
| 626 | map[j++].dev_bus_addr; | ||
| 627 | } | 625 | } |
| 628 | pending_handle(pending_req, i) = | 626 | pending_handle(pending_req, i) = |
| 629 | persistent_gnts[i]->handle; | 627 | persistent_gnts[i]->handle; |
| 630 | 628 | ||
| 631 | if (ret) | 629 | if (ret) |
| 632 | continue; | 630 | continue; |
| 633 | |||
| 634 | seg[i].buf = persistent_gnts[i]->dev_bus_addr | | ||
| 635 | (req->u.rw.seg[i].first_sect << 9); | ||
| 636 | } else { | 631 | } else { |
| 637 | pending_handle(pending_req, i) = map[j].handle; | 632 | pending_handle(pending_req, i) = map[j++].handle; |
| 638 | bitmap_set(pending_req->unmap_seg, i, 1); | 633 | bitmap_set(pending_req->unmap_seg, i, 1); |
| 639 | 634 | ||
| 640 | if (ret) { | 635 | if (ret) |
| 641 | j++; | ||
| 642 | continue; | 636 | continue; |
| 643 | } | ||
| 644 | |||
| 645 | seg[i].buf = map[j++].dev_bus_addr | | ||
| 646 | (req->u.rw.seg[i].first_sect << 9); | ||
| 647 | } | 637 | } |
| 638 | seg[i].offset = (req->u.rw.seg[i].first_sect << 9); | ||
| 648 | } | 639 | } |
| 649 | return ret; | 640 | return ret; |
| 650 | } | 641 | } |
| @@ -679,6 +670,16 @@ static int dispatch_discard_io(struct xen_blkif *blkif, | |||
| 679 | return err; | 670 | return err; |
| 680 | } | 671 | } |
| 681 | 672 | ||
| 673 | static int dispatch_other_io(struct xen_blkif *blkif, | ||
| 674 | struct blkif_request *req, | ||
| 675 | struct pending_req *pending_req) | ||
| 676 | { | ||
| 677 | free_req(pending_req); | ||
| 678 | make_response(blkif, req->u.other.id, req->operation, | ||
| 679 | BLKIF_RSP_EOPNOTSUPP); | ||
| 680 | return -EIO; | ||
| 681 | } | ||
| 682 | |||
| 682 | static void xen_blk_drain_io(struct xen_blkif *blkif) | 683 | static void xen_blk_drain_io(struct xen_blkif *blkif) |
| 683 | { | 684 | { |
| 684 | atomic_set(&blkif->drain, 1); | 685 | atomic_set(&blkif->drain, 1); |
| @@ -800,17 +801,30 @@ __do_block_io_op(struct xen_blkif *blkif) | |||
| 800 | 801 | ||
| 801 | /* Apply all sanity checks to /private copy/ of request. */ | 802 | /* Apply all sanity checks to /private copy/ of request. */ |
| 802 | barrier(); | 803 | barrier(); |
| 803 | if (unlikely(req.operation == BLKIF_OP_DISCARD)) { | 804 | |
| 805 | switch (req.operation) { | ||
| 806 | case BLKIF_OP_READ: | ||
| 807 | case BLKIF_OP_WRITE: | ||
| 808 | case BLKIF_OP_WRITE_BARRIER: | ||
| 809 | case BLKIF_OP_FLUSH_DISKCACHE: | ||
| 810 | if (dispatch_rw_block_io(blkif, &req, pending_req)) | ||
| 811 | goto done; | ||
| 812 | break; | ||
| 813 | case BLKIF_OP_DISCARD: | ||
| 804 | free_req(pending_req); | 814 | free_req(pending_req); |
| 805 | if (dispatch_discard_io(blkif, &req)) | 815 | if (dispatch_discard_io(blkif, &req)) |
| 806 | break; | 816 | goto done; |
| 807 | } else if (dispatch_rw_block_io(blkif, &req, pending_req)) | ||
| 808 | break; | 817 | break; |
| 818 | default: | ||
| 819 | if (dispatch_other_io(blkif, &req, pending_req)) | ||
| 820 | goto done; | ||
| 821 | break; | ||
| 822 | } | ||
| 809 | 823 | ||
| 810 | /* Yield point for this unbounded loop. */ | 824 | /* Yield point for this unbounded loop. */ |
| 811 | cond_resched(); | 825 | cond_resched(); |
| 812 | } | 826 | } |
| 813 | 827 | done: | |
| 814 | return more_to_do; | 828 | return more_to_do; |
| 815 | } | 829 | } |
| 816 | 830 | ||
| @@ -904,7 +918,8 @@ static int dispatch_rw_block_io(struct xen_blkif *blkif, | |||
| 904 | pr_debug(DRV_PFX "access denied: %s of [%llu,%llu] on dev=%04x\n", | 918 | pr_debug(DRV_PFX "access denied: %s of [%llu,%llu] on dev=%04x\n", |
| 905 | operation == READ ? "read" : "write", | 919 | operation == READ ? "read" : "write", |
| 906 | preq.sector_number, | 920 | preq.sector_number, |
| 907 | preq.sector_number + preq.nr_sects, preq.dev); | 921 | preq.sector_number + preq.nr_sects, |
| 922 | blkif->vbd.pdevice); | ||
| 908 | goto fail_response; | 923 | goto fail_response; |
| 909 | } | 924 | } |
| 910 | 925 | ||
| @@ -947,7 +962,7 @@ static int dispatch_rw_block_io(struct xen_blkif *blkif, | |||
| 947 | (bio_add_page(bio, | 962 | (bio_add_page(bio, |
| 948 | pages[i], | 963 | pages[i], |
| 949 | seg[i].nsec << 9, | 964 | seg[i].nsec << 9, |
| 950 | seg[i].buf & ~PAGE_MASK) == 0)) { | 965 | seg[i].offset) == 0)) { |
| 951 | 966 | ||
| 952 | bio = bio_alloc(GFP_KERNEL, nseg-i); | 967 | bio = bio_alloc(GFP_KERNEL, nseg-i); |
| 953 | if (unlikely(bio == NULL)) | 968 | if (unlikely(bio == NULL)) |
| @@ -977,13 +992,7 @@ static int dispatch_rw_block_io(struct xen_blkif *blkif, | |||
| 977 | bio->bi_end_io = end_block_io_op; | 992 | bio->bi_end_io = end_block_io_op; |
| 978 | } | 993 | } |
| 979 | 994 | ||
| 980 | /* | ||
| 981 | * We set it one so that the last submit_bio does not have to call | ||
| 982 | * atomic_inc. | ||
| 983 | */ | ||
| 984 | atomic_set(&pending_req->pendcnt, nbio); | 995 | atomic_set(&pending_req->pendcnt, nbio); |
| 985 | |||
| 986 | /* Get a reference count for the disk queue and start sending I/O */ | ||
| 987 | blk_start_plug(&plug); | 996 | blk_start_plug(&plug); |
| 988 | 997 | ||
| 989 | for (i = 0; i < nbio; i++) | 998 | for (i = 0; i < nbio; i++) |
| @@ -1011,6 +1020,7 @@ static int dispatch_rw_block_io(struct xen_blkif *blkif, | |||
| 1011 | fail_put_bio: | 1020 | fail_put_bio: |
| 1012 | for (i = 0; i < nbio; i++) | 1021 | for (i = 0; i < nbio; i++) |
| 1013 | bio_put(biolist[i]); | 1022 | bio_put(biolist[i]); |
| 1023 | atomic_set(&pending_req->pendcnt, 1); | ||
| 1014 | __end_block_io_op(pending_req, -EINVAL); | 1024 | __end_block_io_op(pending_req, -EINVAL); |
| 1015 | msleep(1); /* back off a bit */ | 1025 | msleep(1); /* back off a bit */ |
| 1016 | return -EIO; | 1026 | return -EIO; |
diff --git a/drivers/block/xen-blkback/common.h b/drivers/block/xen-blkback/common.h index 6072390c7f57..60103e2517ba 100644 --- a/drivers/block/xen-blkback/common.h +++ b/drivers/block/xen-blkback/common.h | |||
| @@ -77,11 +77,18 @@ struct blkif_x86_32_request_discard { | |||
| 77 | uint64_t nr_sectors; | 77 | uint64_t nr_sectors; |
| 78 | } __attribute__((__packed__)); | 78 | } __attribute__((__packed__)); |
| 79 | 79 | ||
| 80 | struct blkif_x86_32_request_other { | ||
| 81 | uint8_t _pad1; | ||
| 82 | blkif_vdev_t _pad2; | ||
| 83 | uint64_t id; /* private guest value, echoed in resp */ | ||
| 84 | } __attribute__((__packed__)); | ||
| 85 | |||
| 80 | struct blkif_x86_32_request { | 86 | struct blkif_x86_32_request { |
| 81 | uint8_t operation; /* BLKIF_OP_??? */ | 87 | uint8_t operation; /* BLKIF_OP_??? */ |
| 82 | union { | 88 | union { |
| 83 | struct blkif_x86_32_request_rw rw; | 89 | struct blkif_x86_32_request_rw rw; |
| 84 | struct blkif_x86_32_request_discard discard; | 90 | struct blkif_x86_32_request_discard discard; |
| 91 | struct blkif_x86_32_request_other other; | ||
| 85 | } u; | 92 | } u; |
| 86 | } __attribute__((__packed__)); | 93 | } __attribute__((__packed__)); |
| 87 | 94 | ||
| @@ -113,11 +120,19 @@ struct blkif_x86_64_request_discard { | |||
| 113 | uint64_t nr_sectors; | 120 | uint64_t nr_sectors; |
| 114 | } __attribute__((__packed__)); | 121 | } __attribute__((__packed__)); |
| 115 | 122 | ||
| 123 | struct blkif_x86_64_request_other { | ||
| 124 | uint8_t _pad1; | ||
| 125 | blkif_vdev_t _pad2; | ||
| 126 | uint32_t _pad3; /* offsetof(blkif_..,u.discard.id)==8 */ | ||
| 127 | uint64_t id; /* private guest value, echoed in resp */ | ||
| 128 | } __attribute__((__packed__)); | ||
| 129 | |||
| 116 | struct blkif_x86_64_request { | 130 | struct blkif_x86_64_request { |
| 117 | uint8_t operation; /* BLKIF_OP_??? */ | 131 | uint8_t operation; /* BLKIF_OP_??? */ |
| 118 | union { | 132 | union { |
| 119 | struct blkif_x86_64_request_rw rw; | 133 | struct blkif_x86_64_request_rw rw; |
| 120 | struct blkif_x86_64_request_discard discard; | 134 | struct blkif_x86_64_request_discard discard; |
| 135 | struct blkif_x86_64_request_other other; | ||
| 121 | } u; | 136 | } u; |
| 122 | } __attribute__((__packed__)); | 137 | } __attribute__((__packed__)); |
| 123 | 138 | ||
| @@ -172,7 +187,6 @@ struct persistent_gnt { | |||
| 172 | struct page *page; | 187 | struct page *page; |
| 173 | grant_ref_t gnt; | 188 | grant_ref_t gnt; |
| 174 | grant_handle_t handle; | 189 | grant_handle_t handle; |
| 175 | uint64_t dev_bus_addr; | ||
| 176 | struct rb_node node; | 190 | struct rb_node node; |
| 177 | }; | 191 | }; |
| 178 | 192 | ||
| @@ -208,13 +222,13 @@ struct xen_blkif { | |||
| 208 | 222 | ||
| 209 | /* statistics */ | 223 | /* statistics */ |
| 210 | unsigned long st_print; | 224 | unsigned long st_print; |
| 211 | int st_rd_req; | 225 | unsigned long long st_rd_req; |
| 212 | int st_wr_req; | 226 | unsigned long long st_wr_req; |
| 213 | int st_oo_req; | 227 | unsigned long long st_oo_req; |
| 214 | int st_f_req; | 228 | unsigned long long st_f_req; |
| 215 | int st_ds_req; | 229 | unsigned long long st_ds_req; |
| 216 | int st_rd_sect; | 230 | unsigned long long st_rd_sect; |
| 217 | int st_wr_sect; | 231 | unsigned long long st_wr_sect; |
| 218 | 232 | ||
| 219 | wait_queue_head_t waiting_to_free; | 233 | wait_queue_head_t waiting_to_free; |
| 220 | }; | 234 | }; |
| @@ -278,6 +292,11 @@ static inline void blkif_get_x86_32_req(struct blkif_request *dst, | |||
| 278 | dst->u.discard.nr_sectors = src->u.discard.nr_sectors; | 292 | dst->u.discard.nr_sectors = src->u.discard.nr_sectors; |
| 279 | break; | 293 | break; |
| 280 | default: | 294 | default: |
| 295 | /* | ||
| 296 | * Don't know how to translate this op. Only get the | ||
| 297 | * ID so failure can be reported to the frontend. | ||
| 298 | */ | ||
| 299 | dst->u.other.id = src->u.other.id; | ||
| 281 | break; | 300 | break; |
| 282 | } | 301 | } |
| 283 | } | 302 | } |
| @@ -309,6 +328,11 @@ static inline void blkif_get_x86_64_req(struct blkif_request *dst, | |||
| 309 | dst->u.discard.nr_sectors = src->u.discard.nr_sectors; | 328 | dst->u.discard.nr_sectors = src->u.discard.nr_sectors; |
| 310 | break; | 329 | break; |
| 311 | default: | 330 | default: |
| 331 | /* | ||
| 332 | * Don't know how to translate this op. Only get the | ||
| 333 | * ID so failure can be reported to the frontend. | ||
| 334 | */ | ||
| 335 | dst->u.other.id = src->u.other.id; | ||
| 312 | break; | 336 | break; |
| 313 | } | 337 | } |
| 314 | } | 338 | } |
diff --git a/drivers/block/xen-blkback/xenbus.c b/drivers/block/xen-blkback/xenbus.c index 5e237f630c47..8bfd1bcf95ec 100644 --- a/drivers/block/xen-blkback/xenbus.c +++ b/drivers/block/xen-blkback/xenbus.c | |||
| @@ -230,13 +230,13 @@ int __init xen_blkif_interface_init(void) | |||
| 230 | } \ | 230 | } \ |
| 231 | static DEVICE_ATTR(name, S_IRUGO, show_##name, NULL) | 231 | static DEVICE_ATTR(name, S_IRUGO, show_##name, NULL) |
| 232 | 232 | ||
| 233 | VBD_SHOW(oo_req, "%d\n", be->blkif->st_oo_req); | 233 | VBD_SHOW(oo_req, "%llu\n", be->blkif->st_oo_req); |
| 234 | VBD_SHOW(rd_req, "%d\n", be->blkif->st_rd_req); | 234 | VBD_SHOW(rd_req, "%llu\n", be->blkif->st_rd_req); |
| 235 | VBD_SHOW(wr_req, "%d\n", be->blkif->st_wr_req); | 235 | VBD_SHOW(wr_req, "%llu\n", be->blkif->st_wr_req); |
| 236 | VBD_SHOW(f_req, "%d\n", be->blkif->st_f_req); | 236 | VBD_SHOW(f_req, "%llu\n", be->blkif->st_f_req); |
| 237 | VBD_SHOW(ds_req, "%d\n", be->blkif->st_ds_req); | 237 | VBD_SHOW(ds_req, "%llu\n", be->blkif->st_ds_req); |
| 238 | VBD_SHOW(rd_sect, "%d\n", be->blkif->st_rd_sect); | 238 | VBD_SHOW(rd_sect, "%llu\n", be->blkif->st_rd_sect); |
| 239 | VBD_SHOW(wr_sect, "%d\n", be->blkif->st_wr_sect); | 239 | VBD_SHOW(wr_sect, "%llu\n", be->blkif->st_wr_sect); |
| 240 | 240 | ||
| 241 | static struct attribute *xen_vbdstat_attrs[] = { | 241 | static struct attribute *xen_vbdstat_attrs[] = { |
| 242 | &dev_attr_oo_req.attr, | 242 | &dev_attr_oo_req.attr, |
diff --git a/drivers/block/xen-blkfront.c b/drivers/block/xen-blkfront.c index c3dae2e0f290..a894f88762d8 100644 --- a/drivers/block/xen-blkfront.c +++ b/drivers/block/xen-blkfront.c | |||
| @@ -44,7 +44,7 @@ | |||
| 44 | #include <linux/mutex.h> | 44 | #include <linux/mutex.h> |
| 45 | #include <linux/scatterlist.h> | 45 | #include <linux/scatterlist.h> |
| 46 | #include <linux/bitmap.h> | 46 | #include <linux/bitmap.h> |
| 47 | #include <linux/llist.h> | 47 | #include <linux/list.h> |
| 48 | 48 | ||
| 49 | #include <xen/xen.h> | 49 | #include <xen/xen.h> |
| 50 | #include <xen/xenbus.h> | 50 | #include <xen/xenbus.h> |
| @@ -68,13 +68,12 @@ enum blkif_state { | |||
| 68 | struct grant { | 68 | struct grant { |
| 69 | grant_ref_t gref; | 69 | grant_ref_t gref; |
| 70 | unsigned long pfn; | 70 | unsigned long pfn; |
| 71 | struct llist_node node; | 71 | struct list_head node; |
| 72 | }; | 72 | }; |
| 73 | 73 | ||
| 74 | struct blk_shadow { | 74 | struct blk_shadow { |
| 75 | struct blkif_request req; | 75 | struct blkif_request req; |
| 76 | struct request *request; | 76 | struct request *request; |
| 77 | unsigned long frame[BLKIF_MAX_SEGMENTS_PER_REQUEST]; | ||
| 78 | struct grant *grants_used[BLKIF_MAX_SEGMENTS_PER_REQUEST]; | 77 | struct grant *grants_used[BLKIF_MAX_SEGMENTS_PER_REQUEST]; |
| 79 | }; | 78 | }; |
| 80 | 79 | ||
| @@ -105,7 +104,7 @@ struct blkfront_info | |||
| 105 | struct work_struct work; | 104 | struct work_struct work; |
| 106 | struct gnttab_free_callback callback; | 105 | struct gnttab_free_callback callback; |
| 107 | struct blk_shadow shadow[BLK_RING_SIZE]; | 106 | struct blk_shadow shadow[BLK_RING_SIZE]; |
| 108 | struct llist_head persistent_gnts; | 107 | struct list_head persistent_gnts; |
| 109 | unsigned int persistent_gnts_c; | 108 | unsigned int persistent_gnts_c; |
| 110 | unsigned long shadow_free; | 109 | unsigned long shadow_free; |
| 111 | unsigned int feature_flush; | 110 | unsigned int feature_flush; |
| @@ -165,6 +164,69 @@ static int add_id_to_freelist(struct blkfront_info *info, | |||
| 165 | return 0; | 164 | return 0; |
| 166 | } | 165 | } |
| 167 | 166 | ||
| 167 | static int fill_grant_buffer(struct blkfront_info *info, int num) | ||
| 168 | { | ||
| 169 | struct page *granted_page; | ||
| 170 | struct grant *gnt_list_entry, *n; | ||
| 171 | int i = 0; | ||
| 172 | |||
| 173 | while(i < num) { | ||
| 174 | gnt_list_entry = kzalloc(sizeof(struct grant), GFP_NOIO); | ||
| 175 | if (!gnt_list_entry) | ||
| 176 | goto out_of_memory; | ||
| 177 | |||
| 178 | granted_page = alloc_page(GFP_NOIO); | ||
| 179 | if (!granted_page) { | ||
| 180 | kfree(gnt_list_entry); | ||
| 181 | goto out_of_memory; | ||
| 182 | } | ||
| 183 | |||
| 184 | gnt_list_entry->pfn = page_to_pfn(granted_page); | ||
| 185 | gnt_list_entry->gref = GRANT_INVALID_REF; | ||
| 186 | list_add(&gnt_list_entry->node, &info->persistent_gnts); | ||
| 187 | i++; | ||
| 188 | } | ||
| 189 | |||
| 190 | return 0; | ||
| 191 | |||
| 192 | out_of_memory: | ||
| 193 | list_for_each_entry_safe(gnt_list_entry, n, | ||
| 194 | &info->persistent_gnts, node) { | ||
| 195 | list_del(&gnt_list_entry->node); | ||
| 196 | __free_page(pfn_to_page(gnt_list_entry->pfn)); | ||
| 197 | kfree(gnt_list_entry); | ||
| 198 | i--; | ||
| 199 | } | ||
| 200 | BUG_ON(i != 0); | ||
| 201 | return -ENOMEM; | ||
| 202 | } | ||
| 203 | |||
| 204 | static struct grant *get_grant(grant_ref_t *gref_head, | ||
| 205 | struct blkfront_info *info) | ||
| 206 | { | ||
| 207 | struct grant *gnt_list_entry; | ||
| 208 | unsigned long buffer_mfn; | ||
| 209 | |||
| 210 | BUG_ON(list_empty(&info->persistent_gnts)); | ||
| 211 | gnt_list_entry = list_first_entry(&info->persistent_gnts, struct grant, | ||
| 212 | node); | ||
| 213 | list_del(&gnt_list_entry->node); | ||
| 214 | |||
| 215 | if (gnt_list_entry->gref != GRANT_INVALID_REF) { | ||
| 216 | info->persistent_gnts_c--; | ||
| 217 | return gnt_list_entry; | ||
| 218 | } | ||
| 219 | |||
| 220 | /* Assign a gref to this page */ | ||
| 221 | gnt_list_entry->gref = gnttab_claim_grant_reference(gref_head); | ||
| 222 | BUG_ON(gnt_list_entry->gref == -ENOSPC); | ||
| 223 | buffer_mfn = pfn_to_mfn(gnt_list_entry->pfn); | ||
| 224 | gnttab_grant_foreign_access_ref(gnt_list_entry->gref, | ||
| 225 | info->xbdev->otherend_id, | ||
| 226 | buffer_mfn, 0); | ||
| 227 | return gnt_list_entry; | ||
| 228 | } | ||
| 229 | |||
| 168 | static const char *op_name(int op) | 230 | static const char *op_name(int op) |
| 169 | { | 231 | { |
| 170 | static const char *const names[] = { | 232 | static const char *const names[] = { |
| @@ -293,7 +355,6 @@ static int blkif_ioctl(struct block_device *bdev, fmode_t mode, | |||
| 293 | static int blkif_queue_request(struct request *req) | 355 | static int blkif_queue_request(struct request *req) |
| 294 | { | 356 | { |
| 295 | struct blkfront_info *info = req->rq_disk->private_data; | 357 | struct blkfront_info *info = req->rq_disk->private_data; |
| 296 | unsigned long buffer_mfn; | ||
| 297 | struct blkif_request *ring_req; | 358 | struct blkif_request *ring_req; |
| 298 | unsigned long id; | 359 | unsigned long id; |
| 299 | unsigned int fsect, lsect; | 360 | unsigned int fsect, lsect; |
| @@ -306,7 +367,6 @@ static int blkif_queue_request(struct request *req) | |||
| 306 | */ | 367 | */ |
| 307 | bool new_persistent_gnts; | 368 | bool new_persistent_gnts; |
| 308 | grant_ref_t gref_head; | 369 | grant_ref_t gref_head; |
| 309 | struct page *granted_page; | ||
| 310 | struct grant *gnt_list_entry = NULL; | 370 | struct grant *gnt_list_entry = NULL; |
| 311 | struct scatterlist *sg; | 371 | struct scatterlist *sg; |
| 312 | 372 | ||
| @@ -370,41 +430,8 @@ static int blkif_queue_request(struct request *req) | |||
| 370 | fsect = sg->offset >> 9; | 430 | fsect = sg->offset >> 9; |
| 371 | lsect = fsect + (sg->length >> 9) - 1; | 431 | lsect = fsect + (sg->length >> 9) - 1; |
| 372 | 432 | ||
| 373 | if (info->persistent_gnts_c) { | 433 | gnt_list_entry = get_grant(&gref_head, info); |
| 374 | BUG_ON(llist_empty(&info->persistent_gnts)); | 434 | ref = gnt_list_entry->gref; |
| 375 | gnt_list_entry = llist_entry( | ||
| 376 | llist_del_first(&info->persistent_gnts), | ||
| 377 | struct grant, node); | ||
| 378 | |||
| 379 | ref = gnt_list_entry->gref; | ||
| 380 | buffer_mfn = pfn_to_mfn(gnt_list_entry->pfn); | ||
| 381 | info->persistent_gnts_c--; | ||
| 382 | } else { | ||
| 383 | ref = gnttab_claim_grant_reference(&gref_head); | ||
| 384 | BUG_ON(ref == -ENOSPC); | ||
| 385 | |||
| 386 | gnt_list_entry = | ||
| 387 | kmalloc(sizeof(struct grant), | ||
| 388 | GFP_ATOMIC); | ||
| 389 | if (!gnt_list_entry) | ||
| 390 | return -ENOMEM; | ||
| 391 | |||
| 392 | granted_page = alloc_page(GFP_ATOMIC); | ||
| 393 | if (!granted_page) { | ||
| 394 | kfree(gnt_list_entry); | ||
| 395 | return -ENOMEM; | ||
| 396 | } | ||
| 397 | |||
| 398 | gnt_list_entry->pfn = | ||
| 399 | page_to_pfn(granted_page); | ||
| 400 | gnt_list_entry->gref = ref; | ||
| 401 | |||
| 402 | buffer_mfn = pfn_to_mfn(page_to_pfn( | ||
| 403 | granted_page)); | ||
| 404 | gnttab_grant_foreign_access_ref(ref, | ||
| 405 | info->xbdev->otherend_id, | ||
| 406 | buffer_mfn, 0); | ||
| 407 | } | ||
| 408 | 435 | ||
| 409 | info->shadow[id].grants_used[i] = gnt_list_entry; | 436 | info->shadow[id].grants_used[i] = gnt_list_entry; |
| 410 | 437 | ||
| @@ -435,7 +462,6 @@ static int blkif_queue_request(struct request *req) | |||
| 435 | kunmap_atomic(shared_data); | 462 | kunmap_atomic(shared_data); |
| 436 | } | 463 | } |
| 437 | 464 | ||
| 438 | info->shadow[id].frame[i] = mfn_to_pfn(buffer_mfn); | ||
| 439 | ring_req->u.rw.seg[i] = | 465 | ring_req->u.rw.seg[i] = |
| 440 | (struct blkif_request_segment) { | 466 | (struct blkif_request_segment) { |
| 441 | .gref = ref, | 467 | .gref = ref, |
| @@ -790,9 +816,8 @@ static void blkif_restart_queue(struct work_struct *work) | |||
| 790 | 816 | ||
| 791 | static void blkif_free(struct blkfront_info *info, int suspend) | 817 | static void blkif_free(struct blkfront_info *info, int suspend) |
| 792 | { | 818 | { |
| 793 | struct llist_node *all_gnts; | 819 | struct grant *persistent_gnt; |
| 794 | struct grant *persistent_gnt, *tmp; | 820 | struct grant *n; |
| 795 | struct llist_node *n; | ||
| 796 | 821 | ||
| 797 | /* Prevent new requests being issued until we fix things up. */ | 822 | /* Prevent new requests being issued until we fix things up. */ |
| 798 | spin_lock_irq(&info->io_lock); | 823 | spin_lock_irq(&info->io_lock); |
| @@ -803,22 +828,20 @@ static void blkif_free(struct blkfront_info *info, int suspend) | |||
| 803 | blk_stop_queue(info->rq); | 828 | blk_stop_queue(info->rq); |
| 804 | 829 | ||
| 805 | /* Remove all persistent grants */ | 830 | /* Remove all persistent grants */ |
| 806 | if (info->persistent_gnts_c) { | 831 | if (!list_empty(&info->persistent_gnts)) { |
| 807 | all_gnts = llist_del_all(&info->persistent_gnts); | 832 | list_for_each_entry_safe(persistent_gnt, n, |
| 808 | persistent_gnt = llist_entry(all_gnts, typeof(*(persistent_gnt)), node); | 833 | &info->persistent_gnts, node) { |
| 809 | while (persistent_gnt) { | 834 | list_del(&persistent_gnt->node); |
| 810 | gnttab_end_foreign_access(persistent_gnt->gref, 0, 0UL); | 835 | if (persistent_gnt->gref != GRANT_INVALID_REF) { |
| 836 | gnttab_end_foreign_access(persistent_gnt->gref, | ||
| 837 | 0, 0UL); | ||
| 838 | info->persistent_gnts_c--; | ||
| 839 | } | ||
| 811 | __free_page(pfn_to_page(persistent_gnt->pfn)); | 840 | __free_page(pfn_to_page(persistent_gnt->pfn)); |
| 812 | tmp = persistent_gnt; | 841 | kfree(persistent_gnt); |
| 813 | n = persistent_gnt->node.next; | ||
| 814 | if (n) | ||
| 815 | persistent_gnt = llist_entry(n, typeof(*(persistent_gnt)), node); | ||
| 816 | else | ||
| 817 | persistent_gnt = NULL; | ||
| 818 | kfree(tmp); | ||
| 819 | } | 842 | } |
| 820 | info->persistent_gnts_c = 0; | ||
| 821 | } | 843 | } |
| 844 | BUG_ON(info->persistent_gnts_c != 0); | ||
| 822 | 845 | ||
| 823 | /* No more gnttab callback work. */ | 846 | /* No more gnttab callback work. */ |
| 824 | gnttab_cancel_free_callback(&info->callback); | 847 | gnttab_cancel_free_callback(&info->callback); |
| @@ -875,7 +898,7 @@ static void blkif_completion(struct blk_shadow *s, struct blkfront_info *info, | |||
| 875 | } | 898 | } |
| 876 | /* Add the persistent grant into the list of free grants */ | 899 | /* Add the persistent grant into the list of free grants */ |
| 877 | for (i = 0; i < s->req.u.rw.nr_segments; i++) { | 900 | for (i = 0; i < s->req.u.rw.nr_segments; i++) { |
| 878 | llist_add(&s->grants_used[i]->node, &info->persistent_gnts); | 901 | list_add(&s->grants_used[i]->node, &info->persistent_gnts); |
| 879 | info->persistent_gnts_c++; | 902 | info->persistent_gnts_c++; |
| 880 | } | 903 | } |
| 881 | } | 904 | } |
| @@ -1013,6 +1036,12 @@ static int setup_blkring(struct xenbus_device *dev, | |||
| 1013 | 1036 | ||
| 1014 | sg_init_table(info->sg, BLKIF_MAX_SEGMENTS_PER_REQUEST); | 1037 | sg_init_table(info->sg, BLKIF_MAX_SEGMENTS_PER_REQUEST); |
| 1015 | 1038 | ||
| 1039 | /* Allocate memory for grants */ | ||
| 1040 | err = fill_grant_buffer(info, BLK_RING_SIZE * | ||
| 1041 | BLKIF_MAX_SEGMENTS_PER_REQUEST); | ||
| 1042 | if (err) | ||
| 1043 | goto fail; | ||
| 1044 | |||
| 1016 | err = xenbus_grant_ring(dev, virt_to_mfn(info->ring.sring)); | 1045 | err = xenbus_grant_ring(dev, virt_to_mfn(info->ring.sring)); |
| 1017 | if (err < 0) { | 1046 | if (err < 0) { |
| 1018 | free_page((unsigned long)sring); | 1047 | free_page((unsigned long)sring); |
| @@ -1171,7 +1200,7 @@ static int blkfront_probe(struct xenbus_device *dev, | |||
| 1171 | spin_lock_init(&info->io_lock); | 1200 | spin_lock_init(&info->io_lock); |
| 1172 | info->xbdev = dev; | 1201 | info->xbdev = dev; |
| 1173 | info->vdevice = vdevice; | 1202 | info->vdevice = vdevice; |
| 1174 | init_llist_head(&info->persistent_gnts); | 1203 | INIT_LIST_HEAD(&info->persistent_gnts); |
| 1175 | info->persistent_gnts_c = 0; | 1204 | info->persistent_gnts_c = 0; |
| 1176 | info->connected = BLKIF_STATE_DISCONNECTED; | 1205 | info->connected = BLKIF_STATE_DISCONNECTED; |
| 1177 | INIT_WORK(&info->work, blkif_restart_queue); | 1206 | INIT_WORK(&info->work, blkif_restart_queue); |
| @@ -1203,11 +1232,10 @@ static int blkif_recover(struct blkfront_info *info) | |||
| 1203 | int j; | 1232 | int j; |
| 1204 | 1233 | ||
| 1205 | /* Stage 1: Make a safe copy of the shadow state. */ | 1234 | /* Stage 1: Make a safe copy of the shadow state. */ |
| 1206 | copy = kmalloc(sizeof(info->shadow), | 1235 | copy = kmemdup(info->shadow, sizeof(info->shadow), |
| 1207 | GFP_NOIO | __GFP_REPEAT | __GFP_HIGH); | 1236 | GFP_NOIO | __GFP_REPEAT | __GFP_HIGH); |
| 1208 | if (!copy) | 1237 | if (!copy) |
| 1209 | return -ENOMEM; | 1238 | return -ENOMEM; |
| 1210 | memcpy(copy, info->shadow, sizeof(info->shadow)); | ||
| 1211 | 1239 | ||
| 1212 | /* Stage 2: Set up free list. */ | 1240 | /* Stage 2: Set up free list. */ |
| 1213 | memset(&info->shadow, 0, sizeof(info->shadow)); | 1241 | memset(&info->shadow, 0, sizeof(info->shadow)); |
| @@ -1236,7 +1264,7 @@ static int blkif_recover(struct blkfront_info *info) | |||
| 1236 | gnttab_grant_foreign_access_ref( | 1264 | gnttab_grant_foreign_access_ref( |
| 1237 | req->u.rw.seg[j].gref, | 1265 | req->u.rw.seg[j].gref, |
| 1238 | info->xbdev->otherend_id, | 1266 | info->xbdev->otherend_id, |
| 1239 | pfn_to_mfn(info->shadow[req->u.rw.id].frame[j]), | 1267 | pfn_to_mfn(copy[i].grants_used[j]->pfn), |
| 1240 | 0); | 1268 | 0); |
| 1241 | } | 1269 | } |
| 1242 | info->shadow[req->u.rw.id].req = *req; | 1270 | info->shadow[req->u.rw.id].req = *req; |
diff --git a/drivers/bluetooth/ath3k.c b/drivers/bluetooth/ath3k.c index b282af181b44..6aab00ef4379 100644 --- a/drivers/bluetooth/ath3k.c +++ b/drivers/bluetooth/ath3k.c | |||
| @@ -73,9 +73,11 @@ static struct usb_device_id ath3k_table[] = { | |||
| 73 | { USB_DEVICE(0x03F0, 0x311D) }, | 73 | { USB_DEVICE(0x03F0, 0x311D) }, |
| 74 | 74 | ||
| 75 | /* Atheros AR3012 with sflash firmware*/ | 75 | /* Atheros AR3012 with sflash firmware*/ |
| 76 | { USB_DEVICE(0x0CF3, 0x0036) }, | ||
| 76 | { USB_DEVICE(0x0CF3, 0x3004) }, | 77 | { USB_DEVICE(0x0CF3, 0x3004) }, |
| 77 | { USB_DEVICE(0x0CF3, 0x3008) }, | 78 | { USB_DEVICE(0x0CF3, 0x3008) }, |
| 78 | { USB_DEVICE(0x0CF3, 0x311D) }, | 79 | { USB_DEVICE(0x0CF3, 0x311D) }, |
| 80 | { USB_DEVICE(0x0CF3, 0x817a) }, | ||
| 79 | { USB_DEVICE(0x13d3, 0x3375) }, | 81 | { USB_DEVICE(0x13d3, 0x3375) }, |
| 80 | { USB_DEVICE(0x04CA, 0x3004) }, | 82 | { USB_DEVICE(0x04CA, 0x3004) }, |
| 81 | { USB_DEVICE(0x04CA, 0x3005) }, | 83 | { USB_DEVICE(0x04CA, 0x3005) }, |
| @@ -107,9 +109,11 @@ MODULE_DEVICE_TABLE(usb, ath3k_table); | |||
| 107 | static struct usb_device_id ath3k_blist_tbl[] = { | 109 | static struct usb_device_id ath3k_blist_tbl[] = { |
| 108 | 110 | ||
| 109 | /* Atheros AR3012 with sflash firmware*/ | 111 | /* Atheros AR3012 with sflash firmware*/ |
| 112 | { USB_DEVICE(0x0CF3, 0x0036), .driver_info = BTUSB_ATH3012 }, | ||
| 110 | { USB_DEVICE(0x0cf3, 0x3004), .driver_info = BTUSB_ATH3012 }, | 113 | { USB_DEVICE(0x0cf3, 0x3004), .driver_info = BTUSB_ATH3012 }, |
| 111 | { USB_DEVICE(0x0cf3, 0x3008), .driver_info = BTUSB_ATH3012 }, | 114 | { USB_DEVICE(0x0cf3, 0x3008), .driver_info = BTUSB_ATH3012 }, |
| 112 | { USB_DEVICE(0x0cf3, 0x311D), .driver_info = BTUSB_ATH3012 }, | 115 | { USB_DEVICE(0x0cf3, 0x311D), .driver_info = BTUSB_ATH3012 }, |
| 116 | { USB_DEVICE(0x0CF3, 0x817a), .driver_info = BTUSB_ATH3012 }, | ||
| 113 | { USB_DEVICE(0x13d3, 0x3375), .driver_info = BTUSB_ATH3012 }, | 117 | { USB_DEVICE(0x13d3, 0x3375), .driver_info = BTUSB_ATH3012 }, |
| 114 | { USB_DEVICE(0x04ca, 0x3004), .driver_info = BTUSB_ATH3012 }, | 118 | { USB_DEVICE(0x04ca, 0x3004), .driver_info = BTUSB_ATH3012 }, |
| 115 | { USB_DEVICE(0x04ca, 0x3005), .driver_info = BTUSB_ATH3012 }, | 119 | { USB_DEVICE(0x04ca, 0x3005), .driver_info = BTUSB_ATH3012 }, |
diff --git a/drivers/bluetooth/btusb.c b/drivers/bluetooth/btusb.c index e547851870e7..2cc5f774a29c 100644 --- a/drivers/bluetooth/btusb.c +++ b/drivers/bluetooth/btusb.c | |||
| @@ -131,9 +131,11 @@ static struct usb_device_id blacklist_table[] = { | |||
| 131 | { USB_DEVICE(0x03f0, 0x311d), .driver_info = BTUSB_IGNORE }, | 131 | { USB_DEVICE(0x03f0, 0x311d), .driver_info = BTUSB_IGNORE }, |
| 132 | 132 | ||
| 133 | /* Atheros 3012 with sflash firmware */ | 133 | /* Atheros 3012 with sflash firmware */ |
| 134 | { USB_DEVICE(0x0cf3, 0x0036), .driver_info = BTUSB_ATH3012 }, | ||
| 134 | { USB_DEVICE(0x0cf3, 0x3004), .driver_info = BTUSB_ATH3012 }, | 135 | { USB_DEVICE(0x0cf3, 0x3004), .driver_info = BTUSB_ATH3012 }, |
| 135 | { USB_DEVICE(0x0cf3, 0x3008), .driver_info = BTUSB_ATH3012 }, | 136 | { USB_DEVICE(0x0cf3, 0x3008), .driver_info = BTUSB_ATH3012 }, |
| 136 | { USB_DEVICE(0x0cf3, 0x311d), .driver_info = BTUSB_ATH3012 }, | 137 | { USB_DEVICE(0x0cf3, 0x311d), .driver_info = BTUSB_ATH3012 }, |
| 138 | { USB_DEVICE(0x0cf3, 0x817a), .driver_info = BTUSB_ATH3012 }, | ||
| 137 | { USB_DEVICE(0x13d3, 0x3375), .driver_info = BTUSB_ATH3012 }, | 139 | { USB_DEVICE(0x13d3, 0x3375), .driver_info = BTUSB_ATH3012 }, |
| 138 | { USB_DEVICE(0x04ca, 0x3004), .driver_info = BTUSB_ATH3012 }, | 140 | { USB_DEVICE(0x04ca, 0x3004), .driver_info = BTUSB_ATH3012 }, |
| 139 | { USB_DEVICE(0x04ca, 0x3005), .driver_info = BTUSB_ATH3012 }, | 141 | { USB_DEVICE(0x04ca, 0x3005), .driver_info = BTUSB_ATH3012 }, |
diff --git a/drivers/cpufreq/acpi-cpufreq.c b/drivers/cpufreq/acpi-cpufreq.c index 937bc286591f..57a8774f0b4e 100644 --- a/drivers/cpufreq/acpi-cpufreq.c +++ b/drivers/cpufreq/acpi-cpufreq.c | |||
| @@ -730,7 +730,6 @@ static int acpi_cpufreq_cpu_init(struct cpufreq_policy *policy) | |||
| 730 | policy->shared_type == CPUFREQ_SHARED_TYPE_ANY) { | 730 | policy->shared_type == CPUFREQ_SHARED_TYPE_ANY) { |
| 731 | cpumask_copy(policy->cpus, perf->shared_cpu_map); | 731 | cpumask_copy(policy->cpus, perf->shared_cpu_map); |
| 732 | } | 732 | } |
| 733 | cpumask_copy(policy->related_cpus, perf->shared_cpu_map); | ||
| 734 | 733 | ||
| 735 | #ifdef CONFIG_SMP | 734 | #ifdef CONFIG_SMP |
| 736 | dmi_check_system(sw_any_bug_dmi_table); | 735 | dmi_check_system(sw_any_bug_dmi_table); |
| @@ -742,7 +741,6 @@ static int acpi_cpufreq_cpu_init(struct cpufreq_policy *policy) | |||
| 742 | if (check_amd_hwpstate_cpu(cpu) && !acpi_pstate_strict) { | 741 | if (check_amd_hwpstate_cpu(cpu) && !acpi_pstate_strict) { |
| 743 | cpumask_clear(policy->cpus); | 742 | cpumask_clear(policy->cpus); |
| 744 | cpumask_set_cpu(cpu, policy->cpus); | 743 | cpumask_set_cpu(cpu, policy->cpus); |
| 745 | cpumask_copy(policy->related_cpus, cpu_sibling_mask(cpu)); | ||
| 746 | policy->shared_type = CPUFREQ_SHARED_TYPE_HW; | 744 | policy->shared_type = CPUFREQ_SHARED_TYPE_HW; |
| 747 | pr_info_once(PFX "overriding BIOS provided _PSD data\n"); | 745 | pr_info_once(PFX "overriding BIOS provided _PSD data\n"); |
| 748 | } | 746 | } |
diff --git a/drivers/cpufreq/cpufreq_stats.c b/drivers/cpufreq/cpufreq_stats.c index 2fd779eb1ed1..bfd6273fd873 100644 --- a/drivers/cpufreq/cpufreq_stats.c +++ b/drivers/cpufreq/cpufreq_stats.c | |||
| @@ -180,15 +180,19 @@ static void cpufreq_stats_free_sysfs(unsigned int cpu) | |||
| 180 | { | 180 | { |
| 181 | struct cpufreq_policy *policy = cpufreq_cpu_get(cpu); | 181 | struct cpufreq_policy *policy = cpufreq_cpu_get(cpu); |
| 182 | 182 | ||
| 183 | if (!cpufreq_frequency_get_table(cpu)) | 183 | if (!policy) |
| 184 | return; | 184 | return; |
| 185 | 185 | ||
| 186 | if (policy && !policy_is_shared(policy)) { | 186 | if (!cpufreq_frequency_get_table(cpu)) |
| 187 | goto put_ref; | ||
| 188 | |||
| 189 | if (!policy_is_shared(policy)) { | ||
| 187 | pr_debug("%s: Free sysfs stat\n", __func__); | 190 | pr_debug("%s: Free sysfs stat\n", __func__); |
| 188 | sysfs_remove_group(&policy->kobj, &stats_attr_group); | 191 | sysfs_remove_group(&policy->kobj, &stats_attr_group); |
| 189 | } | 192 | } |
| 190 | if (policy) | 193 | |
| 191 | cpufreq_cpu_put(policy); | 194 | put_ref: |
| 195 | cpufreq_cpu_put(policy); | ||
| 192 | } | 196 | } |
| 193 | 197 | ||
| 194 | static int cpufreq_stats_create_table(struct cpufreq_policy *policy, | 198 | static int cpufreq_stats_create_table(struct cpufreq_policy *policy, |
diff --git a/drivers/cpufreq/intel_pstate.c b/drivers/cpufreq/intel_pstate.c index f6dd1e761129..ad72922919ed 100644 --- a/drivers/cpufreq/intel_pstate.c +++ b/drivers/cpufreq/intel_pstate.c | |||
| @@ -358,14 +358,14 @@ static void intel_pstate_sysfs_expose_params(void) | |||
| 358 | static int intel_pstate_min_pstate(void) | 358 | static int intel_pstate_min_pstate(void) |
| 359 | { | 359 | { |
| 360 | u64 value; | 360 | u64 value; |
| 361 | rdmsrl(0xCE, value); | 361 | rdmsrl(MSR_PLATFORM_INFO, value); |
| 362 | return (value >> 40) & 0xFF; | 362 | return (value >> 40) & 0xFF; |
| 363 | } | 363 | } |
| 364 | 364 | ||
| 365 | static int intel_pstate_max_pstate(void) | 365 | static int intel_pstate_max_pstate(void) |
| 366 | { | 366 | { |
| 367 | u64 value; | 367 | u64 value; |
| 368 | rdmsrl(0xCE, value); | 368 | rdmsrl(MSR_PLATFORM_INFO, value); |
| 369 | return (value >> 8) & 0xFF; | 369 | return (value >> 8) & 0xFF; |
| 370 | } | 370 | } |
| 371 | 371 | ||
| @@ -373,7 +373,7 @@ static int intel_pstate_turbo_pstate(void) | |||
| 373 | { | 373 | { |
| 374 | u64 value; | 374 | u64 value; |
| 375 | int nont, ret; | 375 | int nont, ret; |
| 376 | rdmsrl(0x1AD, value); | 376 | rdmsrl(MSR_NHM_TURBO_RATIO_LIMIT, value); |
| 377 | nont = intel_pstate_max_pstate(); | 377 | nont = intel_pstate_max_pstate(); |
| 378 | ret = ((value) & 255); | 378 | ret = ((value) & 255); |
| 379 | if (ret <= nont) | 379 | if (ret <= nont) |
| @@ -454,7 +454,7 @@ static inline void intel_pstate_calc_busy(struct cpudata *cpu, | |||
| 454 | sample->idletime_us * 100, | 454 | sample->idletime_us * 100, |
| 455 | sample->duration_us); | 455 | sample->duration_us); |
| 456 | core_pct = div64_u64(sample->aperf * 100, sample->mperf); | 456 | core_pct = div64_u64(sample->aperf * 100, sample->mperf); |
| 457 | sample->freq = cpu->pstate.turbo_pstate * core_pct * 1000; | 457 | sample->freq = cpu->pstate.max_pstate * core_pct * 1000; |
| 458 | 458 | ||
| 459 | sample->core_pct_busy = div_s64((sample->pstate_pct_busy * core_pct), | 459 | sample->core_pct_busy = div_s64((sample->pstate_pct_busy * core_pct), |
| 460 | 100); | 460 | 100); |
| @@ -752,6 +752,29 @@ static struct cpufreq_driver intel_pstate_driver = { | |||
| 752 | 752 | ||
| 753 | static int __initdata no_load; | 753 | static int __initdata no_load; |
| 754 | 754 | ||
| 755 | static int intel_pstate_msrs_not_valid(void) | ||
| 756 | { | ||
| 757 | /* Check that all the msr's we are using are valid. */ | ||
| 758 | u64 aperf, mperf, tmp; | ||
| 759 | |||
| 760 | rdmsrl(MSR_IA32_APERF, aperf); | ||
| 761 | rdmsrl(MSR_IA32_MPERF, mperf); | ||
| 762 | |||
| 763 | if (!intel_pstate_min_pstate() || | ||
| 764 | !intel_pstate_max_pstate() || | ||
| 765 | !intel_pstate_turbo_pstate()) | ||
| 766 | return -ENODEV; | ||
| 767 | |||
| 768 | rdmsrl(MSR_IA32_APERF, tmp); | ||
| 769 | if (!(tmp - aperf)) | ||
| 770 | return -ENODEV; | ||
| 771 | |||
| 772 | rdmsrl(MSR_IA32_MPERF, tmp); | ||
| 773 | if (!(tmp - mperf)) | ||
| 774 | return -ENODEV; | ||
| 775 | |||
| 776 | return 0; | ||
| 777 | } | ||
| 755 | static int __init intel_pstate_init(void) | 778 | static int __init intel_pstate_init(void) |
| 756 | { | 779 | { |
| 757 | int cpu, rc = 0; | 780 | int cpu, rc = 0; |
| @@ -764,6 +787,9 @@ static int __init intel_pstate_init(void) | |||
| 764 | if (!id) | 787 | if (!id) |
| 765 | return -ENODEV; | 788 | return -ENODEV; |
| 766 | 789 | ||
| 790 | if (intel_pstate_msrs_not_valid()) | ||
| 791 | return -ENODEV; | ||
| 792 | |||
| 767 | pr_info("Intel P-state driver initializing.\n"); | 793 | pr_info("Intel P-state driver initializing.\n"); |
| 768 | 794 | ||
| 769 | all_cpu_data = vmalloc(sizeof(void *) * num_possible_cpus()); | 795 | all_cpu_data = vmalloc(sizeof(void *) * num_possible_cpus()); |
diff --git a/drivers/crypto/caam/caamalg.c b/drivers/crypto/caam/caamalg.c index b2a0a0726a54..cf268b14ae9a 100644 --- a/drivers/crypto/caam/caamalg.c +++ b/drivers/crypto/caam/caamalg.c | |||
| @@ -1650,11 +1650,7 @@ struct caam_alg_template { | |||
| 1650 | }; | 1650 | }; |
| 1651 | 1651 | ||
| 1652 | static struct caam_alg_template driver_algs[] = { | 1652 | static struct caam_alg_template driver_algs[] = { |
| 1653 | /* | 1653 | /* single-pass ipsec_esp descriptor */ |
| 1654 | * single-pass ipsec_esp descriptor | ||
| 1655 | * authencesn(*,*) is also registered, although not present | ||
| 1656 | * explicitly here. | ||
| 1657 | */ | ||
| 1658 | { | 1654 | { |
| 1659 | .name = "authenc(hmac(md5),cbc(aes))", | 1655 | .name = "authenc(hmac(md5),cbc(aes))", |
| 1660 | .driver_name = "authenc-hmac-md5-cbc-aes-caam", | 1656 | .driver_name = "authenc-hmac-md5-cbc-aes-caam", |
| @@ -2217,9 +2213,7 @@ static int __init caam_algapi_init(void) | |||
| 2217 | for (i = 0; i < ARRAY_SIZE(driver_algs); i++) { | 2213 | for (i = 0; i < ARRAY_SIZE(driver_algs); i++) { |
| 2218 | /* TODO: check if h/w supports alg */ | 2214 | /* TODO: check if h/w supports alg */ |
| 2219 | struct caam_crypto_alg *t_alg; | 2215 | struct caam_crypto_alg *t_alg; |
| 2220 | bool done = false; | ||
| 2221 | 2216 | ||
| 2222 | authencesn: | ||
| 2223 | t_alg = caam_alg_alloc(ctrldev, &driver_algs[i]); | 2217 | t_alg = caam_alg_alloc(ctrldev, &driver_algs[i]); |
| 2224 | if (IS_ERR(t_alg)) { | 2218 | if (IS_ERR(t_alg)) { |
| 2225 | err = PTR_ERR(t_alg); | 2219 | err = PTR_ERR(t_alg); |
| @@ -2233,25 +2227,8 @@ authencesn: | |||
| 2233 | dev_warn(ctrldev, "%s alg registration failed\n", | 2227 | dev_warn(ctrldev, "%s alg registration failed\n", |
| 2234 | t_alg->crypto_alg.cra_driver_name); | 2228 | t_alg->crypto_alg.cra_driver_name); |
| 2235 | kfree(t_alg); | 2229 | kfree(t_alg); |
| 2236 | } else { | 2230 | } else |
| 2237 | list_add_tail(&t_alg->entry, &priv->alg_list); | 2231 | list_add_tail(&t_alg->entry, &priv->alg_list); |
| 2238 | if (driver_algs[i].type == CRYPTO_ALG_TYPE_AEAD && | ||
| 2239 | !memcmp(driver_algs[i].name, "authenc", 7) && | ||
| 2240 | !done) { | ||
| 2241 | char *name; | ||
| 2242 | |||
| 2243 | name = driver_algs[i].name; | ||
| 2244 | memmove(name + 10, name + 7, strlen(name) - 7); | ||
| 2245 | memcpy(name + 7, "esn", 3); | ||
| 2246 | |||
| 2247 | name = driver_algs[i].driver_name; | ||
| 2248 | memmove(name + 10, name + 7, strlen(name) - 7); | ||
| 2249 | memcpy(name + 7, "esn", 3); | ||
| 2250 | |||
| 2251 | done = true; | ||
| 2252 | goto authencesn; | ||
| 2253 | } | ||
| 2254 | } | ||
| 2255 | } | 2232 | } |
| 2256 | if (!list_empty(&priv->alg_list)) | 2233 | if (!list_empty(&priv->alg_list)) |
| 2257 | dev_info(ctrldev, "%s algorithms registered in /proc/crypto\n", | 2234 | dev_info(ctrldev, "%s algorithms registered in /proc/crypto\n", |
diff --git a/drivers/crypto/caam/compat.h b/drivers/crypto/caam/compat.h index cf15e7813801..762aeff626ac 100644 --- a/drivers/crypto/caam/compat.h +++ b/drivers/crypto/caam/compat.h | |||
| @@ -23,7 +23,6 @@ | |||
| 23 | #include <linux/types.h> | 23 | #include <linux/types.h> |
| 24 | #include <linux/debugfs.h> | 24 | #include <linux/debugfs.h> |
| 25 | #include <linux/circ_buf.h> | 25 | #include <linux/circ_buf.h> |
| 26 | #include <linux/string.h> | ||
| 27 | #include <net/xfrm.h> | 26 | #include <net/xfrm.h> |
| 28 | 27 | ||
| 29 | #include <crypto/algapi.h> | 28 | #include <crypto/algapi.h> |
diff --git a/drivers/crypto/talitos.c b/drivers/crypto/talitos.c index 09b184adf31b..5b2b5e61e4f9 100644 --- a/drivers/crypto/talitos.c +++ b/drivers/crypto/talitos.c | |||
| @@ -38,7 +38,6 @@ | |||
| 38 | #include <linux/spinlock.h> | 38 | #include <linux/spinlock.h> |
| 39 | #include <linux/rtnetlink.h> | 39 | #include <linux/rtnetlink.h> |
| 40 | #include <linux/slab.h> | 40 | #include <linux/slab.h> |
| 41 | #include <linux/string.h> | ||
| 42 | 41 | ||
| 43 | #include <crypto/algapi.h> | 42 | #include <crypto/algapi.h> |
| 44 | #include <crypto/aes.h> | 43 | #include <crypto/aes.h> |
| @@ -1974,11 +1973,7 @@ struct talitos_alg_template { | |||
| 1974 | }; | 1973 | }; |
| 1975 | 1974 | ||
| 1976 | static struct talitos_alg_template driver_algs[] = { | 1975 | static struct talitos_alg_template driver_algs[] = { |
| 1977 | /* | 1976 | /* AEAD algorithms. These use a single-pass ipsec_esp descriptor */ |
| 1978 | * AEAD algorithms. These use a single-pass ipsec_esp descriptor. | ||
| 1979 | * authencesn(*,*) is also registered, although not present | ||
| 1980 | * explicitly here. | ||
| 1981 | */ | ||
| 1982 | { .type = CRYPTO_ALG_TYPE_AEAD, | 1977 | { .type = CRYPTO_ALG_TYPE_AEAD, |
| 1983 | .alg.crypto = { | 1978 | .alg.crypto = { |
| 1984 | .cra_name = "authenc(hmac(sha1),cbc(aes))", | 1979 | .cra_name = "authenc(hmac(sha1),cbc(aes))", |
| @@ -2820,9 +2815,7 @@ static int talitos_probe(struct platform_device *ofdev) | |||
| 2820 | if (hw_supports(dev, driver_algs[i].desc_hdr_template)) { | 2815 | if (hw_supports(dev, driver_algs[i].desc_hdr_template)) { |
| 2821 | struct talitos_crypto_alg *t_alg; | 2816 | struct talitos_crypto_alg *t_alg; |
| 2822 | char *name = NULL; | 2817 | char *name = NULL; |
| 2823 | bool authenc = false; | ||
| 2824 | 2818 | ||
| 2825 | authencesn: | ||
| 2826 | t_alg = talitos_alg_alloc(dev, &driver_algs[i]); | 2819 | t_alg = talitos_alg_alloc(dev, &driver_algs[i]); |
| 2827 | if (IS_ERR(t_alg)) { | 2820 | if (IS_ERR(t_alg)) { |
| 2828 | err = PTR_ERR(t_alg); | 2821 | err = PTR_ERR(t_alg); |
| @@ -2837,8 +2830,6 @@ authencesn: | |||
| 2837 | err = crypto_register_alg( | 2830 | err = crypto_register_alg( |
| 2838 | &t_alg->algt.alg.crypto); | 2831 | &t_alg->algt.alg.crypto); |
| 2839 | name = t_alg->algt.alg.crypto.cra_driver_name; | 2832 | name = t_alg->algt.alg.crypto.cra_driver_name; |
| 2840 | authenc = authenc ? !authenc : | ||
| 2841 | !(bool)memcmp(name, "authenc", 7); | ||
| 2842 | break; | 2833 | break; |
| 2843 | case CRYPTO_ALG_TYPE_AHASH: | 2834 | case CRYPTO_ALG_TYPE_AHASH: |
| 2844 | err = crypto_register_ahash( | 2835 | err = crypto_register_ahash( |
| @@ -2851,25 +2842,8 @@ authencesn: | |||
| 2851 | dev_err(dev, "%s alg registration failed\n", | 2842 | dev_err(dev, "%s alg registration failed\n", |
| 2852 | name); | 2843 | name); |
| 2853 | kfree(t_alg); | 2844 | kfree(t_alg); |
| 2854 | } else { | 2845 | } else |
| 2855 | list_add_tail(&t_alg->entry, &priv->alg_list); | 2846 | list_add_tail(&t_alg->entry, &priv->alg_list); |
| 2856 | if (authenc) { | ||
| 2857 | struct crypto_alg *alg = | ||
| 2858 | &driver_algs[i].alg.crypto; | ||
| 2859 | |||
| 2860 | name = alg->cra_name; | ||
| 2861 | memmove(name + 10, name + 7, | ||
| 2862 | strlen(name) - 7); | ||
| 2863 | memcpy(name + 7, "esn", 3); | ||
| 2864 | |||
| 2865 | name = alg->cra_driver_name; | ||
| 2866 | memmove(name + 10, name + 7, | ||
| 2867 | strlen(name) - 7); | ||
| 2868 | memcpy(name + 7, "esn", 3); | ||
| 2869 | |||
| 2870 | goto authencesn; | ||
| 2871 | } | ||
| 2872 | } | ||
| 2873 | } | 2847 | } |
| 2874 | } | 2848 | } |
| 2875 | if (!list_empty(&priv->alg_list)) | 2849 | if (!list_empty(&priv->alg_list)) |
diff --git a/drivers/dma/dw_dmac.c b/drivers/dma/dw_dmac.c index c599558faeda..43a5329d4483 100644 --- a/drivers/dma/dw_dmac.c +++ b/drivers/dma/dw_dmac.c | |||
| @@ -1001,6 +1001,13 @@ static inline void convert_burst(u32 *maxburst) | |||
| 1001 | *maxburst = 0; | 1001 | *maxburst = 0; |
| 1002 | } | 1002 | } |
| 1003 | 1003 | ||
| 1004 | static inline void convert_slave_id(struct dw_dma_chan *dwc) | ||
| 1005 | { | ||
| 1006 | struct dw_dma *dw = to_dw_dma(dwc->chan.device); | ||
| 1007 | |||
| 1008 | dwc->dma_sconfig.slave_id -= dw->request_line_base; | ||
| 1009 | } | ||
| 1010 | |||
| 1004 | static int | 1011 | static int |
| 1005 | set_runtime_config(struct dma_chan *chan, struct dma_slave_config *sconfig) | 1012 | set_runtime_config(struct dma_chan *chan, struct dma_slave_config *sconfig) |
| 1006 | { | 1013 | { |
| @@ -1015,6 +1022,7 @@ set_runtime_config(struct dma_chan *chan, struct dma_slave_config *sconfig) | |||
| 1015 | 1022 | ||
| 1016 | convert_burst(&dwc->dma_sconfig.src_maxburst); | 1023 | convert_burst(&dwc->dma_sconfig.src_maxburst); |
| 1017 | convert_burst(&dwc->dma_sconfig.dst_maxburst); | 1024 | convert_burst(&dwc->dma_sconfig.dst_maxburst); |
| 1025 | convert_slave_id(dwc); | ||
| 1018 | 1026 | ||
| 1019 | return 0; | 1027 | return 0; |
| 1020 | } | 1028 | } |
| @@ -1276,9 +1284,9 @@ static struct dma_chan *dw_dma_xlate(struct of_phandle_args *dma_spec, | |||
| 1276 | if (dma_spec->args_count != 3) | 1284 | if (dma_spec->args_count != 3) |
| 1277 | return NULL; | 1285 | return NULL; |
| 1278 | 1286 | ||
| 1279 | fargs.req = be32_to_cpup(dma_spec->args+0); | 1287 | fargs.req = dma_spec->args[0]; |
| 1280 | fargs.src = be32_to_cpup(dma_spec->args+1); | 1288 | fargs.src = dma_spec->args[1]; |
| 1281 | fargs.dst = be32_to_cpup(dma_spec->args+2); | 1289 | fargs.dst = dma_spec->args[2]; |
| 1282 | 1290 | ||
| 1283 | if (WARN_ON(fargs.req >= DW_DMA_MAX_NR_REQUESTS || | 1291 | if (WARN_ON(fargs.req >= DW_DMA_MAX_NR_REQUESTS || |
| 1284 | fargs.src >= dw->nr_masters || | 1292 | fargs.src >= dw->nr_masters || |
| @@ -1628,6 +1636,7 @@ dw_dma_parse_dt(struct platform_device *pdev) | |||
| 1628 | 1636 | ||
| 1629 | static int dw_probe(struct platform_device *pdev) | 1637 | static int dw_probe(struct platform_device *pdev) |
| 1630 | { | 1638 | { |
| 1639 | const struct platform_device_id *match; | ||
| 1631 | struct dw_dma_platform_data *pdata; | 1640 | struct dw_dma_platform_data *pdata; |
| 1632 | struct resource *io; | 1641 | struct resource *io; |
| 1633 | struct dw_dma *dw; | 1642 | struct dw_dma *dw; |
| @@ -1711,6 +1720,11 @@ static int dw_probe(struct platform_device *pdev) | |||
| 1711 | memcpy(dw->data_width, pdata->data_width, 4); | 1720 | memcpy(dw->data_width, pdata->data_width, 4); |
| 1712 | } | 1721 | } |
| 1713 | 1722 | ||
| 1723 | /* Get the base request line if set */ | ||
| 1724 | match = platform_get_device_id(pdev); | ||
| 1725 | if (match) | ||
| 1726 | dw->request_line_base = (unsigned int)match->driver_data; | ||
| 1727 | |||
| 1714 | /* Calculate all channel mask before DMA setup */ | 1728 | /* Calculate all channel mask before DMA setup */ |
| 1715 | dw->all_chan_mask = (1 << nr_channels) - 1; | 1729 | dw->all_chan_mask = (1 << nr_channels) - 1; |
| 1716 | 1730 | ||
| @@ -1906,7 +1920,8 @@ MODULE_DEVICE_TABLE(of, dw_dma_id_table); | |||
| 1906 | #endif | 1920 | #endif |
| 1907 | 1921 | ||
| 1908 | static const struct platform_device_id dw_dma_ids[] = { | 1922 | static const struct platform_device_id dw_dma_ids[] = { |
| 1909 | { "INTL9C60", 0 }, | 1923 | /* Name, Request Line Base */ |
| 1924 | { "INTL9C60", (kernel_ulong_t)16 }, | ||
| 1910 | { } | 1925 | { } |
| 1911 | }; | 1926 | }; |
| 1912 | 1927 | ||
diff --git a/drivers/dma/dw_dmac_regs.h b/drivers/dma/dw_dmac_regs.h index cf0ce5c77d60..4d02c3669b75 100644 --- a/drivers/dma/dw_dmac_regs.h +++ b/drivers/dma/dw_dmac_regs.h | |||
| @@ -247,6 +247,7 @@ struct dw_dma { | |||
| 247 | /* hardware configuration */ | 247 | /* hardware configuration */ |
| 248 | unsigned char nr_masters; | 248 | unsigned char nr_masters; |
| 249 | unsigned char data_width[4]; | 249 | unsigned char data_width[4]; |
| 250 | unsigned int request_line_base; | ||
| 250 | 251 | ||
| 251 | struct dw_dma_chan chan[0]; | 252 | struct dw_dma_chan chan[0]; |
| 252 | }; | 253 | }; |
diff --git a/drivers/extcon/extcon-max77693.c b/drivers/extcon/extcon-max77693.c index b70e3815c459..8f3c947b0029 100644 --- a/drivers/extcon/extcon-max77693.c +++ b/drivers/extcon/extcon-max77693.c | |||
| @@ -32,6 +32,38 @@ | |||
| 32 | #define DEV_NAME "max77693-muic" | 32 | #define DEV_NAME "max77693-muic" |
| 33 | #define DELAY_MS_DEFAULT 20000 /* unit: millisecond */ | 33 | #define DELAY_MS_DEFAULT 20000 /* unit: millisecond */ |
| 34 | 34 | ||
| 35 | /* | ||
| 36 | * Default value of MAX77693 register to bring up MUIC device. | ||
| 37 | * If user don't set some initial value for MUIC device through platform data, | ||
| 38 | * extcon-max77693 driver use 'default_init_data' to bring up base operation | ||
| 39 | * of MAX77693 MUIC device. | ||
| 40 | */ | ||
| 41 | struct max77693_reg_data default_init_data[] = { | ||
| 42 | { | ||
| 43 | /* STATUS2 - [3]ChgDetRun */ | ||
| 44 | .addr = MAX77693_MUIC_REG_STATUS2, | ||
| 45 | .data = STATUS2_CHGDETRUN_MASK, | ||
| 46 | }, { | ||
| 47 | /* INTMASK1 - Unmask [3]ADC1KM,[0]ADCM */ | ||
| 48 | .addr = MAX77693_MUIC_REG_INTMASK1, | ||
| 49 | .data = INTMASK1_ADC1K_MASK | ||
| 50 | | INTMASK1_ADC_MASK, | ||
| 51 | }, { | ||
| 52 | /* INTMASK2 - Unmask [0]ChgTypM */ | ||
| 53 | .addr = MAX77693_MUIC_REG_INTMASK2, | ||
| 54 | .data = INTMASK2_CHGTYP_MASK, | ||
| 55 | }, { | ||
| 56 | /* INTMASK3 - Mask all of interrupts */ | ||
| 57 | .addr = MAX77693_MUIC_REG_INTMASK3, | ||
| 58 | .data = 0x0, | ||
| 59 | }, { | ||
| 60 | /* CDETCTRL2 */ | ||
| 61 | .addr = MAX77693_MUIC_REG_CDETCTRL2, | ||
| 62 | .data = CDETCTRL2_VIDRMEN_MASK | ||
| 63 | | CDETCTRL2_DXOVPEN_MASK, | ||
| 64 | }, | ||
| 65 | }; | ||
| 66 | |||
| 35 | enum max77693_muic_adc_debounce_time { | 67 | enum max77693_muic_adc_debounce_time { |
| 36 | ADC_DEBOUNCE_TIME_5MS = 0, | 68 | ADC_DEBOUNCE_TIME_5MS = 0, |
| 37 | ADC_DEBOUNCE_TIME_10MS, | 69 | ADC_DEBOUNCE_TIME_10MS, |
| @@ -1045,8 +1077,9 @@ static int max77693_muic_probe(struct platform_device *pdev) | |||
| 1045 | { | 1077 | { |
| 1046 | struct max77693_dev *max77693 = dev_get_drvdata(pdev->dev.parent); | 1078 | struct max77693_dev *max77693 = dev_get_drvdata(pdev->dev.parent); |
| 1047 | struct max77693_platform_data *pdata = dev_get_platdata(max77693->dev); | 1079 | struct max77693_platform_data *pdata = dev_get_platdata(max77693->dev); |
| 1048 | struct max77693_muic_platform_data *muic_pdata = pdata->muic_data; | ||
| 1049 | struct max77693_muic_info *info; | 1080 | struct max77693_muic_info *info; |
| 1081 | struct max77693_reg_data *init_data; | ||
| 1082 | int num_init_data; | ||
| 1050 | int delay_jiffies; | 1083 | int delay_jiffies; |
| 1051 | int ret; | 1084 | int ret; |
| 1052 | int i; | 1085 | int i; |
| @@ -1145,15 +1178,25 @@ static int max77693_muic_probe(struct platform_device *pdev) | |||
| 1145 | goto err_irq; | 1178 | goto err_irq; |
| 1146 | } | 1179 | } |
| 1147 | 1180 | ||
| 1148 | /* Initialize MUIC register by using platform data */ | 1181 | |
| 1149 | for (i = 0 ; i < muic_pdata->num_init_data ; i++) { | 1182 | /* Initialize MUIC register by using platform data or default data */ |
| 1150 | enum max77693_irq_source irq_src = MAX77693_IRQ_GROUP_NR; | 1183 | if (pdata->muic_data) { |
| 1184 | init_data = pdata->muic_data->init_data; | ||
| 1185 | num_init_data = pdata->muic_data->num_init_data; | ||
| 1186 | } else { | ||
| 1187 | init_data = default_init_data; | ||
| 1188 | num_init_data = ARRAY_SIZE(default_init_data); | ||
| 1189 | } | ||
| 1190 | |||
| 1191 | for (i = 0 ; i < num_init_data ; i++) { | ||
| 1192 | enum max77693_irq_source irq_src | ||
| 1193 | = MAX77693_IRQ_GROUP_NR; | ||
| 1151 | 1194 | ||
| 1152 | max77693_write_reg(info->max77693->regmap_muic, | 1195 | max77693_write_reg(info->max77693->regmap_muic, |
| 1153 | muic_pdata->init_data[i].addr, | 1196 | init_data[i].addr, |
| 1154 | muic_pdata->init_data[i].data); | 1197 | init_data[i].data); |
| 1155 | 1198 | ||
| 1156 | switch (muic_pdata->init_data[i].addr) { | 1199 | switch (init_data[i].addr) { |
| 1157 | case MAX77693_MUIC_REG_INTMASK1: | 1200 | case MAX77693_MUIC_REG_INTMASK1: |
| 1158 | irq_src = MUIC_INT1; | 1201 | irq_src = MUIC_INT1; |
| 1159 | break; | 1202 | break; |
| @@ -1167,22 +1210,40 @@ static int max77693_muic_probe(struct platform_device *pdev) | |||
| 1167 | 1210 | ||
| 1168 | if (irq_src < MAX77693_IRQ_GROUP_NR) | 1211 | if (irq_src < MAX77693_IRQ_GROUP_NR) |
| 1169 | info->max77693->irq_masks_cur[irq_src] | 1212 | info->max77693->irq_masks_cur[irq_src] |
| 1170 | = muic_pdata->init_data[i].data; | 1213 | = init_data[i].data; |
| 1171 | } | 1214 | } |
| 1172 | 1215 | ||
| 1173 | /* | 1216 | if (pdata->muic_data) { |
| 1174 | * Default usb/uart path whether UART/USB or AUX_UART/AUX_USB | 1217 | struct max77693_muic_platform_data *muic_pdata = pdata->muic_data; |
| 1175 | * h/w path of COMP2/COMN1 on CONTROL1 register. | ||
| 1176 | */ | ||
| 1177 | if (muic_pdata->path_uart) | ||
| 1178 | info->path_uart = muic_pdata->path_uart; | ||
| 1179 | else | ||
| 1180 | info->path_uart = CONTROL1_SW_UART; | ||
| 1181 | 1218 | ||
| 1182 | if (muic_pdata->path_usb) | 1219 | /* |
| 1183 | info->path_usb = muic_pdata->path_usb; | 1220 | * Default usb/uart path whether UART/USB or AUX_UART/AUX_USB |
| 1184 | else | 1221 | * h/w path of COMP2/COMN1 on CONTROL1 register. |
| 1222 | */ | ||
| 1223 | if (muic_pdata->path_uart) | ||
| 1224 | info->path_uart = muic_pdata->path_uart; | ||
| 1225 | else | ||
| 1226 | info->path_uart = CONTROL1_SW_UART; | ||
| 1227 | |||
| 1228 | if (muic_pdata->path_usb) | ||
| 1229 | info->path_usb = muic_pdata->path_usb; | ||
| 1230 | else | ||
| 1231 | info->path_usb = CONTROL1_SW_USB; | ||
| 1232 | |||
| 1233 | /* | ||
| 1234 | * Default delay time for detecting cable state | ||
| 1235 | * after certain time. | ||
| 1236 | */ | ||
| 1237 | if (muic_pdata->detcable_delay_ms) | ||
| 1238 | delay_jiffies = | ||
| 1239 | msecs_to_jiffies(muic_pdata->detcable_delay_ms); | ||
| 1240 | else | ||
| 1241 | delay_jiffies = msecs_to_jiffies(DELAY_MS_DEFAULT); | ||
| 1242 | } else { | ||
| 1185 | info->path_usb = CONTROL1_SW_USB; | 1243 | info->path_usb = CONTROL1_SW_USB; |
| 1244 | info->path_uart = CONTROL1_SW_UART; | ||
| 1245 | delay_jiffies = msecs_to_jiffies(DELAY_MS_DEFAULT); | ||
| 1246 | } | ||
| 1186 | 1247 | ||
| 1187 | /* Set initial path for UART */ | 1248 | /* Set initial path for UART */ |
| 1188 | max77693_muic_set_path(info, info->path_uart, true); | 1249 | max77693_muic_set_path(info, info->path_uart, true); |
| @@ -1208,10 +1269,6 @@ static int max77693_muic_probe(struct platform_device *pdev) | |||
| 1208 | * driver should notify cable state to upper layer. | 1269 | * driver should notify cable state to upper layer. |
| 1209 | */ | 1270 | */ |
| 1210 | INIT_DELAYED_WORK(&info->wq_detcable, max77693_muic_detect_cable_wq); | 1271 | INIT_DELAYED_WORK(&info->wq_detcable, max77693_muic_detect_cable_wq); |
| 1211 | if (muic_pdata->detcable_delay_ms) | ||
| 1212 | delay_jiffies = msecs_to_jiffies(muic_pdata->detcable_delay_ms); | ||
| 1213 | else | ||
| 1214 | delay_jiffies = msecs_to_jiffies(DELAY_MS_DEFAULT); | ||
| 1215 | schedule_delayed_work(&info->wq_detcable, delay_jiffies); | 1272 | schedule_delayed_work(&info->wq_detcable, delay_jiffies); |
| 1216 | 1273 | ||
| 1217 | return ret; | 1274 | return ret; |
diff --git a/drivers/extcon/extcon-max8997.c b/drivers/extcon/extcon-max8997.c index e636d950ad6c..69641bcae325 100644 --- a/drivers/extcon/extcon-max8997.c +++ b/drivers/extcon/extcon-max8997.c | |||
| @@ -712,29 +712,45 @@ static int max8997_muic_probe(struct platform_device *pdev) | |||
| 712 | goto err_irq; | 712 | goto err_irq; |
| 713 | } | 713 | } |
| 714 | 714 | ||
| 715 | /* Initialize registers according to platform data */ | ||
| 716 | if (pdata->muic_pdata) { | 715 | if (pdata->muic_pdata) { |
| 717 | struct max8997_muic_platform_data *mdata = info->muic_pdata; | 716 | struct max8997_muic_platform_data *muic_pdata |
| 718 | 717 | = pdata->muic_pdata; | |
| 719 | for (i = 0; i < mdata->num_init_data; i++) { | 718 | |
| 720 | max8997_write_reg(info->muic, mdata->init_data[i].addr, | 719 | /* Initialize registers according to platform data */ |
| 721 | mdata->init_data[i].data); | 720 | for (i = 0; i < muic_pdata->num_init_data; i++) { |
| 721 | max8997_write_reg(info->muic, | ||
| 722 | muic_pdata->init_data[i].addr, | ||
| 723 | muic_pdata->init_data[i].data); | ||
| 722 | } | 724 | } |
| 723 | } | ||
| 724 | 725 | ||
| 725 | /* | 726 | /* |
| 726 | * Default usb/uart path whether UART/USB or AUX_UART/AUX_USB | 727 | * Default usb/uart path whether UART/USB or AUX_UART/AUX_USB |
| 727 | * h/w path of COMP2/COMN1 on CONTROL1 register. | 728 | * h/w path of COMP2/COMN1 on CONTROL1 register. |
| 728 | */ | 729 | */ |
| 729 | if (pdata->muic_pdata->path_uart) | 730 | if (muic_pdata->path_uart) |
| 730 | info->path_uart = pdata->muic_pdata->path_uart; | 731 | info->path_uart = muic_pdata->path_uart; |
| 731 | else | 732 | else |
| 732 | info->path_uart = CONTROL1_SW_UART; | 733 | info->path_uart = CONTROL1_SW_UART; |
| 733 | 734 | ||
| 734 | if (pdata->muic_pdata->path_usb) | 735 | if (muic_pdata->path_usb) |
| 735 | info->path_usb = pdata->muic_pdata->path_usb; | 736 | info->path_usb = muic_pdata->path_usb; |
| 736 | else | 737 | else |
| 738 | info->path_usb = CONTROL1_SW_USB; | ||
| 739 | |||
| 740 | /* | ||
| 741 | * Default delay time for detecting cable state | ||
| 742 | * after certain time. | ||
| 743 | */ | ||
| 744 | if (muic_pdata->detcable_delay_ms) | ||
| 745 | delay_jiffies = | ||
| 746 | msecs_to_jiffies(muic_pdata->detcable_delay_ms); | ||
| 747 | else | ||
| 748 | delay_jiffies = msecs_to_jiffies(DELAY_MS_DEFAULT); | ||
| 749 | } else { | ||
| 750 | info->path_uart = CONTROL1_SW_UART; | ||
| 737 | info->path_usb = CONTROL1_SW_USB; | 751 | info->path_usb = CONTROL1_SW_USB; |
| 752 | delay_jiffies = msecs_to_jiffies(DELAY_MS_DEFAULT); | ||
| 753 | } | ||
| 738 | 754 | ||
| 739 | /* Set initial path for UART */ | 755 | /* Set initial path for UART */ |
| 740 | max8997_muic_set_path(info, info->path_uart, true); | 756 | max8997_muic_set_path(info, info->path_uart, true); |
| @@ -751,10 +767,6 @@ static int max8997_muic_probe(struct platform_device *pdev) | |||
| 751 | * driver should notify cable state to upper layer. | 767 | * driver should notify cable state to upper layer. |
| 752 | */ | 768 | */ |
| 753 | INIT_DELAYED_WORK(&info->wq_detcable, max8997_muic_detect_cable_wq); | 769 | INIT_DELAYED_WORK(&info->wq_detcable, max8997_muic_detect_cable_wq); |
| 754 | if (pdata->muic_pdata->detcable_delay_ms) | ||
| 755 | delay_jiffies = msecs_to_jiffies(pdata->muic_pdata->detcable_delay_ms); | ||
| 756 | else | ||
| 757 | delay_jiffies = msecs_to_jiffies(DELAY_MS_DEFAULT); | ||
| 758 | schedule_delayed_work(&info->wq_detcable, delay_jiffies); | 770 | schedule_delayed_work(&info->wq_detcable, delay_jiffies); |
| 759 | 771 | ||
| 760 | return 0; | 772 | return 0; |
diff --git a/drivers/firmware/Kconfig b/drivers/firmware/Kconfig index 9b00072a020f..42c759a4d047 100644 --- a/drivers/firmware/Kconfig +++ b/drivers/firmware/Kconfig | |||
| @@ -53,6 +53,24 @@ config EFI_VARS | |||
| 53 | Subsequent efibootmgr releases may be found at: | 53 | Subsequent efibootmgr releases may be found at: |
| 54 | <http://linux.dell.com/efibootmgr> | 54 | <http://linux.dell.com/efibootmgr> |
| 55 | 55 | ||
| 56 | config EFI_VARS_PSTORE | ||
| 57 | bool "Register efivars backend for pstore" | ||
| 58 | depends on EFI_VARS && PSTORE | ||
| 59 | default y | ||
| 60 | help | ||
| 61 | Say Y here to enable use efivars as a backend to pstore. This | ||
| 62 | will allow writing console messages, crash dumps, or anything | ||
| 63 | else supported by pstore to EFI variables. | ||
| 64 | |||
| 65 | config EFI_VARS_PSTORE_DEFAULT_DISABLE | ||
| 66 | bool "Disable using efivars as a pstore backend by default" | ||
| 67 | depends on EFI_VARS_PSTORE | ||
| 68 | default n | ||
| 69 | help | ||
| 70 | Saying Y here will disable the use of efivars as a storage | ||
| 71 | backend for pstore by default. This setting can be overridden | ||
| 72 | using the efivars module's pstore_disable parameter. | ||
| 73 | |||
| 56 | config EFI_PCDP | 74 | config EFI_PCDP |
| 57 | bool "Console device selection via EFI PCDP or HCDP table" | 75 | bool "Console device selection via EFI PCDP or HCDP table" |
| 58 | depends on ACPI && EFI && IA64 | 76 | depends on ACPI && EFI && IA64 |
diff --git a/drivers/firmware/efivars.c b/drivers/firmware/efivars.c index fe62aa392239..7acafb80fd4c 100644 --- a/drivers/firmware/efivars.c +++ b/drivers/firmware/efivars.c | |||
| @@ -103,6 +103,11 @@ MODULE_VERSION(EFIVARS_VERSION); | |||
| 103 | */ | 103 | */ |
| 104 | #define GUID_LEN 36 | 104 | #define GUID_LEN 36 |
| 105 | 105 | ||
| 106 | static bool efivars_pstore_disable = | ||
| 107 | IS_ENABLED(CONFIG_EFI_VARS_PSTORE_DEFAULT_DISABLE); | ||
| 108 | |||
| 109 | module_param_named(pstore_disable, efivars_pstore_disable, bool, 0644); | ||
| 110 | |||
| 106 | /* | 111 | /* |
| 107 | * The maximum size of VariableName + Data = 1024 | 112 | * The maximum size of VariableName + Data = 1024 |
| 108 | * Therefore, it's reasonable to save that much | 113 | * Therefore, it's reasonable to save that much |
| @@ -165,6 +170,7 @@ efivar_create_sysfs_entry(struct efivars *efivars, | |||
| 165 | 170 | ||
| 166 | static void efivar_update_sysfs_entries(struct work_struct *); | 171 | static void efivar_update_sysfs_entries(struct work_struct *); |
| 167 | static DECLARE_WORK(efivar_work, efivar_update_sysfs_entries); | 172 | static DECLARE_WORK(efivar_work, efivar_update_sysfs_entries); |
| 173 | static bool efivar_wq_enabled = true; | ||
| 168 | 174 | ||
| 169 | /* Return the number of unicode characters in data */ | 175 | /* Return the number of unicode characters in data */ |
| 170 | static unsigned long | 176 | static unsigned long |
| @@ -1309,9 +1315,7 @@ static const struct inode_operations efivarfs_dir_inode_operations = { | |||
| 1309 | .create = efivarfs_create, | 1315 | .create = efivarfs_create, |
| 1310 | }; | 1316 | }; |
| 1311 | 1317 | ||
| 1312 | static struct pstore_info efi_pstore_info; | 1318 | #ifdef CONFIG_EFI_VARS_PSTORE |
| 1313 | |||
| 1314 | #ifdef CONFIG_PSTORE | ||
| 1315 | 1319 | ||
| 1316 | static int efi_pstore_open(struct pstore_info *psi) | 1320 | static int efi_pstore_open(struct pstore_info *psi) |
| 1317 | { | 1321 | { |
| @@ -1441,7 +1445,7 @@ static int efi_pstore_write(enum pstore_type_id type, | |||
| 1441 | 1445 | ||
| 1442 | spin_unlock_irqrestore(&efivars->lock, flags); | 1446 | spin_unlock_irqrestore(&efivars->lock, flags); |
| 1443 | 1447 | ||
| 1444 | if (reason == KMSG_DUMP_OOPS) | 1448 | if (reason == KMSG_DUMP_OOPS && efivar_wq_enabled) |
| 1445 | schedule_work(&efivar_work); | 1449 | schedule_work(&efivar_work); |
| 1446 | 1450 | ||
| 1447 | *id = part; | 1451 | *id = part; |
| @@ -1514,38 +1518,6 @@ static int efi_pstore_erase(enum pstore_type_id type, u64 id, int count, | |||
| 1514 | 1518 | ||
| 1515 | return 0; | 1519 | return 0; |
| 1516 | } | 1520 | } |
| 1517 | #else | ||
| 1518 | static int efi_pstore_open(struct pstore_info *psi) | ||
| 1519 | { | ||
| 1520 | return 0; | ||
| 1521 | } | ||
| 1522 | |||
| 1523 | static int efi_pstore_close(struct pstore_info *psi) | ||
| 1524 | { | ||
| 1525 | return 0; | ||
| 1526 | } | ||
| 1527 | |||
| 1528 | static ssize_t efi_pstore_read(u64 *id, enum pstore_type_id *type, int *count, | ||
| 1529 | struct timespec *timespec, | ||
| 1530 | char **buf, struct pstore_info *psi) | ||
| 1531 | { | ||
| 1532 | return -1; | ||
| 1533 | } | ||
| 1534 | |||
| 1535 | static int efi_pstore_write(enum pstore_type_id type, | ||
| 1536 | enum kmsg_dump_reason reason, u64 *id, | ||
| 1537 | unsigned int part, int count, size_t size, | ||
| 1538 | struct pstore_info *psi) | ||
| 1539 | { | ||
| 1540 | return 0; | ||
| 1541 | } | ||
| 1542 | |||
| 1543 | static int efi_pstore_erase(enum pstore_type_id type, u64 id, int count, | ||
| 1544 | struct timespec time, struct pstore_info *psi) | ||
| 1545 | { | ||
| 1546 | return 0; | ||
| 1547 | } | ||
| 1548 | #endif | ||
| 1549 | 1521 | ||
| 1550 | static struct pstore_info efi_pstore_info = { | 1522 | static struct pstore_info efi_pstore_info = { |
| 1551 | .owner = THIS_MODULE, | 1523 | .owner = THIS_MODULE, |
| @@ -1557,6 +1529,24 @@ static struct pstore_info efi_pstore_info = { | |||
| 1557 | .erase = efi_pstore_erase, | 1529 | .erase = efi_pstore_erase, |
| 1558 | }; | 1530 | }; |
| 1559 | 1531 | ||
| 1532 | static void efivar_pstore_register(struct efivars *efivars) | ||
| 1533 | { | ||
| 1534 | efivars->efi_pstore_info = efi_pstore_info; | ||
| 1535 | efivars->efi_pstore_info.buf = kmalloc(4096, GFP_KERNEL); | ||
| 1536 | if (efivars->efi_pstore_info.buf) { | ||
| 1537 | efivars->efi_pstore_info.bufsize = 1024; | ||
| 1538 | efivars->efi_pstore_info.data = efivars; | ||
| 1539 | spin_lock_init(&efivars->efi_pstore_info.buf_lock); | ||
| 1540 | pstore_register(&efivars->efi_pstore_info); | ||
| 1541 | } | ||
| 1542 | } | ||
| 1543 | #else | ||
| 1544 | static void efivar_pstore_register(struct efivars *efivars) | ||
| 1545 | { | ||
| 1546 | return; | ||
| 1547 | } | ||
| 1548 | #endif | ||
| 1549 | |||
| 1560 | static ssize_t efivar_create(struct file *filp, struct kobject *kobj, | 1550 | static ssize_t efivar_create(struct file *filp, struct kobject *kobj, |
| 1561 | struct bin_attribute *bin_attr, | 1551 | struct bin_attribute *bin_attr, |
| 1562 | char *buf, loff_t pos, size_t count) | 1552 | char *buf, loff_t pos, size_t count) |
| @@ -1716,6 +1706,31 @@ static bool variable_is_present(efi_char16_t *variable_name, efi_guid_t *vendor) | |||
| 1716 | return found; | 1706 | return found; |
| 1717 | } | 1707 | } |
| 1718 | 1708 | ||
| 1709 | /* | ||
| 1710 | * Returns the size of variable_name, in bytes, including the | ||
| 1711 | * terminating NULL character, or variable_name_size if no NULL | ||
| 1712 | * character is found among the first variable_name_size bytes. | ||
| 1713 | */ | ||
| 1714 | static unsigned long var_name_strnsize(efi_char16_t *variable_name, | ||
| 1715 | unsigned long variable_name_size) | ||
| 1716 | { | ||
| 1717 | unsigned long len; | ||
| 1718 | efi_char16_t c; | ||
| 1719 | |||
| 1720 | /* | ||
| 1721 | * The variable name is, by definition, a NULL-terminated | ||
| 1722 | * string, so make absolutely sure that variable_name_size is | ||
| 1723 | * the value we expect it to be. If not, return the real size. | ||
| 1724 | */ | ||
| 1725 | for (len = 2; len <= variable_name_size; len += sizeof(c)) { | ||
| 1726 | c = variable_name[(len / sizeof(c)) - 1]; | ||
| 1727 | if (!c) | ||
| 1728 | break; | ||
| 1729 | } | ||
| 1730 | |||
| 1731 | return min(len, variable_name_size); | ||
| 1732 | } | ||
| 1733 | |||
| 1719 | static void efivar_update_sysfs_entries(struct work_struct *work) | 1734 | static void efivar_update_sysfs_entries(struct work_struct *work) |
| 1720 | { | 1735 | { |
| 1721 | struct efivars *efivars = &__efivars; | 1736 | struct efivars *efivars = &__efivars; |
| @@ -1756,10 +1771,13 @@ static void efivar_update_sysfs_entries(struct work_struct *work) | |||
| 1756 | if (!found) { | 1771 | if (!found) { |
| 1757 | kfree(variable_name); | 1772 | kfree(variable_name); |
| 1758 | break; | 1773 | break; |
| 1759 | } else | 1774 | } else { |
| 1775 | variable_name_size = var_name_strnsize(variable_name, | ||
| 1776 | variable_name_size); | ||
| 1760 | efivar_create_sysfs_entry(efivars, | 1777 | efivar_create_sysfs_entry(efivars, |
| 1761 | variable_name_size, | 1778 | variable_name_size, |
| 1762 | variable_name, &vendor); | 1779 | variable_name, &vendor); |
| 1780 | } | ||
| 1763 | } | 1781 | } |
| 1764 | } | 1782 | } |
| 1765 | 1783 | ||
| @@ -1958,6 +1976,35 @@ void unregister_efivars(struct efivars *efivars) | |||
| 1958 | } | 1976 | } |
| 1959 | EXPORT_SYMBOL_GPL(unregister_efivars); | 1977 | EXPORT_SYMBOL_GPL(unregister_efivars); |
| 1960 | 1978 | ||
| 1979 | /* | ||
| 1980 | * Print a warning when duplicate EFI variables are encountered and | ||
| 1981 | * disable the sysfs workqueue since the firmware is buggy. | ||
| 1982 | */ | ||
| 1983 | static void dup_variable_bug(efi_char16_t *s16, efi_guid_t *vendor_guid, | ||
| 1984 | unsigned long len16) | ||
| 1985 | { | ||
| 1986 | size_t i, len8 = len16 / sizeof(efi_char16_t); | ||
| 1987 | char *s8; | ||
| 1988 | |||
| 1989 | /* | ||
| 1990 | * Disable the workqueue since the algorithm it uses for | ||
| 1991 | * detecting new variables won't work with this buggy | ||
| 1992 | * implementation of GetNextVariableName(). | ||
| 1993 | */ | ||
| 1994 | efivar_wq_enabled = false; | ||
| 1995 | |||
| 1996 | s8 = kzalloc(len8, GFP_KERNEL); | ||
| 1997 | if (!s8) | ||
| 1998 | return; | ||
| 1999 | |||
| 2000 | for (i = 0; i < len8; i++) | ||
| 2001 | s8[i] = s16[i]; | ||
| 2002 | |||
| 2003 | printk(KERN_WARNING "efivars: duplicate variable: %s-%pUl\n", | ||
| 2004 | s8, vendor_guid); | ||
| 2005 | kfree(s8); | ||
| 2006 | } | ||
| 2007 | |||
| 1961 | int register_efivars(struct efivars *efivars, | 2008 | int register_efivars(struct efivars *efivars, |
| 1962 | const struct efivar_operations *ops, | 2009 | const struct efivar_operations *ops, |
| 1963 | struct kobject *parent_kobj) | 2010 | struct kobject *parent_kobj) |
| @@ -2006,6 +2053,24 @@ int register_efivars(struct efivars *efivars, | |||
| 2006 | &vendor_guid); | 2053 | &vendor_guid); |
| 2007 | switch (status) { | 2054 | switch (status) { |
| 2008 | case EFI_SUCCESS: | 2055 | case EFI_SUCCESS: |
| 2056 | variable_name_size = var_name_strnsize(variable_name, | ||
| 2057 | variable_name_size); | ||
| 2058 | |||
| 2059 | /* | ||
| 2060 | * Some firmware implementations return the | ||
| 2061 | * same variable name on multiple calls to | ||
| 2062 | * get_next_variable(). Terminate the loop | ||
| 2063 | * immediately as there is no guarantee that | ||
| 2064 | * we'll ever see a different variable name, | ||
| 2065 | * and may end up looping here forever. | ||
| 2066 | */ | ||
| 2067 | if (variable_is_present(variable_name, &vendor_guid)) { | ||
| 2068 | dup_variable_bug(variable_name, &vendor_guid, | ||
| 2069 | variable_name_size); | ||
| 2070 | status = EFI_NOT_FOUND; | ||
| 2071 | break; | ||
| 2072 | } | ||
| 2073 | |||
| 2009 | efivar_create_sysfs_entry(efivars, | 2074 | efivar_create_sysfs_entry(efivars, |
| 2010 | variable_name_size, | 2075 | variable_name_size, |
| 2011 | variable_name, | 2076 | variable_name, |
| @@ -2025,15 +2090,8 @@ int register_efivars(struct efivars *efivars, | |||
| 2025 | if (error) | 2090 | if (error) |
| 2026 | unregister_efivars(efivars); | 2091 | unregister_efivars(efivars); |
| 2027 | 2092 | ||
| 2028 | efivars->efi_pstore_info = efi_pstore_info; | 2093 | if (!efivars_pstore_disable) |
| 2029 | 2094 | efivar_pstore_register(efivars); | |
| 2030 | efivars->efi_pstore_info.buf = kmalloc(4096, GFP_KERNEL); | ||
| 2031 | if (efivars->efi_pstore_info.buf) { | ||
| 2032 | efivars->efi_pstore_info.bufsize = 1024; | ||
| 2033 | efivars->efi_pstore_info.data = efivars; | ||
| 2034 | spin_lock_init(&efivars->efi_pstore_info.buf_lock); | ||
| 2035 | pstore_register(&efivars->efi_pstore_info); | ||
| 2036 | } | ||
| 2037 | 2095 | ||
| 2038 | register_filesystem(&efivarfs_type); | 2096 | register_filesystem(&efivarfs_type); |
| 2039 | 2097 | ||
diff --git a/drivers/gpio/gpiolib-of.c b/drivers/gpio/gpiolib-of.c index a71a54a3e3f7..5150df6cba08 100644 --- a/drivers/gpio/gpiolib-of.c +++ b/drivers/gpio/gpiolib-of.c | |||
| @@ -193,7 +193,7 @@ static void of_gpiochip_add_pin_range(struct gpio_chip *chip) | |||
| 193 | if (!np) | 193 | if (!np) |
| 194 | return; | 194 | return; |
| 195 | 195 | ||
| 196 | do { | 196 | for (;; index++) { |
| 197 | ret = of_parse_phandle_with_args(np, "gpio-ranges", | 197 | ret = of_parse_phandle_with_args(np, "gpio-ranges", |
| 198 | "#gpio-range-cells", index, &pinspec); | 198 | "#gpio-range-cells", index, &pinspec); |
| 199 | if (ret) | 199 | if (ret) |
| @@ -222,8 +222,7 @@ static void of_gpiochip_add_pin_range(struct gpio_chip *chip) | |||
| 222 | 222 | ||
| 223 | if (ret) | 223 | if (ret) |
| 224 | break; | 224 | break; |
| 225 | 225 | } | |
| 226 | } while (index++); | ||
| 227 | } | 226 | } |
| 228 | 227 | ||
| 229 | #else | 228 | #else |
diff --git a/drivers/gpu/drm/exynos/exynos_drm_fimd.c b/drivers/gpu/drm/exynos/exynos_drm_fimd.c index 36493ce71f9a..98cc14725ba9 100644 --- a/drivers/gpu/drm/exynos/exynos_drm_fimd.c +++ b/drivers/gpu/drm/exynos/exynos_drm_fimd.c | |||
| @@ -38,11 +38,12 @@ | |||
| 38 | /* position control register for hardware window 0, 2 ~ 4.*/ | 38 | /* position control register for hardware window 0, 2 ~ 4.*/ |
| 39 | #define VIDOSD_A(win) (VIDOSD_BASE + 0x00 + (win) * 16) | 39 | #define VIDOSD_A(win) (VIDOSD_BASE + 0x00 + (win) * 16) |
| 40 | #define VIDOSD_B(win) (VIDOSD_BASE + 0x04 + (win) * 16) | 40 | #define VIDOSD_B(win) (VIDOSD_BASE + 0x04 + (win) * 16) |
| 41 | /* size control register for hardware window 0. */ | 41 | /* |
| 42 | #define VIDOSD_C_SIZE_W0 (VIDOSD_BASE + 0x08) | 42 | * size control register for hardware windows 0 and alpha control register |
| 43 | /* alpha control register for hardware window 1 ~ 4. */ | 43 | * for hardware windows 1 ~ 4 |
| 44 | #define VIDOSD_C(win) (VIDOSD_BASE + 0x18 + (win) * 16) | 44 | */ |
| 45 | /* size control register for hardware window 1 ~ 4. */ | 45 | #define VIDOSD_C(win) (VIDOSD_BASE + 0x08 + (win) * 16) |
| 46 | /* size control register for hardware windows 1 ~ 2. */ | ||
| 46 | #define VIDOSD_D(win) (VIDOSD_BASE + 0x0C + (win) * 16) | 47 | #define VIDOSD_D(win) (VIDOSD_BASE + 0x0C + (win) * 16) |
| 47 | 48 | ||
| 48 | #define VIDWx_BUF_START(win, buf) (VIDW_BUF_START(buf) + (win) * 8) | 49 | #define VIDWx_BUF_START(win, buf) (VIDW_BUF_START(buf) + (win) * 8) |
| @@ -50,9 +51,9 @@ | |||
| 50 | #define VIDWx_BUF_SIZE(win, buf) (VIDW_BUF_SIZE(buf) + (win) * 4) | 51 | #define VIDWx_BUF_SIZE(win, buf) (VIDW_BUF_SIZE(buf) + (win) * 4) |
| 51 | 52 | ||
| 52 | /* color key control register for hardware window 1 ~ 4. */ | 53 | /* color key control register for hardware window 1 ~ 4. */ |
| 53 | #define WKEYCON0_BASE(x) ((WKEYCON0 + 0x140) + (x * 8)) | 54 | #define WKEYCON0_BASE(x) ((WKEYCON0 + 0x140) + ((x - 1) * 8)) |
| 54 | /* color key value register for hardware window 1 ~ 4. */ | 55 | /* color key value register for hardware window 1 ~ 4. */ |
| 55 | #define WKEYCON1_BASE(x) ((WKEYCON1 + 0x140) + (x * 8)) | 56 | #define WKEYCON1_BASE(x) ((WKEYCON1 + 0x140) + ((x - 1) * 8)) |
| 56 | 57 | ||
| 57 | /* FIMD has totally five hardware windows. */ | 58 | /* FIMD has totally five hardware windows. */ |
| 58 | #define WINDOWS_NR 5 | 59 | #define WINDOWS_NR 5 |
| @@ -109,9 +110,9 @@ struct fimd_context { | |||
| 109 | 110 | ||
| 110 | #ifdef CONFIG_OF | 111 | #ifdef CONFIG_OF |
| 111 | static const struct of_device_id fimd_driver_dt_match[] = { | 112 | static const struct of_device_id fimd_driver_dt_match[] = { |
| 112 | { .compatible = "samsung,exynos4-fimd", | 113 | { .compatible = "samsung,exynos4210-fimd", |
| 113 | .data = &exynos4_fimd_driver_data }, | 114 | .data = &exynos4_fimd_driver_data }, |
| 114 | { .compatible = "samsung,exynos5-fimd", | 115 | { .compatible = "samsung,exynos5250-fimd", |
| 115 | .data = &exynos5_fimd_driver_data }, | 116 | .data = &exynos5_fimd_driver_data }, |
| 116 | {}, | 117 | {}, |
| 117 | }; | 118 | }; |
| @@ -581,7 +582,7 @@ static void fimd_win_commit(struct device *dev, int zpos) | |||
| 581 | if (win != 3 && win != 4) { | 582 | if (win != 3 && win != 4) { |
| 582 | u32 offset = VIDOSD_D(win); | 583 | u32 offset = VIDOSD_D(win); |
| 583 | if (win == 0) | 584 | if (win == 0) |
| 584 | offset = VIDOSD_C_SIZE_W0; | 585 | offset = VIDOSD_C(win); |
| 585 | val = win_data->ovl_width * win_data->ovl_height; | 586 | val = win_data->ovl_width * win_data->ovl_height; |
| 586 | writel(val, ctx->regs + offset); | 587 | writel(val, ctx->regs + offset); |
| 587 | 588 | ||
diff --git a/drivers/gpu/drm/exynos/exynos_drm_g2d.c b/drivers/gpu/drm/exynos/exynos_drm_g2d.c index 3b0da0378acf..47a493c8a71f 100644 --- a/drivers/gpu/drm/exynos/exynos_drm_g2d.c +++ b/drivers/gpu/drm/exynos/exynos_drm_g2d.c | |||
| @@ -48,8 +48,14 @@ | |||
| 48 | 48 | ||
| 49 | /* registers for base address */ | 49 | /* registers for base address */ |
| 50 | #define G2D_SRC_BASE_ADDR 0x0304 | 50 | #define G2D_SRC_BASE_ADDR 0x0304 |
| 51 | #define G2D_SRC_COLOR_MODE 0x030C | ||
| 52 | #define G2D_SRC_LEFT_TOP 0x0310 | ||
| 53 | #define G2D_SRC_RIGHT_BOTTOM 0x0314 | ||
| 51 | #define G2D_SRC_PLANE2_BASE_ADDR 0x0318 | 54 | #define G2D_SRC_PLANE2_BASE_ADDR 0x0318 |
| 52 | #define G2D_DST_BASE_ADDR 0x0404 | 55 | #define G2D_DST_BASE_ADDR 0x0404 |
| 56 | #define G2D_DST_COLOR_MODE 0x040C | ||
| 57 | #define G2D_DST_LEFT_TOP 0x0410 | ||
| 58 | #define G2D_DST_RIGHT_BOTTOM 0x0414 | ||
| 53 | #define G2D_DST_PLANE2_BASE_ADDR 0x0418 | 59 | #define G2D_DST_PLANE2_BASE_ADDR 0x0418 |
| 54 | #define G2D_PAT_BASE_ADDR 0x0500 | 60 | #define G2D_PAT_BASE_ADDR 0x0500 |
| 55 | #define G2D_MSK_BASE_ADDR 0x0520 | 61 | #define G2D_MSK_BASE_ADDR 0x0520 |
| @@ -82,7 +88,7 @@ | |||
| 82 | #define G2D_DMA_LIST_DONE_COUNT_OFFSET 17 | 88 | #define G2D_DMA_LIST_DONE_COUNT_OFFSET 17 |
| 83 | 89 | ||
| 84 | /* G2D_DMA_HOLD_CMD */ | 90 | /* G2D_DMA_HOLD_CMD */ |
| 85 | #define G2D_USET_HOLD (1 << 2) | 91 | #define G2D_USER_HOLD (1 << 2) |
| 86 | #define G2D_LIST_HOLD (1 << 1) | 92 | #define G2D_LIST_HOLD (1 << 1) |
| 87 | #define G2D_BITBLT_HOLD (1 << 0) | 93 | #define G2D_BITBLT_HOLD (1 << 0) |
| 88 | 94 | ||
| @@ -91,13 +97,27 @@ | |||
| 91 | #define G2D_START_NHOLT (1 << 1) | 97 | #define G2D_START_NHOLT (1 << 1) |
| 92 | #define G2D_START_BITBLT (1 << 0) | 98 | #define G2D_START_BITBLT (1 << 0) |
| 93 | 99 | ||
| 100 | /* buffer color format */ | ||
| 101 | #define G2D_FMT_XRGB8888 0 | ||
| 102 | #define G2D_FMT_ARGB8888 1 | ||
| 103 | #define G2D_FMT_RGB565 2 | ||
| 104 | #define G2D_FMT_XRGB1555 3 | ||
| 105 | #define G2D_FMT_ARGB1555 4 | ||
| 106 | #define G2D_FMT_XRGB4444 5 | ||
| 107 | #define G2D_FMT_ARGB4444 6 | ||
| 108 | #define G2D_FMT_PACKED_RGB888 7 | ||
| 109 | #define G2D_FMT_A8 11 | ||
| 110 | #define G2D_FMT_L8 12 | ||
| 111 | |||
| 112 | /* buffer valid length */ | ||
| 113 | #define G2D_LEN_MIN 1 | ||
| 114 | #define G2D_LEN_MAX 8000 | ||
| 115 | |||
| 94 | #define G2D_CMDLIST_SIZE (PAGE_SIZE / 4) | 116 | #define G2D_CMDLIST_SIZE (PAGE_SIZE / 4) |
| 95 | #define G2D_CMDLIST_NUM 64 | 117 | #define G2D_CMDLIST_NUM 64 |
| 96 | #define G2D_CMDLIST_POOL_SIZE (G2D_CMDLIST_SIZE * G2D_CMDLIST_NUM) | 118 | #define G2D_CMDLIST_POOL_SIZE (G2D_CMDLIST_SIZE * G2D_CMDLIST_NUM) |
| 97 | #define G2D_CMDLIST_DATA_NUM (G2D_CMDLIST_SIZE / sizeof(u32) - 2) | 119 | #define G2D_CMDLIST_DATA_NUM (G2D_CMDLIST_SIZE / sizeof(u32) - 2) |
| 98 | 120 | ||
| 99 | #define MAX_BUF_ADDR_NR 6 | ||
| 100 | |||
| 101 | /* maximum buffer pool size of userptr is 64MB as default */ | 121 | /* maximum buffer pool size of userptr is 64MB as default */ |
| 102 | #define MAX_POOL (64 * 1024 * 1024) | 122 | #define MAX_POOL (64 * 1024 * 1024) |
| 103 | 123 | ||
| @@ -106,6 +126,17 @@ enum { | |||
| 106 | BUF_TYPE_USERPTR, | 126 | BUF_TYPE_USERPTR, |
| 107 | }; | 127 | }; |
| 108 | 128 | ||
| 129 | enum g2d_reg_type { | ||
| 130 | REG_TYPE_NONE = -1, | ||
| 131 | REG_TYPE_SRC, | ||
| 132 | REG_TYPE_SRC_PLANE2, | ||
| 133 | REG_TYPE_DST, | ||
| 134 | REG_TYPE_DST_PLANE2, | ||
| 135 | REG_TYPE_PAT, | ||
| 136 | REG_TYPE_MSK, | ||
| 137 | MAX_REG_TYPE_NR | ||
| 138 | }; | ||
| 139 | |||
| 109 | /* cmdlist data structure */ | 140 | /* cmdlist data structure */ |
| 110 | struct g2d_cmdlist { | 141 | struct g2d_cmdlist { |
| 111 | u32 head; | 142 | u32 head; |
| @@ -113,6 +144,42 @@ struct g2d_cmdlist { | |||
| 113 | u32 last; /* last data offset */ | 144 | u32 last; /* last data offset */ |
| 114 | }; | 145 | }; |
| 115 | 146 | ||
| 147 | /* | ||
| 148 | * A structure of buffer description | ||
| 149 | * | ||
| 150 | * @format: color format | ||
| 151 | * @left_x: the x coordinates of left top corner | ||
| 152 | * @top_y: the y coordinates of left top corner | ||
| 153 | * @right_x: the x coordinates of right bottom corner | ||
| 154 | * @bottom_y: the y coordinates of right bottom corner | ||
| 155 | * | ||
| 156 | */ | ||
| 157 | struct g2d_buf_desc { | ||
| 158 | unsigned int format; | ||
| 159 | unsigned int left_x; | ||
| 160 | unsigned int top_y; | ||
| 161 | unsigned int right_x; | ||
| 162 | unsigned int bottom_y; | ||
| 163 | }; | ||
| 164 | |||
| 165 | /* | ||
| 166 | * A structure of buffer information | ||
| 167 | * | ||
| 168 | * @map_nr: manages the number of mapped buffers | ||
| 169 | * @reg_types: stores regitster type in the order of requested command | ||
| 170 | * @handles: stores buffer handle in its reg_type position | ||
| 171 | * @types: stores buffer type in its reg_type position | ||
| 172 | * @descs: stores buffer description in its reg_type position | ||
| 173 | * | ||
| 174 | */ | ||
| 175 | struct g2d_buf_info { | ||
| 176 | unsigned int map_nr; | ||
| 177 | enum g2d_reg_type reg_types[MAX_REG_TYPE_NR]; | ||
| 178 | unsigned long handles[MAX_REG_TYPE_NR]; | ||
| 179 | unsigned int types[MAX_REG_TYPE_NR]; | ||
| 180 | struct g2d_buf_desc descs[MAX_REG_TYPE_NR]; | ||
| 181 | }; | ||
| 182 | |||
| 116 | struct drm_exynos_pending_g2d_event { | 183 | struct drm_exynos_pending_g2d_event { |
| 117 | struct drm_pending_event base; | 184 | struct drm_pending_event base; |
| 118 | struct drm_exynos_g2d_event event; | 185 | struct drm_exynos_g2d_event event; |
| @@ -131,14 +198,11 @@ struct g2d_cmdlist_userptr { | |||
| 131 | bool in_pool; | 198 | bool in_pool; |
| 132 | bool out_of_list; | 199 | bool out_of_list; |
| 133 | }; | 200 | }; |
| 134 | |||
| 135 | struct g2d_cmdlist_node { | 201 | struct g2d_cmdlist_node { |
| 136 | struct list_head list; | 202 | struct list_head list; |
| 137 | struct g2d_cmdlist *cmdlist; | 203 | struct g2d_cmdlist *cmdlist; |
| 138 | unsigned int map_nr; | ||
| 139 | unsigned long handles[MAX_BUF_ADDR_NR]; | ||
| 140 | unsigned int obj_type[MAX_BUF_ADDR_NR]; | ||
| 141 | dma_addr_t dma_addr; | 204 | dma_addr_t dma_addr; |
| 205 | struct g2d_buf_info buf_info; | ||
| 142 | 206 | ||
| 143 | struct drm_exynos_pending_g2d_event *event; | 207 | struct drm_exynos_pending_g2d_event *event; |
| 144 | }; | 208 | }; |
| @@ -188,6 +252,7 @@ static int g2d_init_cmdlist(struct g2d_data *g2d) | |||
| 188 | struct exynos_drm_subdrv *subdrv = &g2d->subdrv; | 252 | struct exynos_drm_subdrv *subdrv = &g2d->subdrv; |
| 189 | int nr; | 253 | int nr; |
| 190 | int ret; | 254 | int ret; |
| 255 | struct g2d_buf_info *buf_info; | ||
| 191 | 256 | ||
| 192 | init_dma_attrs(&g2d->cmdlist_dma_attrs); | 257 | init_dma_attrs(&g2d->cmdlist_dma_attrs); |
| 193 | dma_set_attr(DMA_ATTR_WRITE_COMBINE, &g2d->cmdlist_dma_attrs); | 258 | dma_set_attr(DMA_ATTR_WRITE_COMBINE, &g2d->cmdlist_dma_attrs); |
| @@ -209,11 +274,17 @@ static int g2d_init_cmdlist(struct g2d_data *g2d) | |||
| 209 | } | 274 | } |
| 210 | 275 | ||
| 211 | for (nr = 0; nr < G2D_CMDLIST_NUM; nr++) { | 276 | for (nr = 0; nr < G2D_CMDLIST_NUM; nr++) { |
| 277 | unsigned int i; | ||
| 278 | |||
| 212 | node[nr].cmdlist = | 279 | node[nr].cmdlist = |
| 213 | g2d->cmdlist_pool_virt + nr * G2D_CMDLIST_SIZE; | 280 | g2d->cmdlist_pool_virt + nr * G2D_CMDLIST_SIZE; |
| 214 | node[nr].dma_addr = | 281 | node[nr].dma_addr = |
| 215 | g2d->cmdlist_pool + nr * G2D_CMDLIST_SIZE; | 282 | g2d->cmdlist_pool + nr * G2D_CMDLIST_SIZE; |
| 216 | 283 | ||
| 284 | buf_info = &node[nr].buf_info; | ||
| 285 | for (i = 0; i < MAX_REG_TYPE_NR; i++) | ||
| 286 | buf_info->reg_types[i] = REG_TYPE_NONE; | ||
| 287 | |||
| 217 | list_add_tail(&node[nr].list, &g2d->free_cmdlist); | 288 | list_add_tail(&node[nr].list, &g2d->free_cmdlist); |
| 218 | } | 289 | } |
| 219 | 290 | ||
| @@ -450,7 +521,7 @@ static dma_addr_t *g2d_userptr_get_dma_addr(struct drm_device *drm_dev, | |||
| 450 | DMA_BIDIRECTIONAL); | 521 | DMA_BIDIRECTIONAL); |
| 451 | if (ret < 0) { | 522 | if (ret < 0) { |
| 452 | DRM_ERROR("failed to map sgt with dma region.\n"); | 523 | DRM_ERROR("failed to map sgt with dma region.\n"); |
| 453 | goto err_free_sgt; | 524 | goto err_sg_free_table; |
| 454 | } | 525 | } |
| 455 | 526 | ||
| 456 | g2d_userptr->dma_addr = sgt->sgl[0].dma_address; | 527 | g2d_userptr->dma_addr = sgt->sgl[0].dma_address; |
| @@ -467,8 +538,10 @@ static dma_addr_t *g2d_userptr_get_dma_addr(struct drm_device *drm_dev, | |||
| 467 | 538 | ||
| 468 | return &g2d_userptr->dma_addr; | 539 | return &g2d_userptr->dma_addr; |
| 469 | 540 | ||
| 470 | err_free_sgt: | 541 | err_sg_free_table: |
| 471 | sg_free_table(sgt); | 542 | sg_free_table(sgt); |
| 543 | |||
| 544 | err_free_sgt: | ||
| 472 | kfree(sgt); | 545 | kfree(sgt); |
| 473 | sgt = NULL; | 546 | sgt = NULL; |
| 474 | 547 | ||
| @@ -506,36 +579,172 @@ static void g2d_userptr_free_all(struct drm_device *drm_dev, | |||
| 506 | g2d->current_pool = 0; | 579 | g2d->current_pool = 0; |
| 507 | } | 580 | } |
| 508 | 581 | ||
| 582 | static enum g2d_reg_type g2d_get_reg_type(int reg_offset) | ||
| 583 | { | ||
| 584 | enum g2d_reg_type reg_type; | ||
| 585 | |||
| 586 | switch (reg_offset) { | ||
| 587 | case G2D_SRC_BASE_ADDR: | ||
| 588 | case G2D_SRC_COLOR_MODE: | ||
| 589 | case G2D_SRC_LEFT_TOP: | ||
| 590 | case G2D_SRC_RIGHT_BOTTOM: | ||
| 591 | reg_type = REG_TYPE_SRC; | ||
| 592 | break; | ||
| 593 | case G2D_SRC_PLANE2_BASE_ADDR: | ||
| 594 | reg_type = REG_TYPE_SRC_PLANE2; | ||
| 595 | break; | ||
| 596 | case G2D_DST_BASE_ADDR: | ||
| 597 | case G2D_DST_COLOR_MODE: | ||
| 598 | case G2D_DST_LEFT_TOP: | ||
| 599 | case G2D_DST_RIGHT_BOTTOM: | ||
| 600 | reg_type = REG_TYPE_DST; | ||
| 601 | break; | ||
| 602 | case G2D_DST_PLANE2_BASE_ADDR: | ||
| 603 | reg_type = REG_TYPE_DST_PLANE2; | ||
| 604 | break; | ||
| 605 | case G2D_PAT_BASE_ADDR: | ||
| 606 | reg_type = REG_TYPE_PAT; | ||
| 607 | break; | ||
| 608 | case G2D_MSK_BASE_ADDR: | ||
| 609 | reg_type = REG_TYPE_MSK; | ||
| 610 | break; | ||
| 611 | default: | ||
| 612 | reg_type = REG_TYPE_NONE; | ||
| 613 | DRM_ERROR("Unknown register offset![%d]\n", reg_offset); | ||
| 614 | break; | ||
| 615 | }; | ||
| 616 | |||
| 617 | return reg_type; | ||
| 618 | } | ||
| 619 | |||
| 620 | static unsigned long g2d_get_buf_bpp(unsigned int format) | ||
| 621 | { | ||
| 622 | unsigned long bpp; | ||
| 623 | |||
| 624 | switch (format) { | ||
| 625 | case G2D_FMT_XRGB8888: | ||
| 626 | case G2D_FMT_ARGB8888: | ||
| 627 | bpp = 4; | ||
| 628 | break; | ||
| 629 | case G2D_FMT_RGB565: | ||
| 630 | case G2D_FMT_XRGB1555: | ||
| 631 | case G2D_FMT_ARGB1555: | ||
| 632 | case G2D_FMT_XRGB4444: | ||
| 633 | case G2D_FMT_ARGB4444: | ||
| 634 | bpp = 2; | ||
| 635 | break; | ||
| 636 | case G2D_FMT_PACKED_RGB888: | ||
| 637 | bpp = 3; | ||
| 638 | break; | ||
| 639 | default: | ||
| 640 | bpp = 1; | ||
| 641 | break; | ||
| 642 | } | ||
| 643 | |||
| 644 | return bpp; | ||
| 645 | } | ||
| 646 | |||
| 647 | static bool g2d_check_buf_desc_is_valid(struct g2d_buf_desc *buf_desc, | ||
| 648 | enum g2d_reg_type reg_type, | ||
| 649 | unsigned long size) | ||
| 650 | { | ||
| 651 | unsigned int width, height; | ||
| 652 | unsigned long area; | ||
| 653 | |||
| 654 | /* | ||
| 655 | * check source and destination buffers only. | ||
| 656 | * so the others are always valid. | ||
| 657 | */ | ||
| 658 | if (reg_type != REG_TYPE_SRC && reg_type != REG_TYPE_DST) | ||
| 659 | return true; | ||
| 660 | |||
| 661 | width = buf_desc->right_x - buf_desc->left_x; | ||
| 662 | if (width < G2D_LEN_MIN || width > G2D_LEN_MAX) { | ||
| 663 | DRM_ERROR("width[%u] is out of range!\n", width); | ||
| 664 | return false; | ||
| 665 | } | ||
| 666 | |||
| 667 | height = buf_desc->bottom_y - buf_desc->top_y; | ||
| 668 | if (height < G2D_LEN_MIN || height > G2D_LEN_MAX) { | ||
| 669 | DRM_ERROR("height[%u] is out of range!\n", height); | ||
| 670 | return false; | ||
| 671 | } | ||
| 672 | |||
| 673 | area = (unsigned long)width * (unsigned long)height * | ||
| 674 | g2d_get_buf_bpp(buf_desc->format); | ||
| 675 | if (area > size) { | ||
| 676 | DRM_ERROR("area[%lu] is out of range[%lu]!\n", area, size); | ||
| 677 | return false; | ||
| 678 | } | ||
| 679 | |||
| 680 | return true; | ||
| 681 | } | ||
| 682 | |||
| 509 | static int g2d_map_cmdlist_gem(struct g2d_data *g2d, | 683 | static int g2d_map_cmdlist_gem(struct g2d_data *g2d, |
| 510 | struct g2d_cmdlist_node *node, | 684 | struct g2d_cmdlist_node *node, |
| 511 | struct drm_device *drm_dev, | 685 | struct drm_device *drm_dev, |
| 512 | struct drm_file *file) | 686 | struct drm_file *file) |
| 513 | { | 687 | { |
| 514 | struct g2d_cmdlist *cmdlist = node->cmdlist; | 688 | struct g2d_cmdlist *cmdlist = node->cmdlist; |
| 689 | struct g2d_buf_info *buf_info = &node->buf_info; | ||
| 515 | int offset; | 690 | int offset; |
| 691 | int ret; | ||
| 516 | int i; | 692 | int i; |
| 517 | 693 | ||
| 518 | for (i = 0; i < node->map_nr; i++) { | 694 | for (i = 0; i < buf_info->map_nr; i++) { |
| 695 | struct g2d_buf_desc *buf_desc; | ||
| 696 | enum g2d_reg_type reg_type; | ||
| 697 | int reg_pos; | ||
| 519 | unsigned long handle; | 698 | unsigned long handle; |
| 520 | dma_addr_t *addr; | 699 | dma_addr_t *addr; |
| 521 | 700 | ||
| 522 | offset = cmdlist->last - (i * 2 + 1); | 701 | reg_pos = cmdlist->last - 2 * (i + 1); |
| 523 | handle = cmdlist->data[offset]; | 702 | |
| 703 | offset = cmdlist->data[reg_pos]; | ||
| 704 | handle = cmdlist->data[reg_pos + 1]; | ||
| 705 | |||
| 706 | reg_type = g2d_get_reg_type(offset); | ||
| 707 | if (reg_type == REG_TYPE_NONE) { | ||
| 708 | ret = -EFAULT; | ||
| 709 | goto err; | ||
| 710 | } | ||
| 711 | |||
| 712 | buf_desc = &buf_info->descs[reg_type]; | ||
| 713 | |||
| 714 | if (buf_info->types[reg_type] == BUF_TYPE_GEM) { | ||
| 715 | unsigned long size; | ||
| 716 | |||
| 717 | size = exynos_drm_gem_get_size(drm_dev, handle, file); | ||
| 718 | if (!size) { | ||
| 719 | ret = -EFAULT; | ||
| 720 | goto err; | ||
| 721 | } | ||
| 722 | |||
| 723 | if (!g2d_check_buf_desc_is_valid(buf_desc, reg_type, | ||
| 724 | size)) { | ||
| 725 | ret = -EFAULT; | ||
| 726 | goto err; | ||
| 727 | } | ||
| 524 | 728 | ||
| 525 | if (node->obj_type[i] == BUF_TYPE_GEM) { | ||
| 526 | addr = exynos_drm_gem_get_dma_addr(drm_dev, handle, | 729 | addr = exynos_drm_gem_get_dma_addr(drm_dev, handle, |
| 527 | file); | 730 | file); |
| 528 | if (IS_ERR(addr)) { | 731 | if (IS_ERR(addr)) { |
| 529 | node->map_nr = i; | 732 | ret = -EFAULT; |
| 530 | return -EFAULT; | 733 | goto err; |
| 531 | } | 734 | } |
| 532 | } else { | 735 | } else { |
| 533 | struct drm_exynos_g2d_userptr g2d_userptr; | 736 | struct drm_exynos_g2d_userptr g2d_userptr; |
| 534 | 737 | ||
| 535 | if (copy_from_user(&g2d_userptr, (void __user *)handle, | 738 | if (copy_from_user(&g2d_userptr, (void __user *)handle, |
| 536 | sizeof(struct drm_exynos_g2d_userptr))) { | 739 | sizeof(struct drm_exynos_g2d_userptr))) { |
| 537 | node->map_nr = i; | 740 | ret = -EFAULT; |
| 538 | return -EFAULT; | 741 | goto err; |
| 742 | } | ||
| 743 | |||
| 744 | if (!g2d_check_buf_desc_is_valid(buf_desc, reg_type, | ||
| 745 | g2d_userptr.size)) { | ||
| 746 | ret = -EFAULT; | ||
| 747 | goto err; | ||
| 539 | } | 748 | } |
| 540 | 749 | ||
| 541 | addr = g2d_userptr_get_dma_addr(drm_dev, | 750 | addr = g2d_userptr_get_dma_addr(drm_dev, |
| @@ -544,16 +753,21 @@ static int g2d_map_cmdlist_gem(struct g2d_data *g2d, | |||
| 544 | file, | 753 | file, |
| 545 | &handle); | 754 | &handle); |
| 546 | if (IS_ERR(addr)) { | 755 | if (IS_ERR(addr)) { |
| 547 | node->map_nr = i; | 756 | ret = -EFAULT; |
| 548 | return -EFAULT; | 757 | goto err; |
| 549 | } | 758 | } |
| 550 | } | 759 | } |
| 551 | 760 | ||
| 552 | cmdlist->data[offset] = *addr; | 761 | cmdlist->data[reg_pos + 1] = *addr; |
| 553 | node->handles[i] = handle; | 762 | buf_info->reg_types[i] = reg_type; |
| 763 | buf_info->handles[reg_type] = handle; | ||
| 554 | } | 764 | } |
| 555 | 765 | ||
| 556 | return 0; | 766 | return 0; |
| 767 | |||
| 768 | err: | ||
| 769 | buf_info->map_nr = i; | ||
| 770 | return ret; | ||
| 557 | } | 771 | } |
| 558 | 772 | ||
| 559 | static void g2d_unmap_cmdlist_gem(struct g2d_data *g2d, | 773 | static void g2d_unmap_cmdlist_gem(struct g2d_data *g2d, |
| @@ -561,22 +775,33 @@ static void g2d_unmap_cmdlist_gem(struct g2d_data *g2d, | |||
| 561 | struct drm_file *filp) | 775 | struct drm_file *filp) |
| 562 | { | 776 | { |
| 563 | struct exynos_drm_subdrv *subdrv = &g2d->subdrv; | 777 | struct exynos_drm_subdrv *subdrv = &g2d->subdrv; |
| 778 | struct g2d_buf_info *buf_info = &node->buf_info; | ||
| 564 | int i; | 779 | int i; |
| 565 | 780 | ||
| 566 | for (i = 0; i < node->map_nr; i++) { | 781 | for (i = 0; i < buf_info->map_nr; i++) { |
| 567 | unsigned long handle = node->handles[i]; | 782 | struct g2d_buf_desc *buf_desc; |
| 783 | enum g2d_reg_type reg_type; | ||
| 784 | unsigned long handle; | ||
| 785 | |||
| 786 | reg_type = buf_info->reg_types[i]; | ||
| 787 | |||
| 788 | buf_desc = &buf_info->descs[reg_type]; | ||
| 789 | handle = buf_info->handles[reg_type]; | ||
| 568 | 790 | ||
| 569 | if (node->obj_type[i] == BUF_TYPE_GEM) | 791 | if (buf_info->types[reg_type] == BUF_TYPE_GEM) |
| 570 | exynos_drm_gem_put_dma_addr(subdrv->drm_dev, handle, | 792 | exynos_drm_gem_put_dma_addr(subdrv->drm_dev, handle, |
| 571 | filp); | 793 | filp); |
| 572 | else | 794 | else |
| 573 | g2d_userptr_put_dma_addr(subdrv->drm_dev, handle, | 795 | g2d_userptr_put_dma_addr(subdrv->drm_dev, handle, |
| 574 | false); | 796 | false); |
| 575 | 797 | ||
| 576 | node->handles[i] = 0; | 798 | buf_info->reg_types[i] = REG_TYPE_NONE; |
| 799 | buf_info->handles[reg_type] = 0; | ||
| 800 | buf_info->types[reg_type] = 0; | ||
| 801 | memset(buf_desc, 0x00, sizeof(*buf_desc)); | ||
| 577 | } | 802 | } |
| 578 | 803 | ||
| 579 | node->map_nr = 0; | 804 | buf_info->map_nr = 0; |
| 580 | } | 805 | } |
| 581 | 806 | ||
| 582 | static void g2d_dma_start(struct g2d_data *g2d, | 807 | static void g2d_dma_start(struct g2d_data *g2d, |
| @@ -589,10 +814,6 @@ static void g2d_dma_start(struct g2d_data *g2d, | |||
| 589 | pm_runtime_get_sync(g2d->dev); | 814 | pm_runtime_get_sync(g2d->dev); |
| 590 | clk_enable(g2d->gate_clk); | 815 | clk_enable(g2d->gate_clk); |
| 591 | 816 | ||
| 592 | /* interrupt enable */ | ||
| 593 | writel_relaxed(G2D_INTEN_ACF | G2D_INTEN_UCF | G2D_INTEN_GCF, | ||
| 594 | g2d->regs + G2D_INTEN); | ||
| 595 | |||
| 596 | writel_relaxed(node->dma_addr, g2d->regs + G2D_DMA_SFR_BASE_ADDR); | 817 | writel_relaxed(node->dma_addr, g2d->regs + G2D_DMA_SFR_BASE_ADDR); |
| 597 | writel_relaxed(G2D_DMA_START, g2d->regs + G2D_DMA_COMMAND); | 818 | writel_relaxed(G2D_DMA_START, g2d->regs + G2D_DMA_COMMAND); |
| 598 | } | 819 | } |
| @@ -643,7 +864,6 @@ static void g2d_runqueue_worker(struct work_struct *work) | |||
| 643 | struct g2d_data *g2d = container_of(work, struct g2d_data, | 864 | struct g2d_data *g2d = container_of(work, struct g2d_data, |
| 644 | runqueue_work); | 865 | runqueue_work); |
| 645 | 866 | ||
| 646 | |||
| 647 | mutex_lock(&g2d->runqueue_mutex); | 867 | mutex_lock(&g2d->runqueue_mutex); |
| 648 | clk_disable(g2d->gate_clk); | 868 | clk_disable(g2d->gate_clk); |
| 649 | pm_runtime_put_sync(g2d->dev); | 869 | pm_runtime_put_sync(g2d->dev); |
| @@ -724,20 +944,14 @@ static int g2d_check_reg_offset(struct device *dev, | |||
| 724 | int i; | 944 | int i; |
| 725 | 945 | ||
| 726 | for (i = 0; i < nr; i++) { | 946 | for (i = 0; i < nr; i++) { |
| 727 | index = cmdlist->last - 2 * (i + 1); | 947 | struct g2d_buf_info *buf_info = &node->buf_info; |
| 948 | struct g2d_buf_desc *buf_desc; | ||
| 949 | enum g2d_reg_type reg_type; | ||
| 950 | unsigned long value; | ||
| 728 | 951 | ||
| 729 | if (for_addr) { | 952 | index = cmdlist->last - 2 * (i + 1); |
| 730 | /* check userptr buffer type. */ | ||
| 731 | reg_offset = (cmdlist->data[index] & | ||
| 732 | ~0x7fffffff) >> 31; | ||
| 733 | if (reg_offset) { | ||
| 734 | node->obj_type[i] = BUF_TYPE_USERPTR; | ||
| 735 | cmdlist->data[index] &= ~G2D_BUF_USERPTR; | ||
| 736 | } | ||
| 737 | } | ||
| 738 | 953 | ||
| 739 | reg_offset = cmdlist->data[index] & ~0xfffff000; | 954 | reg_offset = cmdlist->data[index] & ~0xfffff000; |
| 740 | |||
| 741 | if (reg_offset < G2D_VALID_START || reg_offset > G2D_VALID_END) | 955 | if (reg_offset < G2D_VALID_START || reg_offset > G2D_VALID_END) |
| 742 | goto err; | 956 | goto err; |
| 743 | if (reg_offset % 4) | 957 | if (reg_offset % 4) |
| @@ -753,8 +967,60 @@ static int g2d_check_reg_offset(struct device *dev, | |||
| 753 | if (!for_addr) | 967 | if (!for_addr) |
| 754 | goto err; | 968 | goto err; |
| 755 | 969 | ||
| 756 | if (node->obj_type[i] != BUF_TYPE_USERPTR) | 970 | reg_type = g2d_get_reg_type(reg_offset); |
| 757 | node->obj_type[i] = BUF_TYPE_GEM; | 971 | if (reg_type == REG_TYPE_NONE) |
| 972 | goto err; | ||
| 973 | |||
| 974 | /* check userptr buffer type. */ | ||
| 975 | if ((cmdlist->data[index] & ~0x7fffffff) >> 31) { | ||
| 976 | buf_info->types[reg_type] = BUF_TYPE_USERPTR; | ||
| 977 | cmdlist->data[index] &= ~G2D_BUF_USERPTR; | ||
| 978 | } else | ||
| 979 | buf_info->types[reg_type] = BUF_TYPE_GEM; | ||
| 980 | break; | ||
| 981 | case G2D_SRC_COLOR_MODE: | ||
| 982 | case G2D_DST_COLOR_MODE: | ||
| 983 | if (for_addr) | ||
| 984 | goto err; | ||
| 985 | |||
| 986 | reg_type = g2d_get_reg_type(reg_offset); | ||
| 987 | if (reg_type == REG_TYPE_NONE) | ||
| 988 | goto err; | ||
| 989 | |||
| 990 | buf_desc = &buf_info->descs[reg_type]; | ||
| 991 | value = cmdlist->data[index + 1]; | ||
| 992 | |||
| 993 | buf_desc->format = value & 0xf; | ||
| 994 | break; | ||
| 995 | case G2D_SRC_LEFT_TOP: | ||
| 996 | case G2D_DST_LEFT_TOP: | ||
| 997 | if (for_addr) | ||
| 998 | goto err; | ||
| 999 | |||
| 1000 | reg_type = g2d_get_reg_type(reg_offset); | ||
| 1001 | if (reg_type == REG_TYPE_NONE) | ||
| 1002 | goto err; | ||
| 1003 | |||
| 1004 | buf_desc = &buf_info->descs[reg_type]; | ||
| 1005 | value = cmdlist->data[index + 1]; | ||
| 1006 | |||
| 1007 | buf_desc->left_x = value & 0x1fff; | ||
| 1008 | buf_desc->top_y = (value & 0x1fff0000) >> 16; | ||
| 1009 | break; | ||
| 1010 | case G2D_SRC_RIGHT_BOTTOM: | ||
| 1011 | case G2D_DST_RIGHT_BOTTOM: | ||
| 1012 | if (for_addr) | ||
| 1013 | goto err; | ||
| 1014 | |||
| 1015 | reg_type = g2d_get_reg_type(reg_offset); | ||
| 1016 | if (reg_type == REG_TYPE_NONE) | ||
| 1017 | goto err; | ||
| 1018 | |||
| 1019 | buf_desc = &buf_info->descs[reg_type]; | ||
| 1020 | value = cmdlist->data[index + 1]; | ||
| 1021 | |||
| 1022 | buf_desc->right_x = value & 0x1fff; | ||
| 1023 | buf_desc->bottom_y = (value & 0x1fff0000) >> 16; | ||
| 758 | break; | 1024 | break; |
| 759 | default: | 1025 | default: |
| 760 | if (for_addr) | 1026 | if (for_addr) |
| @@ -860,9 +1126,23 @@ int exynos_g2d_set_cmdlist_ioctl(struct drm_device *drm_dev, void *data, | |||
| 860 | cmdlist->data[cmdlist->last++] = G2D_SRC_BASE_ADDR; | 1126 | cmdlist->data[cmdlist->last++] = G2D_SRC_BASE_ADDR; |
| 861 | cmdlist->data[cmdlist->last++] = 0; | 1127 | cmdlist->data[cmdlist->last++] = 0; |
| 862 | 1128 | ||
| 1129 | /* | ||
| 1130 | * 'LIST_HOLD' command should be set to the DMA_HOLD_CMD_REG | ||
| 1131 | * and GCF bit should be set to INTEN register if user wants | ||
| 1132 | * G2D interrupt event once current command list execution is | ||
| 1133 | * finished. | ||
| 1134 | * Otherwise only ACF bit should be set to INTEN register so | ||
| 1135 | * that one interrupt is occured after all command lists | ||
| 1136 | * have been completed. | ||
| 1137 | */ | ||
| 863 | if (node->event) { | 1138 | if (node->event) { |
| 1139 | cmdlist->data[cmdlist->last++] = G2D_INTEN; | ||
| 1140 | cmdlist->data[cmdlist->last++] = G2D_INTEN_ACF | G2D_INTEN_GCF; | ||
| 864 | cmdlist->data[cmdlist->last++] = G2D_DMA_HOLD_CMD; | 1141 | cmdlist->data[cmdlist->last++] = G2D_DMA_HOLD_CMD; |
| 865 | cmdlist->data[cmdlist->last++] = G2D_LIST_HOLD; | 1142 | cmdlist->data[cmdlist->last++] = G2D_LIST_HOLD; |
| 1143 | } else { | ||
| 1144 | cmdlist->data[cmdlist->last++] = G2D_INTEN; | ||
| 1145 | cmdlist->data[cmdlist->last++] = G2D_INTEN_ACF; | ||
| 866 | } | 1146 | } |
| 867 | 1147 | ||
| 868 | /* Check size of cmdlist: last 2 is about G2D_BITBLT_START */ | 1148 | /* Check size of cmdlist: last 2 is about G2D_BITBLT_START */ |
| @@ -887,7 +1167,7 @@ int exynos_g2d_set_cmdlist_ioctl(struct drm_device *drm_dev, void *data, | |||
| 887 | if (ret < 0) | 1167 | if (ret < 0) |
| 888 | goto err_free_event; | 1168 | goto err_free_event; |
| 889 | 1169 | ||
| 890 | node->map_nr = req->cmd_buf_nr; | 1170 | node->buf_info.map_nr = req->cmd_buf_nr; |
| 891 | if (req->cmd_buf_nr) { | 1171 | if (req->cmd_buf_nr) { |
| 892 | struct drm_exynos_g2d_cmd *cmd_buf; | 1172 | struct drm_exynos_g2d_cmd *cmd_buf; |
| 893 | 1173 | ||
diff --git a/drivers/gpu/drm/exynos/exynos_drm_gem.c b/drivers/gpu/drm/exynos/exynos_drm_gem.c index 67e17ce112b6..0e6fe000578c 100644 --- a/drivers/gpu/drm/exynos/exynos_drm_gem.c +++ b/drivers/gpu/drm/exynos/exynos_drm_gem.c | |||
| @@ -164,6 +164,27 @@ out: | |||
| 164 | exynos_gem_obj = NULL; | 164 | exynos_gem_obj = NULL; |
| 165 | } | 165 | } |
| 166 | 166 | ||
| 167 | unsigned long exynos_drm_gem_get_size(struct drm_device *dev, | ||
| 168 | unsigned int gem_handle, | ||
| 169 | struct drm_file *file_priv) | ||
| 170 | { | ||
| 171 | struct exynos_drm_gem_obj *exynos_gem_obj; | ||
| 172 | struct drm_gem_object *obj; | ||
| 173 | |||
| 174 | obj = drm_gem_object_lookup(dev, file_priv, gem_handle); | ||
| 175 | if (!obj) { | ||
| 176 | DRM_ERROR("failed to lookup gem object.\n"); | ||
| 177 | return 0; | ||
| 178 | } | ||
| 179 | |||
| 180 | exynos_gem_obj = to_exynos_gem_obj(obj); | ||
| 181 | |||
| 182 | drm_gem_object_unreference_unlocked(obj); | ||
| 183 | |||
| 184 | return exynos_gem_obj->buffer->size; | ||
| 185 | } | ||
| 186 | |||
| 187 | |||
| 167 | struct exynos_drm_gem_obj *exynos_drm_gem_init(struct drm_device *dev, | 188 | struct exynos_drm_gem_obj *exynos_drm_gem_init(struct drm_device *dev, |
| 168 | unsigned long size) | 189 | unsigned long size) |
| 169 | { | 190 | { |
diff --git a/drivers/gpu/drm/exynos/exynos_drm_gem.h b/drivers/gpu/drm/exynos/exynos_drm_gem.h index 35ebac47dc2b..468766bee450 100644 --- a/drivers/gpu/drm/exynos/exynos_drm_gem.h +++ b/drivers/gpu/drm/exynos/exynos_drm_gem.h | |||
| @@ -130,6 +130,11 @@ int exynos_drm_gem_userptr_ioctl(struct drm_device *dev, void *data, | |||
| 130 | int exynos_drm_gem_get_ioctl(struct drm_device *dev, void *data, | 130 | int exynos_drm_gem_get_ioctl(struct drm_device *dev, void *data, |
| 131 | struct drm_file *file_priv); | 131 | struct drm_file *file_priv); |
| 132 | 132 | ||
| 133 | /* get buffer size to gem handle. */ | ||
| 134 | unsigned long exynos_drm_gem_get_size(struct drm_device *dev, | ||
| 135 | unsigned int gem_handle, | ||
| 136 | struct drm_file *file_priv); | ||
| 137 | |||
| 133 | /* initialize gem object. */ | 138 | /* initialize gem object. */ |
| 134 | int exynos_drm_gem_init_object(struct drm_gem_object *obj); | 139 | int exynos_drm_gem_init_object(struct drm_gem_object *obj); |
| 135 | 140 | ||
diff --git a/drivers/gpu/drm/exynos/exynos_drm_vidi.c b/drivers/gpu/drm/exynos/exynos_drm_vidi.c index 13ccbd4bcfaa..9504b0cd825a 100644 --- a/drivers/gpu/drm/exynos/exynos_drm_vidi.c +++ b/drivers/gpu/drm/exynos/exynos_drm_vidi.c | |||
| @@ -117,13 +117,12 @@ static struct edid *vidi_get_edid(struct device *dev, | |||
| 117 | } | 117 | } |
| 118 | 118 | ||
| 119 | edid_len = (1 + ctx->raw_edid->extensions) * EDID_LENGTH; | 119 | edid_len = (1 + ctx->raw_edid->extensions) * EDID_LENGTH; |
| 120 | edid = kzalloc(edid_len, GFP_KERNEL); | 120 | edid = kmemdup(ctx->raw_edid, edid_len, GFP_KERNEL); |
| 121 | if (!edid) { | 121 | if (!edid) { |
| 122 | DRM_DEBUG_KMS("failed to allocate edid\n"); | 122 | DRM_DEBUG_KMS("failed to allocate edid\n"); |
| 123 | return ERR_PTR(-ENOMEM); | 123 | return ERR_PTR(-ENOMEM); |
| 124 | } | 124 | } |
| 125 | 125 | ||
| 126 | memcpy(edid, ctx->raw_edid, edid_len); | ||
| 127 | return edid; | 126 | return edid; |
| 128 | } | 127 | } |
| 129 | 128 | ||
| @@ -563,12 +562,11 @@ int vidi_connection_ioctl(struct drm_device *drm_dev, void *data, | |||
| 563 | return -EINVAL; | 562 | return -EINVAL; |
| 564 | } | 563 | } |
| 565 | edid_len = (1 + raw_edid->extensions) * EDID_LENGTH; | 564 | edid_len = (1 + raw_edid->extensions) * EDID_LENGTH; |
| 566 | ctx->raw_edid = kzalloc(edid_len, GFP_KERNEL); | 565 | ctx->raw_edid = kmemdup(raw_edid, edid_len, GFP_KERNEL); |
| 567 | if (!ctx->raw_edid) { | 566 | if (!ctx->raw_edid) { |
| 568 | DRM_DEBUG_KMS("failed to allocate raw_edid.\n"); | 567 | DRM_DEBUG_KMS("failed to allocate raw_edid.\n"); |
| 569 | return -ENOMEM; | 568 | return -ENOMEM; |
| 570 | } | 569 | } |
| 571 | memcpy(ctx->raw_edid, raw_edid, edid_len); | ||
| 572 | } else { | 570 | } else { |
| 573 | /* | 571 | /* |
| 574 | * with connection = 0, free raw_edid | 572 | * with connection = 0, free raw_edid |
diff --git a/drivers/gpu/drm/exynos/exynos_mixer.c b/drivers/gpu/drm/exynos/exynos_mixer.c index e919aba29b3d..2f4f72f07047 100644 --- a/drivers/gpu/drm/exynos/exynos_mixer.c +++ b/drivers/gpu/drm/exynos/exynos_mixer.c | |||
| @@ -818,7 +818,7 @@ static void mixer_win_disable(void *ctx, int win) | |||
| 818 | mixer_ctx->win_data[win].enabled = false; | 818 | mixer_ctx->win_data[win].enabled = false; |
| 819 | } | 819 | } |
| 820 | 820 | ||
| 821 | int mixer_check_timing(void *ctx, struct fb_videomode *timing) | 821 | static int mixer_check_timing(void *ctx, struct fb_videomode *timing) |
| 822 | { | 822 | { |
| 823 | struct mixer_context *mixer_ctx = ctx; | 823 | struct mixer_context *mixer_ctx = ctx; |
| 824 | u32 w, h; | 824 | u32 w, h; |
diff --git a/drivers/gpu/drm/i915/i915_drv.c b/drivers/gpu/drm/i915/i915_drv.c index 0a8eceb75902..e9b57893db2b 100644 --- a/drivers/gpu/drm/i915/i915_drv.c +++ b/drivers/gpu/drm/i915/i915_drv.c | |||
| @@ -125,6 +125,11 @@ MODULE_PARM_DESC(preliminary_hw_support, | |||
| 125 | "Enable Haswell and ValleyView Support. " | 125 | "Enable Haswell and ValleyView Support. " |
| 126 | "(default: false)"); | 126 | "(default: false)"); |
| 127 | 127 | ||
| 128 | int i915_disable_power_well __read_mostly = 0; | ||
| 129 | module_param_named(disable_power_well, i915_disable_power_well, int, 0600); | ||
| 130 | MODULE_PARM_DESC(disable_power_well, | ||
| 131 | "Disable the power well when possible (default: false)"); | ||
| 132 | |||
| 128 | static struct drm_driver driver; | 133 | static struct drm_driver driver; |
| 129 | extern int intel_agp_enabled; | 134 | extern int intel_agp_enabled; |
| 130 | 135 | ||
diff --git a/drivers/gpu/drm/i915/i915_drv.h b/drivers/gpu/drm/i915/i915_drv.h index e95337c97459..01769e2a9953 100644 --- a/drivers/gpu/drm/i915/i915_drv.h +++ b/drivers/gpu/drm/i915/i915_drv.h | |||
| @@ -1398,6 +1398,7 @@ extern int i915_enable_fbc __read_mostly; | |||
| 1398 | extern bool i915_enable_hangcheck __read_mostly; | 1398 | extern bool i915_enable_hangcheck __read_mostly; |
| 1399 | extern int i915_enable_ppgtt __read_mostly; | 1399 | extern int i915_enable_ppgtt __read_mostly; |
| 1400 | extern unsigned int i915_preliminary_hw_support __read_mostly; | 1400 | extern unsigned int i915_preliminary_hw_support __read_mostly; |
| 1401 | extern int i915_disable_power_well __read_mostly; | ||
| 1401 | 1402 | ||
| 1402 | extern int i915_suspend(struct drm_device *dev, pm_message_t state); | 1403 | extern int i915_suspend(struct drm_device *dev, pm_message_t state); |
| 1403 | extern int i915_resume(struct drm_device *dev); | 1404 | extern int i915_resume(struct drm_device *dev); |
diff --git a/drivers/gpu/drm/i915/intel_display.c b/drivers/gpu/drm/i915/intel_display.c index 287b42c9d1a8..b20d50192fcc 100644 --- a/drivers/gpu/drm/i915/intel_display.c +++ b/drivers/gpu/drm/i915/intel_display.c | |||
| @@ -5771,6 +5771,11 @@ static int haswell_crtc_mode_set(struct drm_crtc *crtc, | |||
| 5771 | num_connectors++; | 5771 | num_connectors++; |
| 5772 | } | 5772 | } |
| 5773 | 5773 | ||
| 5774 | if (is_cpu_edp) | ||
| 5775 | intel_crtc->cpu_transcoder = TRANSCODER_EDP; | ||
| 5776 | else | ||
| 5777 | intel_crtc->cpu_transcoder = pipe; | ||
| 5778 | |||
| 5774 | /* We are not sure yet this won't happen. */ | 5779 | /* We are not sure yet this won't happen. */ |
| 5775 | WARN(!HAS_PCH_LPT(dev), "Unexpected PCH type %d\n", | 5780 | WARN(!HAS_PCH_LPT(dev), "Unexpected PCH type %d\n", |
| 5776 | INTEL_PCH_TYPE(dev)); | 5781 | INTEL_PCH_TYPE(dev)); |
| @@ -5837,11 +5842,6 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 5837 | int pipe = intel_crtc->pipe; | 5842 | int pipe = intel_crtc->pipe; |
| 5838 | int ret; | 5843 | int ret; |
| 5839 | 5844 | ||
| 5840 | if (IS_HASWELL(dev) && intel_pipe_has_type(crtc, INTEL_OUTPUT_EDP)) | ||
| 5841 | intel_crtc->cpu_transcoder = TRANSCODER_EDP; | ||
| 5842 | else | ||
| 5843 | intel_crtc->cpu_transcoder = pipe; | ||
| 5844 | |||
| 5845 | drm_vblank_pre_modeset(dev, pipe); | 5845 | drm_vblank_pre_modeset(dev, pipe); |
| 5846 | 5846 | ||
| 5847 | ret = dev_priv->display.crtc_mode_set(crtc, mode, adjusted_mode, | 5847 | ret = dev_priv->display.crtc_mode_set(crtc, mode, adjusted_mode, |
diff --git a/drivers/gpu/drm/i915/intel_panel.c b/drivers/gpu/drm/i915/intel_panel.c index a3730e0289e5..bee8cb6108a7 100644 --- a/drivers/gpu/drm/i915/intel_panel.c +++ b/drivers/gpu/drm/i915/intel_panel.c | |||
| @@ -321,9 +321,6 @@ void intel_panel_enable_backlight(struct drm_device *dev, | |||
| 321 | if (dev_priv->backlight_level == 0) | 321 | if (dev_priv->backlight_level == 0) |
| 322 | dev_priv->backlight_level = intel_panel_get_max_backlight(dev); | 322 | dev_priv->backlight_level = intel_panel_get_max_backlight(dev); |
| 323 | 323 | ||
| 324 | dev_priv->backlight_enabled = true; | ||
| 325 | intel_panel_actually_set_backlight(dev, dev_priv->backlight_level); | ||
| 326 | |||
| 327 | if (INTEL_INFO(dev)->gen >= 4) { | 324 | if (INTEL_INFO(dev)->gen >= 4) { |
| 328 | uint32_t reg, tmp; | 325 | uint32_t reg, tmp; |
| 329 | 326 | ||
| @@ -359,12 +356,12 @@ void intel_panel_enable_backlight(struct drm_device *dev, | |||
| 359 | } | 356 | } |
| 360 | 357 | ||
| 361 | set_level: | 358 | set_level: |
| 362 | /* Check the current backlight level and try to set again if it's zero. | 359 | /* Call below after setting BLC_PWM_CPU_CTL2 and BLC_PWM_PCH_CTL1. |
| 363 | * On some machines, BLC_PWM_CPU_CTL is cleared to zero automatically | 360 | * BLC_PWM_CPU_CTL may be cleared to zero automatically when these |
| 364 | * when BLC_PWM_CPU_CTL2 and BLC_PWM_PCH_CTL1 are written. | 361 | * registers are set. |
| 365 | */ | 362 | */ |
| 366 | if (!intel_panel_get_backlight(dev)) | 363 | dev_priv->backlight_enabled = true; |
| 367 | intel_panel_actually_set_backlight(dev, dev_priv->backlight_level); | 364 | intel_panel_actually_set_backlight(dev, dev_priv->backlight_level); |
| 368 | } | 365 | } |
| 369 | 366 | ||
| 370 | static void intel_panel_init_backlight(struct drm_device *dev) | 367 | static void intel_panel_init_backlight(struct drm_device *dev) |
diff --git a/drivers/gpu/drm/i915/intel_pm.c b/drivers/gpu/drm/i915/intel_pm.c index a1794c6df1bf..adca00783e61 100644 --- a/drivers/gpu/drm/i915/intel_pm.c +++ b/drivers/gpu/drm/i915/intel_pm.c | |||
| @@ -4079,6 +4079,9 @@ void intel_set_power_well(struct drm_device *dev, bool enable) | |||
| 4079 | if (!IS_HASWELL(dev)) | 4079 | if (!IS_HASWELL(dev)) |
| 4080 | return; | 4080 | return; |
| 4081 | 4081 | ||
| 4082 | if (!i915_disable_power_well && !enable) | ||
| 4083 | return; | ||
| 4084 | |||
| 4082 | tmp = I915_READ(HSW_PWR_WELL_DRIVER); | 4085 | tmp = I915_READ(HSW_PWR_WELL_DRIVER); |
| 4083 | is_enabled = tmp & HSW_PWR_WELL_STATE; | 4086 | is_enabled = tmp & HSW_PWR_WELL_STATE; |
| 4084 | enable_requested = tmp & HSW_PWR_WELL_ENABLE; | 4087 | enable_requested = tmp & HSW_PWR_WELL_ENABLE; |
diff --git a/drivers/hid/hid-ids.h b/drivers/hid/hid-ids.h index 92e47e5c9564..c4388776f4e4 100644 --- a/drivers/hid/hid-ids.h +++ b/drivers/hid/hid-ids.h | |||
| @@ -590,6 +590,9 @@ | |||
| 590 | #define USB_VENDOR_ID_MONTEREY 0x0566 | 590 | #define USB_VENDOR_ID_MONTEREY 0x0566 |
| 591 | #define USB_DEVICE_ID_GENIUS_KB29E 0x3004 | 591 | #define USB_DEVICE_ID_GENIUS_KB29E 0x3004 |
| 592 | 592 | ||
| 593 | #define USB_VENDOR_ID_MSI 0x1770 | ||
| 594 | #define USB_DEVICE_ID_MSI_GX680R_LED_PANEL 0xff00 | ||
| 595 | |||
| 593 | #define USB_VENDOR_ID_NATIONAL_SEMICONDUCTOR 0x0400 | 596 | #define USB_VENDOR_ID_NATIONAL_SEMICONDUCTOR 0x0400 |
| 594 | #define USB_DEVICE_ID_N_S_HARMONY 0xc359 | 597 | #define USB_DEVICE_ID_N_S_HARMONY 0xc359 |
| 595 | 598 | ||
| @@ -684,6 +687,9 @@ | |||
| 684 | #define USB_DEVICE_ID_QUANTA_OPTICAL_TOUCH_3001 0x3001 | 687 | #define USB_DEVICE_ID_QUANTA_OPTICAL_TOUCH_3001 0x3001 |
| 685 | #define USB_DEVICE_ID_QUANTA_OPTICAL_TOUCH_3008 0x3008 | 688 | #define USB_DEVICE_ID_QUANTA_OPTICAL_TOUCH_3008 0x3008 |
| 686 | 689 | ||
| 690 | #define USB_VENDOR_ID_REALTEK 0x0bda | ||
| 691 | #define USB_DEVICE_ID_REALTEK_READER 0x0152 | ||
| 692 | |||
| 687 | #define USB_VENDOR_ID_ROCCAT 0x1e7d | 693 | #define USB_VENDOR_ID_ROCCAT 0x1e7d |
| 688 | #define USB_DEVICE_ID_ROCCAT_ARVO 0x30d4 | 694 | #define USB_DEVICE_ID_ROCCAT_ARVO 0x30d4 |
| 689 | #define USB_DEVICE_ID_ROCCAT_ISKU 0x319c | 695 | #define USB_DEVICE_ID_ROCCAT_ISKU 0x319c |
diff --git a/drivers/hid/hid-multitouch.c b/drivers/hid/hid-multitouch.c index 7a1ebb867cf4..82e9211b3ca9 100644 --- a/drivers/hid/hid-multitouch.c +++ b/drivers/hid/hid-multitouch.c | |||
| @@ -621,6 +621,7 @@ static void mt_process_mt_event(struct hid_device *hid, struct hid_field *field, | |||
| 621 | { | 621 | { |
| 622 | struct mt_device *td = hid_get_drvdata(hid); | 622 | struct mt_device *td = hid_get_drvdata(hid); |
| 623 | __s32 quirks = td->mtclass.quirks; | 623 | __s32 quirks = td->mtclass.quirks; |
| 624 | struct input_dev *input = field->hidinput->input; | ||
| 624 | 625 | ||
| 625 | if (hid->claimed & HID_CLAIMED_INPUT) { | 626 | if (hid->claimed & HID_CLAIMED_INPUT) { |
| 626 | switch (usage->hid) { | 627 | switch (usage->hid) { |
| @@ -670,13 +671,16 @@ static void mt_process_mt_event(struct hid_device *hid, struct hid_field *field, | |||
| 670 | break; | 671 | break; |
| 671 | 672 | ||
| 672 | default: | 673 | default: |
| 674 | if (usage->type) | ||
| 675 | input_event(input, usage->type, usage->code, | ||
| 676 | value); | ||
| 673 | return; | 677 | return; |
| 674 | } | 678 | } |
| 675 | 679 | ||
| 676 | if (usage->usage_index + 1 == field->report_count) { | 680 | if (usage->usage_index + 1 == field->report_count) { |
| 677 | /* we only take into account the last report. */ | 681 | /* we only take into account the last report. */ |
| 678 | if (usage->hid == td->last_slot_field) | 682 | if (usage->hid == td->last_slot_field) |
| 679 | mt_complete_slot(td, field->hidinput->input); | 683 | mt_complete_slot(td, input); |
| 680 | 684 | ||
| 681 | if (field->index == td->last_field_index | 685 | if (field->index == td->last_field_index |
| 682 | && td->num_received >= td->num_expected) | 686 | && td->num_received >= td->num_expected) |
diff --git a/drivers/hid/usbhid/hid-quirks.c b/drivers/hid/usbhid/hid-quirks.c index e0e6abf1cd3b..19b8360f2330 100644 --- a/drivers/hid/usbhid/hid-quirks.c +++ b/drivers/hid/usbhid/hid-quirks.c | |||
| @@ -73,6 +73,7 @@ static const struct hid_blacklist { | |||
| 73 | { USB_VENDOR_ID_FORMOSA, USB_DEVICE_ID_FORMOSA_IR_RECEIVER, HID_QUIRK_NO_INIT_REPORTS }, | 73 | { USB_VENDOR_ID_FORMOSA, USB_DEVICE_ID_FORMOSA_IR_RECEIVER, HID_QUIRK_NO_INIT_REPORTS }, |
| 74 | { USB_VENDOR_ID_FREESCALE, USB_DEVICE_ID_FREESCALE_MX28, HID_QUIRK_NOGET }, | 74 | { USB_VENDOR_ID_FREESCALE, USB_DEVICE_ID_FREESCALE_MX28, HID_QUIRK_NOGET }, |
| 75 | { USB_VENDOR_ID_MGE, USB_DEVICE_ID_MGE_UPS, HID_QUIRK_NOGET }, | 75 | { USB_VENDOR_ID_MGE, USB_DEVICE_ID_MGE_UPS, HID_QUIRK_NOGET }, |
| 76 | { USB_VENDOR_ID_MSI, USB_DEVICE_ID_MSI_GX680R_LED_PANEL, HID_QUIRK_NO_INIT_REPORTS }, | ||
| 76 | { USB_VENDOR_ID_NOVATEK, USB_DEVICE_ID_NOVATEK_MOUSE, HID_QUIRK_NO_INIT_REPORTS }, | 77 | { USB_VENDOR_ID_NOVATEK, USB_DEVICE_ID_NOVATEK_MOUSE, HID_QUIRK_NO_INIT_REPORTS }, |
| 77 | { USB_VENDOR_ID_PIXART, USB_DEVICE_ID_PIXART_OPTICAL_TOUCH_SCREEN, HID_QUIRK_NO_INIT_REPORTS }, | 78 | { USB_VENDOR_ID_PIXART, USB_DEVICE_ID_PIXART_OPTICAL_TOUCH_SCREEN, HID_QUIRK_NO_INIT_REPORTS }, |
| 78 | { USB_VENDOR_ID_PIXART, USB_DEVICE_ID_PIXART_OPTICAL_TOUCH_SCREEN1, HID_QUIRK_NO_INIT_REPORTS }, | 79 | { USB_VENDOR_ID_PIXART, USB_DEVICE_ID_PIXART_OPTICAL_TOUCH_SCREEN1, HID_QUIRK_NO_INIT_REPORTS }, |
| @@ -80,6 +81,7 @@ static const struct hid_blacklist { | |||
| 80 | { USB_VENDOR_ID_PRODIGE, USB_DEVICE_ID_PRODIGE_CORDLESS, HID_QUIRK_NOGET }, | 81 | { USB_VENDOR_ID_PRODIGE, USB_DEVICE_ID_PRODIGE_CORDLESS, HID_QUIRK_NOGET }, |
| 81 | { USB_VENDOR_ID_QUANTA, USB_DEVICE_ID_QUANTA_OPTICAL_TOUCH_3001, HID_QUIRK_NOGET }, | 82 | { USB_VENDOR_ID_QUANTA, USB_DEVICE_ID_QUANTA_OPTICAL_TOUCH_3001, HID_QUIRK_NOGET }, |
| 82 | { USB_VENDOR_ID_QUANTA, USB_DEVICE_ID_QUANTA_OPTICAL_TOUCH_3008, HID_QUIRK_NOGET }, | 83 | { USB_VENDOR_ID_QUANTA, USB_DEVICE_ID_QUANTA_OPTICAL_TOUCH_3008, HID_QUIRK_NOGET }, |
| 84 | { USB_VENDOR_ID_REALTEK, USB_DEVICE_ID_REALTEK_READER, HID_QUIRK_NO_INIT_REPORTS }, | ||
| 83 | { USB_VENDOR_ID_SENNHEISER, USB_DEVICE_ID_SENNHEISER_BTD500USB, HID_QUIRK_NOGET }, | 85 | { USB_VENDOR_ID_SENNHEISER, USB_DEVICE_ID_SENNHEISER_BTD500USB, HID_QUIRK_NOGET }, |
| 84 | { USB_VENDOR_ID_SIGMATEL, USB_DEVICE_ID_SIGMATEL_STMP3780, HID_QUIRK_NOGET }, | 86 | { USB_VENDOR_ID_SIGMATEL, USB_DEVICE_ID_SIGMATEL_STMP3780, HID_QUIRK_NOGET }, |
| 85 | { USB_VENDOR_ID_SUN, USB_DEVICE_ID_RARITAN_KVM_DONGLE, HID_QUIRK_NOGET }, | 87 | { USB_VENDOR_ID_SUN, USB_DEVICE_ID_RARITAN_KVM_DONGLE, HID_QUIRK_NOGET }, |
diff --git a/drivers/infiniband/hw/cxgb4/qp.c b/drivers/infiniband/hw/cxgb4/qp.c index 17ba4f8bc12d..70b1808a08f4 100644 --- a/drivers/infiniband/hw/cxgb4/qp.c +++ b/drivers/infiniband/hw/cxgb4/qp.c | |||
| @@ -186,8 +186,10 @@ static int create_qp(struct c4iw_rdev *rdev, struct t4_wq *wq, | |||
| 186 | wq->rq.queue = dma_alloc_coherent(&(rdev->lldi.pdev->dev), | 186 | wq->rq.queue = dma_alloc_coherent(&(rdev->lldi.pdev->dev), |
| 187 | wq->rq.memsize, &(wq->rq.dma_addr), | 187 | wq->rq.memsize, &(wq->rq.dma_addr), |
| 188 | GFP_KERNEL); | 188 | GFP_KERNEL); |
| 189 | if (!wq->rq.queue) | 189 | if (!wq->rq.queue) { |
| 190 | ret = -ENOMEM; | ||
| 190 | goto free_sq; | 191 | goto free_sq; |
| 192 | } | ||
| 191 | PDBG("%s sq base va 0x%p pa 0x%llx rq base va 0x%p pa 0x%llx\n", | 193 | PDBG("%s sq base va 0x%p pa 0x%llx rq base va 0x%p pa 0x%llx\n", |
| 192 | __func__, wq->sq.queue, | 194 | __func__, wq->sq.queue, |
| 193 | (unsigned long long)virt_to_phys(wq->sq.queue), | 195 | (unsigned long long)virt_to_phys(wq->sq.queue), |
diff --git a/drivers/infiniband/hw/ipath/ipath_verbs.c b/drivers/infiniband/hw/ipath/ipath_verbs.c index 439c35d4a669..ea93870266eb 100644 --- a/drivers/infiniband/hw/ipath/ipath_verbs.c +++ b/drivers/infiniband/hw/ipath/ipath_verbs.c | |||
| @@ -620,7 +620,7 @@ void ipath_ib_rcv(struct ipath_ibdev *dev, void *rhdr, void *data, | |||
| 620 | goto bail; | 620 | goto bail; |
| 621 | } | 621 | } |
| 622 | 622 | ||
| 623 | opcode = be32_to_cpu(ohdr->bth[0]) >> 24; | 623 | opcode = (be32_to_cpu(ohdr->bth[0]) >> 24) & 0x7f; |
| 624 | dev->opstats[opcode].n_bytes += tlen; | 624 | dev->opstats[opcode].n_bytes += tlen; |
| 625 | dev->opstats[opcode].n_packets++; | 625 | dev->opstats[opcode].n_packets++; |
| 626 | 626 | ||
diff --git a/drivers/infiniband/hw/qib/Kconfig b/drivers/infiniband/hw/qib/Kconfig index 8349f9c5064c..1e603a375069 100644 --- a/drivers/infiniband/hw/qib/Kconfig +++ b/drivers/infiniband/hw/qib/Kconfig | |||
| @@ -1,7 +1,7 @@ | |||
| 1 | config INFINIBAND_QIB | 1 | config INFINIBAND_QIB |
| 2 | tristate "QLogic PCIe HCA support" | 2 | tristate "Intel PCIe HCA support" |
| 3 | depends on 64BIT | 3 | depends on 64BIT |
| 4 | ---help--- | 4 | ---help--- |
| 5 | This is a low-level driver for QLogic PCIe QLE InfiniBand host | 5 | This is a low-level driver for Intel PCIe QLE InfiniBand host |
| 6 | channel adapters. This driver does not support the QLogic | 6 | channel adapters. This driver does not support the Intel |
| 7 | HyperTransport card (model QHT7140). | 7 | HyperTransport card (model QHT7140). |
diff --git a/drivers/infiniband/hw/qib/qib_driver.c b/drivers/infiniband/hw/qib/qib_driver.c index 5423edcab51f..216092477dfc 100644 --- a/drivers/infiniband/hw/qib/qib_driver.c +++ b/drivers/infiniband/hw/qib/qib_driver.c | |||
| @@ -1,4 +1,5 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * Copyright (c) 2013 Intel Corporation. All rights reserved. | ||
| 2 | * Copyright (c) 2006, 2007, 2008, 2009 QLogic Corporation. All rights reserved. | 3 | * Copyright (c) 2006, 2007, 2008, 2009 QLogic Corporation. All rights reserved. |
| 3 | * Copyright (c) 2003, 2004, 2005, 2006 PathScale, Inc. All rights reserved. | 4 | * Copyright (c) 2003, 2004, 2005, 2006 PathScale, Inc. All rights reserved. |
| 4 | * | 5 | * |
| @@ -63,8 +64,8 @@ MODULE_PARM_DESC(compat_ddr_negotiate, | |||
| 63 | "Attempt pre-IBTA 1.2 DDR speed negotiation"); | 64 | "Attempt pre-IBTA 1.2 DDR speed negotiation"); |
| 64 | 65 | ||
| 65 | MODULE_LICENSE("Dual BSD/GPL"); | 66 | MODULE_LICENSE("Dual BSD/GPL"); |
| 66 | MODULE_AUTHOR("QLogic <support@qlogic.com>"); | 67 | MODULE_AUTHOR("Intel <ibsupport@intel.com>"); |
| 67 | MODULE_DESCRIPTION("QLogic IB driver"); | 68 | MODULE_DESCRIPTION("Intel IB driver"); |
| 68 | MODULE_VERSION(QIB_DRIVER_VERSION); | 69 | MODULE_VERSION(QIB_DRIVER_VERSION); |
| 69 | 70 | ||
| 70 | /* | 71 | /* |
diff --git a/drivers/infiniband/hw/qib/qib_iba6120.c b/drivers/infiniband/hw/qib/qib_iba6120.c index a099ac171e22..0232ae56b1fa 100644 --- a/drivers/infiniband/hw/qib/qib_iba6120.c +++ b/drivers/infiniband/hw/qib/qib_iba6120.c | |||
| @@ -1,4 +1,5 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * Copyright (c) 2013 Intel Corporation. All rights reserved. | ||
| 2 | * Copyright (c) 2006, 2007, 2008, 2009, 2010 QLogic Corporation. | 3 | * Copyright (c) 2006, 2007, 2008, 2009, 2010 QLogic Corporation. |
| 3 | * All rights reserved. | 4 | * All rights reserved. |
| 4 | * Copyright (c) 2003, 2004, 2005, 2006 PathScale, Inc. All rights reserved. | 5 | * Copyright (c) 2003, 2004, 2005, 2006 PathScale, Inc. All rights reserved. |
| @@ -51,7 +52,7 @@ static u32 qib_6120_iblink_state(u64); | |||
| 51 | 52 | ||
| 52 | /* | 53 | /* |
| 53 | * This file contains all the chip-specific register information and | 54 | * This file contains all the chip-specific register information and |
| 54 | * access functions for the QLogic QLogic_IB PCI-Express chip. | 55 | * access functions for the Intel Intel_IB PCI-Express chip. |
| 55 | * | 56 | * |
| 56 | */ | 57 | */ |
| 57 | 58 | ||
diff --git a/drivers/infiniband/hw/qib/qib_init.c b/drivers/infiniband/hw/qib/qib_init.c index 50e33aa0b4e3..173f805790da 100644 --- a/drivers/infiniband/hw/qib/qib_init.c +++ b/drivers/infiniband/hw/qib/qib_init.c | |||
| @@ -1,5 +1,5 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * Copyright (c) 2012 Intel Corporation. All rights reserved. | 2 | * Copyright (c) 2012, 2013 Intel Corporation. All rights reserved. |
| 3 | * Copyright (c) 2006 - 2012 QLogic Corporation. All rights reserved. | 3 | * Copyright (c) 2006 - 2012 QLogic Corporation. All rights reserved. |
| 4 | * Copyright (c) 2003, 2004, 2005, 2006 PathScale, Inc. All rights reserved. | 4 | * Copyright (c) 2003, 2004, 2005, 2006 PathScale, Inc. All rights reserved. |
| 5 | * | 5 | * |
| @@ -1138,7 +1138,7 @@ void qib_disable_after_error(struct qib_devdata *dd) | |||
| 1138 | static void qib_remove_one(struct pci_dev *); | 1138 | static void qib_remove_one(struct pci_dev *); |
| 1139 | static int qib_init_one(struct pci_dev *, const struct pci_device_id *); | 1139 | static int qib_init_one(struct pci_dev *, const struct pci_device_id *); |
| 1140 | 1140 | ||
| 1141 | #define DRIVER_LOAD_MSG "QLogic " QIB_DRV_NAME " loaded: " | 1141 | #define DRIVER_LOAD_MSG "Intel " QIB_DRV_NAME " loaded: " |
| 1142 | #define PFX QIB_DRV_NAME ": " | 1142 | #define PFX QIB_DRV_NAME ": " |
| 1143 | 1143 | ||
| 1144 | static DEFINE_PCI_DEVICE_TABLE(qib_pci_tbl) = { | 1144 | static DEFINE_PCI_DEVICE_TABLE(qib_pci_tbl) = { |
| @@ -1355,7 +1355,7 @@ static int qib_init_one(struct pci_dev *pdev, const struct pci_device_id *ent) | |||
| 1355 | dd = qib_init_iba6120_funcs(pdev, ent); | 1355 | dd = qib_init_iba6120_funcs(pdev, ent); |
| 1356 | #else | 1356 | #else |
| 1357 | qib_early_err(&pdev->dev, | 1357 | qib_early_err(&pdev->dev, |
| 1358 | "QLogic PCIE device 0x%x cannot work if CONFIG_PCI_MSI is not enabled\n", | 1358 | "Intel PCIE device 0x%x cannot work if CONFIG_PCI_MSI is not enabled\n", |
| 1359 | ent->device); | 1359 | ent->device); |
| 1360 | dd = ERR_PTR(-ENODEV); | 1360 | dd = ERR_PTR(-ENODEV); |
| 1361 | #endif | 1361 | #endif |
| @@ -1371,7 +1371,7 @@ static int qib_init_one(struct pci_dev *pdev, const struct pci_device_id *ent) | |||
| 1371 | 1371 | ||
| 1372 | default: | 1372 | default: |
| 1373 | qib_early_err(&pdev->dev, | 1373 | qib_early_err(&pdev->dev, |
| 1374 | "Failing on unknown QLogic deviceid 0x%x\n", | 1374 | "Failing on unknown Intel deviceid 0x%x\n", |
| 1375 | ent->device); | 1375 | ent->device); |
| 1376 | ret = -ENODEV; | 1376 | ret = -ENODEV; |
| 1377 | } | 1377 | } |
diff --git a/drivers/infiniband/hw/qib/qib_sd7220.c b/drivers/infiniband/hw/qib/qib_sd7220.c index 50a8a0d4fe67..08a6c6d39e56 100644 --- a/drivers/infiniband/hw/qib/qib_sd7220.c +++ b/drivers/infiniband/hw/qib/qib_sd7220.c | |||
| @@ -1,5 +1,5 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * Copyright (c) 2012 Intel Corporation. All rights reserved. | 2 | * Copyright (c) 2013 Intel Corporation. All rights reserved. |
| 3 | * Copyright (c) 2006 - 2012 QLogic Corporation. All rights reserved. | 3 | * Copyright (c) 2006 - 2012 QLogic Corporation. All rights reserved. |
| 4 | * Copyright (c) 2003, 2004, 2005, 2006 PathScale, Inc. All rights reserved. | 4 | * Copyright (c) 2003, 2004, 2005, 2006 PathScale, Inc. All rights reserved. |
| 5 | * | 5 | * |
| @@ -44,7 +44,7 @@ | |||
| 44 | #include "qib.h" | 44 | #include "qib.h" |
| 45 | #include "qib_7220.h" | 45 | #include "qib_7220.h" |
| 46 | 46 | ||
| 47 | #define SD7220_FW_NAME "qlogic/sd7220.fw" | 47 | #define SD7220_FW_NAME "intel/sd7220.fw" |
| 48 | MODULE_FIRMWARE(SD7220_FW_NAME); | 48 | MODULE_FIRMWARE(SD7220_FW_NAME); |
| 49 | 49 | ||
| 50 | /* | 50 | /* |
diff --git a/drivers/infiniband/hw/qib/qib_verbs.c b/drivers/infiniband/hw/qib/qib_verbs.c index ba51a4715a1d..7c0ab16a2fe2 100644 --- a/drivers/infiniband/hw/qib/qib_verbs.c +++ b/drivers/infiniband/hw/qib/qib_verbs.c | |||
| @@ -1,5 +1,5 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * Copyright (c) 2012 Intel Corporation. All rights reserved. | 2 | * Copyright (c) 2012, 2013 Intel Corporation. All rights reserved. |
| 3 | * Copyright (c) 2006 - 2012 QLogic Corporation. All rights reserved. | 3 | * Copyright (c) 2006 - 2012 QLogic Corporation. All rights reserved. |
| 4 | * Copyright (c) 2005, 2006 PathScale, Inc. All rights reserved. | 4 | * Copyright (c) 2005, 2006 PathScale, Inc. All rights reserved. |
| 5 | * | 5 | * |
| @@ -2224,7 +2224,7 @@ int qib_register_ib_device(struct qib_devdata *dd) | |||
| 2224 | ibdev->dma_ops = &qib_dma_mapping_ops; | 2224 | ibdev->dma_ops = &qib_dma_mapping_ops; |
| 2225 | 2225 | ||
| 2226 | snprintf(ibdev->node_desc, sizeof(ibdev->node_desc), | 2226 | snprintf(ibdev->node_desc, sizeof(ibdev->node_desc), |
| 2227 | "QLogic Infiniband HCA %s", init_utsname()->nodename); | 2227 | "Intel Infiniband HCA %s", init_utsname()->nodename); |
| 2228 | 2228 | ||
| 2229 | ret = ib_register_device(ibdev, qib_create_port_files); | 2229 | ret = ib_register_device(ibdev, qib_create_port_files); |
| 2230 | if (ret) | 2230 | if (ret) |
diff --git a/drivers/infiniband/ulp/ipoib/ipoib_cm.c b/drivers/infiniband/ulp/ipoib/ipoib_cm.c index 67b0c1d23678..1ef880de3a41 100644 --- a/drivers/infiniband/ulp/ipoib/ipoib_cm.c +++ b/drivers/infiniband/ulp/ipoib/ipoib_cm.c | |||
| @@ -758,9 +758,13 @@ void ipoib_cm_send(struct net_device *dev, struct sk_buff *skb, struct ipoib_cm_ | |||
| 758 | if (++priv->tx_outstanding == ipoib_sendq_size) { | 758 | if (++priv->tx_outstanding == ipoib_sendq_size) { |
| 759 | ipoib_dbg(priv, "TX ring 0x%x full, stopping kernel net queue\n", | 759 | ipoib_dbg(priv, "TX ring 0x%x full, stopping kernel net queue\n", |
| 760 | tx->qp->qp_num); | 760 | tx->qp->qp_num); |
| 761 | if (ib_req_notify_cq(priv->send_cq, IB_CQ_NEXT_COMP)) | ||
| 762 | ipoib_warn(priv, "request notify on send CQ failed\n"); | ||
| 763 | netif_stop_queue(dev); | 761 | netif_stop_queue(dev); |
| 762 | rc = ib_req_notify_cq(priv->send_cq, | ||
| 763 | IB_CQ_NEXT_COMP | IB_CQ_REPORT_MISSED_EVENTS); | ||
| 764 | if (rc < 0) | ||
| 765 | ipoib_warn(priv, "request notify on send CQ failed\n"); | ||
| 766 | else if (rc) | ||
| 767 | ipoib_send_comp_handler(priv->send_cq, dev); | ||
| 764 | } | 768 | } |
| 765 | } | 769 | } |
| 766 | } | 770 | } |
diff --git a/drivers/iommu/Kconfig b/drivers/iommu/Kconfig index 5c514d0711d1..c332fb98480d 100644 --- a/drivers/iommu/Kconfig +++ b/drivers/iommu/Kconfig | |||
| @@ -130,7 +130,7 @@ config IRQ_REMAP | |||
| 130 | # OMAP IOMMU support | 130 | # OMAP IOMMU support |
| 131 | config OMAP_IOMMU | 131 | config OMAP_IOMMU |
| 132 | bool "OMAP IOMMU Support" | 132 | bool "OMAP IOMMU Support" |
| 133 | depends on ARCH_OMAP | 133 | depends on ARCH_OMAP2PLUS |
| 134 | select IOMMU_API | 134 | select IOMMU_API |
| 135 | 135 | ||
| 136 | config OMAP_IOVMM | 136 | config OMAP_IOVMM |
diff --git a/drivers/iommu/amd_iommu.c b/drivers/iommu/amd_iommu.c index 98f555dafb55..b287ca33833d 100644 --- a/drivers/iommu/amd_iommu.c +++ b/drivers/iommu/amd_iommu.c | |||
| @@ -2466,18 +2466,16 @@ static int device_change_notifier(struct notifier_block *nb, | |||
| 2466 | 2466 | ||
| 2467 | /* allocate a protection domain if a device is added */ | 2467 | /* allocate a protection domain if a device is added */ |
| 2468 | dma_domain = find_protection_domain(devid); | 2468 | dma_domain = find_protection_domain(devid); |
| 2469 | if (dma_domain) | 2469 | if (!dma_domain) { |
| 2470 | goto out; | 2470 | dma_domain = dma_ops_domain_alloc(); |
| 2471 | dma_domain = dma_ops_domain_alloc(); | 2471 | if (!dma_domain) |
| 2472 | if (!dma_domain) | 2472 | goto out; |
| 2473 | goto out; | 2473 | dma_domain->target_dev = devid; |
| 2474 | dma_domain->target_dev = devid; | 2474 | |
| 2475 | 2475 | spin_lock_irqsave(&iommu_pd_list_lock, flags); | |
| 2476 | spin_lock_irqsave(&iommu_pd_list_lock, flags); | 2476 | list_add_tail(&dma_domain->list, &iommu_pd_list); |
| 2477 | list_add_tail(&dma_domain->list, &iommu_pd_list); | 2477 | spin_unlock_irqrestore(&iommu_pd_list_lock, flags); |
| 2478 | spin_unlock_irqrestore(&iommu_pd_list_lock, flags); | 2478 | } |
| 2479 | |||
| 2480 | dev_data = get_dev_data(dev); | ||
| 2481 | 2479 | ||
| 2482 | dev->archdata.dma_ops = &amd_iommu_dma_ops; | 2480 | dev->archdata.dma_ops = &amd_iommu_dma_ops; |
| 2483 | 2481 | ||
diff --git a/drivers/iommu/amd_iommu_init.c b/drivers/iommu/amd_iommu_init.c index b6ecddb63cd0..e3c2d74b7684 100644 --- a/drivers/iommu/amd_iommu_init.c +++ b/drivers/iommu/amd_iommu_init.c | |||
| @@ -980,7 +980,7 @@ static void __init free_iommu_all(void) | |||
| 980 | * BIOS should disable L2B micellaneous clock gating by setting | 980 | * BIOS should disable L2B micellaneous clock gating by setting |
| 981 | * L2_L2B_CK_GATE_CONTROL[CKGateL2BMiscDisable](D0F2xF4_x90[2]) = 1b | 981 | * L2_L2B_CK_GATE_CONTROL[CKGateL2BMiscDisable](D0F2xF4_x90[2]) = 1b |
| 982 | */ | 982 | */ |
| 983 | static void __init amd_iommu_erratum_746_workaround(struct amd_iommu *iommu) | 983 | static void amd_iommu_erratum_746_workaround(struct amd_iommu *iommu) |
| 984 | { | 984 | { |
| 985 | u32 value; | 985 | u32 value; |
| 986 | 986 | ||
diff --git a/drivers/iommu/irq_remapping.c b/drivers/iommu/irq_remapping.c index d56f8c17c5fe..7c11ff368d07 100644 --- a/drivers/iommu/irq_remapping.c +++ b/drivers/iommu/irq_remapping.c | |||
| @@ -2,7 +2,6 @@ | |||
| 2 | #include <linux/cpumask.h> | 2 | #include <linux/cpumask.h> |
| 3 | #include <linux/kernel.h> | 3 | #include <linux/kernel.h> |
| 4 | #include <linux/string.h> | 4 | #include <linux/string.h> |
| 5 | #include <linux/cpumask.h> | ||
| 6 | #include <linux/errno.h> | 5 | #include <linux/errno.h> |
| 7 | #include <linux/msi.h> | 6 | #include <linux/msi.h> |
| 8 | #include <linux/irq.h> | 7 | #include <linux/irq.h> |
diff --git a/drivers/media/i2c/m5mols/m5mols_core.c b/drivers/media/i2c/m5mols/m5mols_core.c index d4e7567b367c..0b899cb6cda1 100644 --- a/drivers/media/i2c/m5mols/m5mols_core.c +++ b/drivers/media/i2c/m5mols/m5mols_core.c | |||
| @@ -724,7 +724,7 @@ static int m5mols_s_stream(struct v4l2_subdev *sd, int enable) | |||
| 724 | if (enable) { | 724 | if (enable) { |
| 725 | if (is_code(code, M5MOLS_RESTYPE_MONITOR)) | 725 | if (is_code(code, M5MOLS_RESTYPE_MONITOR)) |
| 726 | ret = m5mols_start_monitor(info); | 726 | ret = m5mols_start_monitor(info); |
| 727 | if (is_code(code, M5MOLS_RESTYPE_CAPTURE)) | 727 | else if (is_code(code, M5MOLS_RESTYPE_CAPTURE)) |
| 728 | ret = m5mols_start_capture(info); | 728 | ret = m5mols_start_capture(info); |
| 729 | else | 729 | else |
| 730 | ret = -EINVAL; | 730 | ret = -EINVAL; |
diff --git a/drivers/media/pci/bt8xx/bttv-driver.c b/drivers/media/pci/bt8xx/bttv-driver.c index ccd18e4ee789..54579e4c740b 100644 --- a/drivers/media/pci/bt8xx/bttv-driver.c +++ b/drivers/media/pci/bt8xx/bttv-driver.c | |||
| @@ -250,17 +250,19 @@ static u8 SRAM_Table[][60] = | |||
| 250 | vdelay start of active video in 2 * field lines relative to | 250 | vdelay start of active video in 2 * field lines relative to |
| 251 | trailing edge of /VRESET pulse (VDELAY register). | 251 | trailing edge of /VRESET pulse (VDELAY register). |
| 252 | sheight height of active video in 2 * field lines. | 252 | sheight height of active video in 2 * field lines. |
| 253 | extraheight Added to sheight for cropcap.bounds.height only | ||
| 253 | videostart0 ITU-R frame line number of the line corresponding | 254 | videostart0 ITU-R frame line number of the line corresponding |
| 254 | to vdelay in the first field. */ | 255 | to vdelay in the first field. */ |
| 255 | #define CROPCAP(minhdelayx1, hdelayx1, swidth, totalwidth, sqwidth, \ | 256 | #define CROPCAP(minhdelayx1, hdelayx1, swidth, totalwidth, sqwidth, \ |
| 256 | vdelay, sheight, videostart0) \ | 257 | vdelay, sheight, extraheight, videostart0) \ |
| 257 | .cropcap.bounds.left = minhdelayx1, \ | 258 | .cropcap.bounds.left = minhdelayx1, \ |
| 258 | /* * 2 because vertically we count field lines times two, */ \ | 259 | /* * 2 because vertically we count field lines times two, */ \ |
| 259 | /* e.g. 23 * 2 to 23 * 2 + 576 in PAL-BGHI defrect. */ \ | 260 | /* e.g. 23 * 2 to 23 * 2 + 576 in PAL-BGHI defrect. */ \ |
| 260 | .cropcap.bounds.top = (videostart0) * 2 - (vdelay) + MIN_VDELAY, \ | 261 | .cropcap.bounds.top = (videostart0) * 2 - (vdelay) + MIN_VDELAY, \ |
| 261 | /* 4 is a safety margin at the end of the line. */ \ | 262 | /* 4 is a safety margin at the end of the line. */ \ |
| 262 | .cropcap.bounds.width = (totalwidth) - (minhdelayx1) - 4, \ | 263 | .cropcap.bounds.width = (totalwidth) - (minhdelayx1) - 4, \ |
| 263 | .cropcap.bounds.height = (sheight) + (vdelay) - MIN_VDELAY, \ | 264 | .cropcap.bounds.height = (sheight) + (extraheight) + (vdelay) - \ |
| 265 | MIN_VDELAY, \ | ||
| 264 | .cropcap.defrect.left = hdelayx1, \ | 266 | .cropcap.defrect.left = hdelayx1, \ |
| 265 | .cropcap.defrect.top = (videostart0) * 2, \ | 267 | .cropcap.defrect.top = (videostart0) * 2, \ |
| 266 | .cropcap.defrect.width = swidth, \ | 268 | .cropcap.defrect.width = swidth, \ |
| @@ -301,9 +303,10 @@ const struct bttv_tvnorm bttv_tvnorms[] = { | |||
| 301 | /* totalwidth */ 1135, | 303 | /* totalwidth */ 1135, |
| 302 | /* sqwidth */ 944, | 304 | /* sqwidth */ 944, |
| 303 | /* vdelay */ 0x20, | 305 | /* vdelay */ 0x20, |
| 304 | /* bt878 (and bt848?) can capture another | 306 | /* sheight */ 576, |
| 305 | line below active video. */ | 307 | /* bt878 (and bt848?) can capture another |
| 306 | /* sheight */ (576 + 2) + 0x20 - 2, | 308 | line below active video. */ |
| 309 | /* extraheight */ 2, | ||
| 307 | /* videostart0 */ 23) | 310 | /* videostart0 */ 23) |
| 308 | },{ | 311 | },{ |
| 309 | .v4l2_id = V4L2_STD_NTSC_M | V4L2_STD_NTSC_M_KR, | 312 | .v4l2_id = V4L2_STD_NTSC_M | V4L2_STD_NTSC_M_KR, |
| @@ -330,6 +333,7 @@ const struct bttv_tvnorm bttv_tvnorms[] = { | |||
| 330 | /* sqwidth */ 780, | 333 | /* sqwidth */ 780, |
| 331 | /* vdelay */ 0x1a, | 334 | /* vdelay */ 0x1a, |
| 332 | /* sheight */ 480, | 335 | /* sheight */ 480, |
| 336 | /* extraheight */ 0, | ||
| 333 | /* videostart0 */ 23) | 337 | /* videostart0 */ 23) |
| 334 | },{ | 338 | },{ |
| 335 | .v4l2_id = V4L2_STD_SECAM, | 339 | .v4l2_id = V4L2_STD_SECAM, |
| @@ -355,6 +359,7 @@ const struct bttv_tvnorm bttv_tvnorms[] = { | |||
| 355 | /* sqwidth */ 944, | 359 | /* sqwidth */ 944, |
| 356 | /* vdelay */ 0x20, | 360 | /* vdelay */ 0x20, |
| 357 | /* sheight */ 576, | 361 | /* sheight */ 576, |
| 362 | /* extraheight */ 0, | ||
| 358 | /* videostart0 */ 23) | 363 | /* videostart0 */ 23) |
| 359 | },{ | 364 | },{ |
| 360 | .v4l2_id = V4L2_STD_PAL_Nc, | 365 | .v4l2_id = V4L2_STD_PAL_Nc, |
| @@ -380,6 +385,7 @@ const struct bttv_tvnorm bttv_tvnorms[] = { | |||
| 380 | /* sqwidth */ 780, | 385 | /* sqwidth */ 780, |
| 381 | /* vdelay */ 0x1a, | 386 | /* vdelay */ 0x1a, |
| 382 | /* sheight */ 576, | 387 | /* sheight */ 576, |
| 388 | /* extraheight */ 0, | ||
| 383 | /* videostart0 */ 23) | 389 | /* videostart0 */ 23) |
| 384 | },{ | 390 | },{ |
| 385 | .v4l2_id = V4L2_STD_PAL_M, | 391 | .v4l2_id = V4L2_STD_PAL_M, |
| @@ -405,6 +411,7 @@ const struct bttv_tvnorm bttv_tvnorms[] = { | |||
| 405 | /* sqwidth */ 780, | 411 | /* sqwidth */ 780, |
| 406 | /* vdelay */ 0x1a, | 412 | /* vdelay */ 0x1a, |
| 407 | /* sheight */ 480, | 413 | /* sheight */ 480, |
| 414 | /* extraheight */ 0, | ||
| 408 | /* videostart0 */ 23) | 415 | /* videostart0 */ 23) |
| 409 | },{ | 416 | },{ |
| 410 | .v4l2_id = V4L2_STD_PAL_N, | 417 | .v4l2_id = V4L2_STD_PAL_N, |
| @@ -430,6 +437,7 @@ const struct bttv_tvnorm bttv_tvnorms[] = { | |||
| 430 | /* sqwidth */ 944, | 437 | /* sqwidth */ 944, |
| 431 | /* vdelay */ 0x20, | 438 | /* vdelay */ 0x20, |
| 432 | /* sheight */ 576, | 439 | /* sheight */ 576, |
| 440 | /* extraheight */ 0, | ||
| 433 | /* videostart0 */ 23) | 441 | /* videostart0 */ 23) |
| 434 | },{ | 442 | },{ |
| 435 | .v4l2_id = V4L2_STD_NTSC_M_JP, | 443 | .v4l2_id = V4L2_STD_NTSC_M_JP, |
| @@ -455,6 +463,7 @@ const struct bttv_tvnorm bttv_tvnorms[] = { | |||
| 455 | /* sqwidth */ 780, | 463 | /* sqwidth */ 780, |
| 456 | /* vdelay */ 0x16, | 464 | /* vdelay */ 0x16, |
| 457 | /* sheight */ 480, | 465 | /* sheight */ 480, |
| 466 | /* extraheight */ 0, | ||
| 458 | /* videostart0 */ 23) | 467 | /* videostart0 */ 23) |
| 459 | },{ | 468 | },{ |
| 460 | /* that one hopefully works with the strange timing | 469 | /* that one hopefully works with the strange timing |
| @@ -484,6 +493,7 @@ const struct bttv_tvnorm bttv_tvnorms[] = { | |||
| 484 | /* sqwidth */ 944, | 493 | /* sqwidth */ 944, |
| 485 | /* vdelay */ 0x1a, | 494 | /* vdelay */ 0x1a, |
| 486 | /* sheight */ 480, | 495 | /* sheight */ 480, |
| 496 | /* extraheight */ 0, | ||
| 487 | /* videostart0 */ 23) | 497 | /* videostart0 */ 23) |
| 488 | } | 498 | } |
| 489 | }; | 499 | }; |
diff --git a/drivers/media/platform/exynos-gsc/gsc-core.c b/drivers/media/platform/exynos-gsc/gsc-core.c index 82d9f6ac12f3..33b5ffc8d66d 100644 --- a/drivers/media/platform/exynos-gsc/gsc-core.c +++ b/drivers/media/platform/exynos-gsc/gsc-core.c | |||
| @@ -1054,16 +1054,18 @@ static int gsc_m2m_suspend(struct gsc_dev *gsc) | |||
| 1054 | 1054 | ||
| 1055 | static int gsc_m2m_resume(struct gsc_dev *gsc) | 1055 | static int gsc_m2m_resume(struct gsc_dev *gsc) |
| 1056 | { | 1056 | { |
| 1057 | struct gsc_ctx *ctx; | ||
| 1057 | unsigned long flags; | 1058 | unsigned long flags; |
| 1058 | 1059 | ||
| 1059 | spin_lock_irqsave(&gsc->slock, flags); | 1060 | spin_lock_irqsave(&gsc->slock, flags); |
| 1060 | /* Clear for full H/W setup in first run after resume */ | 1061 | /* Clear for full H/W setup in first run after resume */ |
| 1062 | ctx = gsc->m2m.ctx; | ||
| 1061 | gsc->m2m.ctx = NULL; | 1063 | gsc->m2m.ctx = NULL; |
| 1062 | spin_unlock_irqrestore(&gsc->slock, flags); | 1064 | spin_unlock_irqrestore(&gsc->slock, flags); |
| 1063 | 1065 | ||
| 1064 | if (test_and_clear_bit(ST_M2M_SUSPENDED, &gsc->state)) | 1066 | if (test_and_clear_bit(ST_M2M_SUSPENDED, &gsc->state)) |
| 1065 | gsc_m2m_job_finish(gsc->m2m.ctx, | 1067 | gsc_m2m_job_finish(ctx, VB2_BUF_STATE_ERROR); |
| 1066 | VB2_BUF_STATE_ERROR); | 1068 | |
| 1067 | return 0; | 1069 | return 0; |
| 1068 | } | 1070 | } |
| 1069 | 1071 | ||
| @@ -1204,7 +1206,7 @@ static int gsc_resume(struct device *dev) | |||
| 1204 | /* Do not resume if the device was idle before system suspend */ | 1206 | /* Do not resume if the device was idle before system suspend */ |
| 1205 | spin_lock_irqsave(&gsc->slock, flags); | 1207 | spin_lock_irqsave(&gsc->slock, flags); |
| 1206 | if (!test_and_clear_bit(ST_SUSPEND, &gsc->state) || | 1208 | if (!test_and_clear_bit(ST_SUSPEND, &gsc->state) || |
| 1207 | !gsc_m2m_active(gsc)) { | 1209 | !gsc_m2m_opened(gsc)) { |
| 1208 | spin_unlock_irqrestore(&gsc->slock, flags); | 1210 | spin_unlock_irqrestore(&gsc->slock, flags); |
| 1209 | return 0; | 1211 | return 0; |
| 1210 | } | 1212 | } |
diff --git a/drivers/media/platform/s5p-fimc/fimc-core.c b/drivers/media/platform/s5p-fimc/fimc-core.c index e3916bde45cf..0f513dd19f86 100644 --- a/drivers/media/platform/s5p-fimc/fimc-core.c +++ b/drivers/media/platform/s5p-fimc/fimc-core.c | |||
| @@ -850,16 +850,18 @@ static int fimc_m2m_suspend(struct fimc_dev *fimc) | |||
| 850 | 850 | ||
| 851 | static int fimc_m2m_resume(struct fimc_dev *fimc) | 851 | static int fimc_m2m_resume(struct fimc_dev *fimc) |
| 852 | { | 852 | { |
| 853 | struct fimc_ctx *ctx; | ||
| 853 | unsigned long flags; | 854 | unsigned long flags; |
| 854 | 855 | ||
| 855 | spin_lock_irqsave(&fimc->slock, flags); | 856 | spin_lock_irqsave(&fimc->slock, flags); |
| 856 | /* Clear for full H/W setup in first run after resume */ | 857 | /* Clear for full H/W setup in first run after resume */ |
| 858 | ctx = fimc->m2m.ctx; | ||
| 857 | fimc->m2m.ctx = NULL; | 859 | fimc->m2m.ctx = NULL; |
| 858 | spin_unlock_irqrestore(&fimc->slock, flags); | 860 | spin_unlock_irqrestore(&fimc->slock, flags); |
| 859 | 861 | ||
| 860 | if (test_and_clear_bit(ST_M2M_SUSPENDED, &fimc->state)) | 862 | if (test_and_clear_bit(ST_M2M_SUSPENDED, &fimc->state)) |
| 861 | fimc_m2m_job_finish(fimc->m2m.ctx, | 863 | fimc_m2m_job_finish(ctx, VB2_BUF_STATE_ERROR); |
| 862 | VB2_BUF_STATE_ERROR); | 864 | |
| 863 | return 0; | 865 | return 0; |
| 864 | } | 866 | } |
| 865 | 867 | ||
diff --git a/drivers/media/platform/s5p-fimc/fimc-lite-reg.c b/drivers/media/platform/s5p-fimc/fimc-lite-reg.c index f0af0754a7b4..ac9663ce2a49 100644 --- a/drivers/media/platform/s5p-fimc/fimc-lite-reg.c +++ b/drivers/media/platform/s5p-fimc/fimc-lite-reg.c | |||
| @@ -128,10 +128,10 @@ static const u32 src_pixfmt_map[8][3] = { | |||
| 128 | void flite_hw_set_source_format(struct fimc_lite *dev, struct flite_frame *f) | 128 | void flite_hw_set_source_format(struct fimc_lite *dev, struct flite_frame *f) |
| 129 | { | 129 | { |
| 130 | enum v4l2_mbus_pixelcode pixelcode = dev->fmt->mbus_code; | 130 | enum v4l2_mbus_pixelcode pixelcode = dev->fmt->mbus_code; |
| 131 | unsigned int i = ARRAY_SIZE(src_pixfmt_map); | 131 | int i = ARRAY_SIZE(src_pixfmt_map); |
| 132 | u32 cfg; | 132 | u32 cfg; |
| 133 | 133 | ||
| 134 | while (i-- >= 0) { | 134 | while (--i >= 0) { |
| 135 | if (src_pixfmt_map[i][0] == pixelcode) | 135 | if (src_pixfmt_map[i][0] == pixelcode) |
| 136 | break; | 136 | break; |
| 137 | } | 137 | } |
| @@ -224,9 +224,9 @@ static void flite_hw_set_out_order(struct fimc_lite *dev, struct flite_frame *f) | |||
| 224 | { V4L2_MBUS_FMT_VYUY8_2X8, FLITE_REG_CIODMAFMT_CRYCBY }, | 224 | { V4L2_MBUS_FMT_VYUY8_2X8, FLITE_REG_CIODMAFMT_CRYCBY }, |
| 225 | }; | 225 | }; |
| 226 | u32 cfg = readl(dev->regs + FLITE_REG_CIODMAFMT); | 226 | u32 cfg = readl(dev->regs + FLITE_REG_CIODMAFMT); |
| 227 | unsigned int i = ARRAY_SIZE(pixcode); | 227 | int i = ARRAY_SIZE(pixcode); |
| 228 | 228 | ||
| 229 | while (i-- >= 0) | 229 | while (--i >= 0) |
| 230 | if (pixcode[i][0] == dev->fmt->mbus_code) | 230 | if (pixcode[i][0] == dev->fmt->mbus_code) |
| 231 | break; | 231 | break; |
| 232 | cfg &= ~FLITE_REG_CIODMAFMT_YCBCR_ORDER_MASK; | 232 | cfg &= ~FLITE_REG_CIODMAFMT_YCBCR_ORDER_MASK; |
diff --git a/drivers/media/platform/s5p-fimc/fimc-lite.c b/drivers/media/platform/s5p-fimc/fimc-lite.c index bfc4206935c8..bbc35de7db27 100644 --- a/drivers/media/platform/s5p-fimc/fimc-lite.c +++ b/drivers/media/platform/s5p-fimc/fimc-lite.c | |||
| @@ -1408,6 +1408,7 @@ static const struct v4l2_ctrl_config fimc_lite_ctrl = { | |||
| 1408 | .id = V4L2_CTRL_CLASS_USER | 0x1001, | 1408 | .id = V4L2_CTRL_CLASS_USER | 0x1001, |
| 1409 | .type = V4L2_CTRL_TYPE_BOOLEAN, | 1409 | .type = V4L2_CTRL_TYPE_BOOLEAN, |
| 1410 | .name = "Test Pattern 640x480", | 1410 | .name = "Test Pattern 640x480", |
| 1411 | .step = 1, | ||
| 1411 | }; | 1412 | }; |
| 1412 | 1413 | ||
| 1413 | static int fimc_lite_create_capture_subdev(struct fimc_lite *fimc) | 1414 | static int fimc_lite_create_capture_subdev(struct fimc_lite *fimc) |
diff --git a/drivers/media/platform/s5p-fimc/fimc-mdevice.c b/drivers/media/platform/s5p-fimc/fimc-mdevice.c index a17fcb2d5d41..cd38d708ab58 100644 --- a/drivers/media/platform/s5p-fimc/fimc-mdevice.c +++ b/drivers/media/platform/s5p-fimc/fimc-mdevice.c | |||
| @@ -827,7 +827,7 @@ static int fimc_md_link_notify(struct media_pad *source, | |||
| 827 | struct fimc_pipeline *pipeline; | 827 | struct fimc_pipeline *pipeline; |
| 828 | struct v4l2_subdev *sd; | 828 | struct v4l2_subdev *sd; |
| 829 | struct mutex *lock; | 829 | struct mutex *lock; |
| 830 | int ret = 0; | 830 | int i, ret = 0; |
| 831 | int ref_count; | 831 | int ref_count; |
| 832 | 832 | ||
| 833 | if (media_entity_type(sink->entity) != MEDIA_ENT_T_V4L2_SUBDEV) | 833 | if (media_entity_type(sink->entity) != MEDIA_ENT_T_V4L2_SUBDEV) |
| @@ -854,29 +854,28 @@ static int fimc_md_link_notify(struct media_pad *source, | |||
| 854 | return 0; | 854 | return 0; |
| 855 | } | 855 | } |
| 856 | 856 | ||
| 857 | mutex_lock(lock); | ||
| 858 | ref_count = fimc ? fimc->vid_cap.refcnt : fimc_lite->ref_count; | ||
| 859 | |||
| 857 | if (!(flags & MEDIA_LNK_FL_ENABLED)) { | 860 | if (!(flags & MEDIA_LNK_FL_ENABLED)) { |
| 858 | int i; | 861 | if (ref_count > 0) { |
| 859 | mutex_lock(lock); | 862 | ret = __fimc_pipeline_close(pipeline); |
| 860 | ret = __fimc_pipeline_close(pipeline); | 863 | if (!ret && fimc) |
| 864 | fimc_ctrls_delete(fimc->vid_cap.ctx); | ||
| 865 | } | ||
| 861 | for (i = 0; i < IDX_MAX; i++) | 866 | for (i = 0; i < IDX_MAX; i++) |
| 862 | pipeline->subdevs[i] = NULL; | 867 | pipeline->subdevs[i] = NULL; |
| 863 | if (fimc) | 868 | } else if (ref_count > 0) { |
| 864 | fimc_ctrls_delete(fimc->vid_cap.ctx); | 869 | /* |
| 865 | mutex_unlock(lock); | 870 | * Link activation. Enable power of pipeline elements only if |
| 866 | return ret; | 871 | * the pipeline is already in use, i.e. its video node is open. |
| 872 | * Recreate the controls destroyed during the link deactivation. | ||
| 873 | */ | ||
| 874 | ret = __fimc_pipeline_open(pipeline, | ||
| 875 | source->entity, true); | ||
| 876 | if (!ret && fimc) | ||
| 877 | ret = fimc_capture_ctrls_create(fimc); | ||
| 867 | } | 878 | } |
| 868 | /* | ||
| 869 | * Link activation. Enable power of pipeline elements only if the | ||
| 870 | * pipeline is already in use, i.e. its video node is opened. | ||
| 871 | * Recreate the controls destroyed during the link deactivation. | ||
| 872 | */ | ||
| 873 | mutex_lock(lock); | ||
| 874 | |||
| 875 | ref_count = fimc ? fimc->vid_cap.refcnt : fimc_lite->ref_count; | ||
| 876 | if (ref_count > 0) | ||
| 877 | ret = __fimc_pipeline_open(pipeline, source->entity, true); | ||
| 878 | if (!ret && fimc) | ||
| 879 | ret = fimc_capture_ctrls_create(fimc); | ||
| 880 | 879 | ||
| 881 | mutex_unlock(lock); | 880 | mutex_unlock(lock); |
| 882 | return ret ? -EPIPE : ret; | 881 | return ret ? -EPIPE : ret; |
diff --git a/drivers/media/platform/s5p-mfc/s5p_mfc.c b/drivers/media/platform/s5p-mfc/s5p_mfc.c index e84703c314ce..1cb6d57987c6 100644 --- a/drivers/media/platform/s5p-mfc/s5p_mfc.c +++ b/drivers/media/platform/s5p-mfc/s5p_mfc.c | |||
| @@ -276,7 +276,7 @@ static void s5p_mfc_handle_frame_new(struct s5p_mfc_ctx *ctx, unsigned int err) | |||
| 276 | unsigned int frame_type; | 276 | unsigned int frame_type; |
| 277 | 277 | ||
| 278 | dspl_y_addr = s5p_mfc_hw_call(dev->mfc_ops, get_dspl_y_adr, dev); | 278 | dspl_y_addr = s5p_mfc_hw_call(dev->mfc_ops, get_dspl_y_adr, dev); |
| 279 | frame_type = s5p_mfc_hw_call(dev->mfc_ops, get_dec_frame_type, dev); | 279 | frame_type = s5p_mfc_hw_call(dev->mfc_ops, get_disp_frame_type, ctx); |
| 280 | 280 | ||
| 281 | /* If frame is same as previous then skip and do not dequeue */ | 281 | /* If frame is same as previous then skip and do not dequeue */ |
| 282 | if (frame_type == S5P_FIMV_DECODE_FRAME_SKIPPED) { | 282 | if (frame_type == S5P_FIMV_DECODE_FRAME_SKIPPED) { |
diff --git a/drivers/media/platform/s5p-mfc/s5p_mfc_enc.c b/drivers/media/platform/s5p-mfc/s5p_mfc_enc.c index 2356fd52a169..4f6b553c4b2d 100644 --- a/drivers/media/platform/s5p-mfc/s5p_mfc_enc.c +++ b/drivers/media/platform/s5p-mfc/s5p_mfc_enc.c | |||
| @@ -232,6 +232,7 @@ static struct mfc_control controls[] = { | |||
| 232 | .minimum = 0, | 232 | .minimum = 0, |
| 233 | .maximum = 1, | 233 | .maximum = 1, |
| 234 | .default_value = 0, | 234 | .default_value = 0, |
| 235 | .step = 1, | ||
| 235 | .menu_skip_mask = 0, | 236 | .menu_skip_mask = 0, |
| 236 | }, | 237 | }, |
| 237 | { | 238 | { |
diff --git a/drivers/media/rc/Kconfig b/drivers/media/rc/Kconfig index 19f3563c61da..5a79c333d45e 100644 --- a/drivers/media/rc/Kconfig +++ b/drivers/media/rc/Kconfig | |||
| @@ -291,7 +291,7 @@ config IR_TTUSBIR | |||
| 291 | 291 | ||
| 292 | config IR_RX51 | 292 | config IR_RX51 |
| 293 | tristate "Nokia N900 IR transmitter diode" | 293 | tristate "Nokia N900 IR transmitter diode" |
| 294 | depends on OMAP_DM_TIMER && LIRC && !ARCH_MULTIPLATFORM | 294 | depends on OMAP_DM_TIMER && ARCH_OMAP2PLUS && LIRC && !ARCH_MULTIPLATFORM |
| 295 | ---help--- | 295 | ---help--- |
| 296 | Say Y or M here if you want to enable support for the IR | 296 | Say Y or M here if you want to enable support for the IR |
| 297 | transmitter diode built in the Nokia N900 (RX51) device. | 297 | transmitter diode built in the Nokia N900 (RX51) device. |
diff --git a/drivers/media/v4l2-core/Makefile b/drivers/media/v4l2-core/Makefile index a9d355230e8e..768aaf62d5dc 100644 --- a/drivers/media/v4l2-core/Makefile +++ b/drivers/media/v4l2-core/Makefile | |||
| @@ -10,7 +10,7 @@ ifeq ($(CONFIG_COMPAT),y) | |||
| 10 | videodev-objs += v4l2-compat-ioctl32.o | 10 | videodev-objs += v4l2-compat-ioctl32.o |
| 11 | endif | 11 | endif |
| 12 | 12 | ||
| 13 | obj-$(CONFIG_VIDEO_DEV) += videodev.o | 13 | obj-$(CONFIG_VIDEO_V4L2) += videodev.o |
| 14 | obj-$(CONFIG_VIDEO_V4L2_INT_DEVICE) += v4l2-int-device.o | 14 | obj-$(CONFIG_VIDEO_V4L2_INT_DEVICE) += v4l2-int-device.o |
| 15 | obj-$(CONFIG_VIDEO_V4L2) += v4l2-common.o | 15 | obj-$(CONFIG_VIDEO_V4L2) += v4l2-common.o |
| 16 | 16 | ||
diff --git a/drivers/misc/mei/hw-me.c b/drivers/misc/mei/hw-me.c index 45ea7185c003..642c6223fa6c 100644 --- a/drivers/misc/mei/hw-me.c +++ b/drivers/misc/mei/hw-me.c | |||
| @@ -152,6 +152,20 @@ static void mei_me_intr_disable(struct mei_device *dev) | |||
| 152 | } | 152 | } |
| 153 | 153 | ||
| 154 | /** | 154 | /** |
| 155 | * mei_me_hw_reset_release - release device from the reset | ||
| 156 | * | ||
| 157 | * @dev: the device structure | ||
| 158 | */ | ||
| 159 | static void mei_me_hw_reset_release(struct mei_device *dev) | ||
| 160 | { | ||
| 161 | struct mei_me_hw *hw = to_me_hw(dev); | ||
| 162 | u32 hcsr = mei_hcsr_read(hw); | ||
| 163 | |||
| 164 | hcsr |= H_IG; | ||
| 165 | hcsr &= ~H_RST; | ||
| 166 | mei_hcsr_set(hw, hcsr); | ||
| 167 | } | ||
| 168 | /** | ||
| 155 | * mei_me_hw_reset - resets fw via mei csr register. | 169 | * mei_me_hw_reset - resets fw via mei csr register. |
| 156 | * | 170 | * |
| 157 | * @dev: the device structure | 171 | * @dev: the device structure |
| @@ -169,18 +183,14 @@ static void mei_me_hw_reset(struct mei_device *dev, bool intr_enable) | |||
| 169 | if (intr_enable) | 183 | if (intr_enable) |
| 170 | hcsr |= H_IE; | 184 | hcsr |= H_IE; |
| 171 | else | 185 | else |
| 172 | hcsr &= ~H_IE; | 186 | hcsr |= ~H_IE; |
| 173 | |||
| 174 | mei_hcsr_set(hw, hcsr); | ||
| 175 | |||
| 176 | hcsr = mei_hcsr_read(hw) | H_IG; | ||
| 177 | hcsr &= ~H_RST; | ||
| 178 | 187 | ||
| 179 | mei_hcsr_set(hw, hcsr); | 188 | mei_hcsr_set(hw, hcsr); |
| 180 | 189 | ||
| 181 | hcsr = mei_hcsr_read(hw); | 190 | if (dev->dev_state == MEI_DEV_POWER_DOWN) |
| 191 | mei_me_hw_reset_release(dev); | ||
| 182 | 192 | ||
| 183 | dev_dbg(&dev->pdev->dev, "current HCSR = 0x%08x.\n", hcsr); | 193 | dev_dbg(&dev->pdev->dev, "current HCSR = 0x%08x.\n", mei_hcsr_read(hw)); |
| 184 | } | 194 | } |
| 185 | 195 | ||
| 186 | /** | 196 | /** |
| @@ -466,7 +476,8 @@ irqreturn_t mei_me_irq_thread_handler(int irq, void *dev_id) | |||
| 466 | mutex_unlock(&dev->device_lock); | 476 | mutex_unlock(&dev->device_lock); |
| 467 | return IRQ_HANDLED; | 477 | return IRQ_HANDLED; |
| 468 | } else { | 478 | } else { |
| 469 | dev_dbg(&dev->pdev->dev, "FW not ready.\n"); | 479 | dev_dbg(&dev->pdev->dev, "Reset Completed.\n"); |
| 480 | mei_me_hw_reset_release(dev); | ||
| 470 | mutex_unlock(&dev->device_lock); | 481 | mutex_unlock(&dev->device_lock); |
| 471 | return IRQ_HANDLED; | 482 | return IRQ_HANDLED; |
| 472 | } | 483 | } |
diff --git a/drivers/misc/mei/init.c b/drivers/misc/mei/init.c index 6ec530168afb..356179991a2e 100644 --- a/drivers/misc/mei/init.c +++ b/drivers/misc/mei/init.c | |||
| @@ -183,6 +183,24 @@ void mei_reset(struct mei_device *dev, int interrupts_enabled) | |||
| 183 | mei_cl_all_write_clear(dev); | 183 | mei_cl_all_write_clear(dev); |
| 184 | } | 184 | } |
| 185 | 185 | ||
| 186 | void mei_stop(struct mei_device *dev) | ||
| 187 | { | ||
| 188 | dev_dbg(&dev->pdev->dev, "stopping the device.\n"); | ||
| 189 | |||
| 190 | mutex_lock(&dev->device_lock); | ||
| 191 | |||
| 192 | cancel_delayed_work(&dev->timer_work); | ||
| 193 | |||
| 194 | mei_wd_stop(dev); | ||
| 195 | |||
| 196 | dev->dev_state = MEI_DEV_POWER_DOWN; | ||
| 197 | mei_reset(dev, 0); | ||
| 198 | |||
| 199 | mutex_unlock(&dev->device_lock); | ||
| 200 | |||
| 201 | flush_scheduled_work(); | ||
| 202 | } | ||
| 203 | |||
| 186 | 204 | ||
| 187 | 205 | ||
| 188 | 206 | ||
diff --git a/drivers/misc/mei/mei_dev.h b/drivers/misc/mei/mei_dev.h index cb80166161f0..97873812e33b 100644 --- a/drivers/misc/mei/mei_dev.h +++ b/drivers/misc/mei/mei_dev.h | |||
| @@ -381,6 +381,7 @@ static inline unsigned long mei_secs_to_jiffies(unsigned long sec) | |||
| 381 | void mei_device_init(struct mei_device *dev); | 381 | void mei_device_init(struct mei_device *dev); |
| 382 | void mei_reset(struct mei_device *dev, int interrupts); | 382 | void mei_reset(struct mei_device *dev, int interrupts); |
| 383 | int mei_hw_init(struct mei_device *dev); | 383 | int mei_hw_init(struct mei_device *dev); |
| 384 | void mei_stop(struct mei_device *dev); | ||
| 384 | 385 | ||
| 385 | /* | 386 | /* |
| 386 | * MEI interrupt functions prototype | 387 | * MEI interrupt functions prototype |
diff --git a/drivers/misc/mei/pci-me.c b/drivers/misc/mei/pci-me.c index b40ec0601ab0..b8b5c9c3ad03 100644 --- a/drivers/misc/mei/pci-me.c +++ b/drivers/misc/mei/pci-me.c | |||
| @@ -247,44 +247,14 @@ static void mei_remove(struct pci_dev *pdev) | |||
| 247 | 247 | ||
| 248 | hw = to_me_hw(dev); | 248 | hw = to_me_hw(dev); |
| 249 | 249 | ||
| 250 | mutex_lock(&dev->device_lock); | ||
| 251 | |||
| 252 | cancel_delayed_work(&dev->timer_work); | ||
| 253 | 250 | ||
| 254 | mei_wd_stop(dev); | 251 | dev_err(&pdev->dev, "stop\n"); |
| 252 | mei_stop(dev); | ||
| 255 | 253 | ||
| 256 | mei_pdev = NULL; | 254 | mei_pdev = NULL; |
| 257 | 255 | ||
| 258 | if (dev->iamthif_cl.state == MEI_FILE_CONNECTED) { | ||
| 259 | dev->iamthif_cl.state = MEI_FILE_DISCONNECTING; | ||
| 260 | mei_cl_disconnect(&dev->iamthif_cl); | ||
| 261 | } | ||
| 262 | if (dev->wd_cl.state == MEI_FILE_CONNECTED) { | ||
| 263 | dev->wd_cl.state = MEI_FILE_DISCONNECTING; | ||
| 264 | mei_cl_disconnect(&dev->wd_cl); | ||
| 265 | } | ||
| 266 | |||
| 267 | /* Unregistering watchdog device */ | ||
| 268 | mei_watchdog_unregister(dev); | 256 | mei_watchdog_unregister(dev); |
| 269 | 257 | ||
| 270 | /* remove entry if already in list */ | ||
| 271 | dev_dbg(&pdev->dev, "list del iamthif and wd file list.\n"); | ||
| 272 | |||
| 273 | if (dev->open_handle_count > 0) | ||
| 274 | dev->open_handle_count--; | ||
| 275 | mei_cl_unlink(&dev->wd_cl); | ||
| 276 | |||
| 277 | if (dev->open_handle_count > 0) | ||
| 278 | dev->open_handle_count--; | ||
| 279 | mei_cl_unlink(&dev->iamthif_cl); | ||
| 280 | |||
| 281 | dev->iamthif_current_cb = NULL; | ||
| 282 | dev->me_clients_num = 0; | ||
| 283 | |||
| 284 | mutex_unlock(&dev->device_lock); | ||
| 285 | |||
| 286 | flush_scheduled_work(); | ||
| 287 | |||
| 288 | /* disable interrupts */ | 258 | /* disable interrupts */ |
| 289 | mei_disable_interrupts(dev); | 259 | mei_disable_interrupts(dev); |
| 290 | 260 | ||
| @@ -308,28 +278,20 @@ static int mei_pci_suspend(struct device *device) | |||
| 308 | { | 278 | { |
| 309 | struct pci_dev *pdev = to_pci_dev(device); | 279 | struct pci_dev *pdev = to_pci_dev(device); |
| 310 | struct mei_device *dev = pci_get_drvdata(pdev); | 280 | struct mei_device *dev = pci_get_drvdata(pdev); |
| 311 | int err; | ||
| 312 | 281 | ||
| 313 | if (!dev) | 282 | if (!dev) |
| 314 | return -ENODEV; | 283 | return -ENODEV; |
| 315 | mutex_lock(&dev->device_lock); | ||
| 316 | 284 | ||
| 317 | cancel_delayed_work(&dev->timer_work); | 285 | dev_err(&pdev->dev, "suspend\n"); |
| 318 | 286 | ||
| 319 | /* Stop watchdog if exists */ | 287 | mei_stop(dev); |
| 320 | err = mei_wd_stop(dev); | 288 | |
| 321 | /* Set new mei state */ | 289 | mei_disable_interrupts(dev); |
| 322 | if (dev->dev_state == MEI_DEV_ENABLED || | ||
| 323 | dev->dev_state == MEI_DEV_RECOVERING_FROM_RESET) { | ||
| 324 | dev->dev_state = MEI_DEV_POWER_DOWN; | ||
| 325 | mei_reset(dev, 0); | ||
| 326 | } | ||
| 327 | mutex_unlock(&dev->device_lock); | ||
| 328 | 290 | ||
| 329 | free_irq(pdev->irq, dev); | 291 | free_irq(pdev->irq, dev); |
| 330 | pci_disable_msi(pdev); | 292 | pci_disable_msi(pdev); |
| 331 | 293 | ||
| 332 | return err; | 294 | return 0; |
| 333 | } | 295 | } |
| 334 | 296 | ||
| 335 | static int mei_pci_resume(struct device *device) | 297 | static int mei_pci_resume(struct device *device) |
diff --git a/drivers/misc/vmw_vmci/vmci_datagram.c b/drivers/misc/vmw_vmci/vmci_datagram.c index ed5c433cd493..f3cdd904fe4d 100644 --- a/drivers/misc/vmw_vmci/vmci_datagram.c +++ b/drivers/misc/vmw_vmci/vmci_datagram.c | |||
| @@ -42,9 +42,11 @@ struct datagram_entry { | |||
| 42 | 42 | ||
| 43 | struct delayed_datagram_info { | 43 | struct delayed_datagram_info { |
| 44 | struct datagram_entry *entry; | 44 | struct datagram_entry *entry; |
| 45 | struct vmci_datagram msg; | ||
| 46 | struct work_struct work; | 45 | struct work_struct work; |
| 47 | bool in_dg_host_queue; | 46 | bool in_dg_host_queue; |
| 47 | /* msg and msg_payload must be together. */ | ||
| 48 | struct vmci_datagram msg; | ||
| 49 | u8 msg_payload[]; | ||
| 48 | }; | 50 | }; |
| 49 | 51 | ||
| 50 | /* Number of in-flight host->host datagrams */ | 52 | /* Number of in-flight host->host datagrams */ |
diff --git a/drivers/net/bonding/bond_sysfs.c b/drivers/net/bonding/bond_sysfs.c index 1c9e09fbdff8..db103e03ba05 100644 --- a/drivers/net/bonding/bond_sysfs.c +++ b/drivers/net/bonding/bond_sysfs.c | |||
| @@ -183,6 +183,11 @@ int bond_create_slave_symlinks(struct net_device *master, | |||
| 183 | sprintf(linkname, "slave_%s", slave->name); | 183 | sprintf(linkname, "slave_%s", slave->name); |
| 184 | ret = sysfs_create_link(&(master->dev.kobj), &(slave->dev.kobj), | 184 | ret = sysfs_create_link(&(master->dev.kobj), &(slave->dev.kobj), |
| 185 | linkname); | 185 | linkname); |
| 186 | |||
| 187 | /* free the master link created earlier in case of error */ | ||
| 188 | if (ret) | ||
| 189 | sysfs_remove_link(&(slave->dev.kobj), "master"); | ||
| 190 | |||
| 186 | return ret; | 191 | return ret; |
| 187 | 192 | ||
| 188 | } | 193 | } |
diff --git a/drivers/net/ethernet/broadcom/bnx2x/bnx2x_dcb.c b/drivers/net/ethernet/broadcom/bnx2x/bnx2x_dcb.c index 568205436a15..91ecd6a00d05 100644 --- a/drivers/net/ethernet/broadcom/bnx2x/bnx2x_dcb.c +++ b/drivers/net/ethernet/broadcom/bnx2x/bnx2x_dcb.c | |||
| @@ -2139,12 +2139,12 @@ static u8 bnx2x_dcbnl_get_cap(struct net_device *netdev, int capid, u8 *cap) | |||
| 2139 | break; | 2139 | break; |
| 2140 | default: | 2140 | default: |
| 2141 | BNX2X_ERR("Non valid capability ID\n"); | 2141 | BNX2X_ERR("Non valid capability ID\n"); |
| 2142 | rval = -EINVAL; | 2142 | rval = 1; |
| 2143 | break; | 2143 | break; |
| 2144 | } | 2144 | } |
| 2145 | } else { | 2145 | } else { |
| 2146 | DP(BNX2X_MSG_DCB, "DCB disabled\n"); | 2146 | DP(BNX2X_MSG_DCB, "DCB disabled\n"); |
| 2147 | rval = -EINVAL; | 2147 | rval = 1; |
| 2148 | } | 2148 | } |
| 2149 | 2149 | ||
| 2150 | DP(BNX2X_MSG_DCB, "capid %d:%x\n", capid, *cap); | 2150 | DP(BNX2X_MSG_DCB, "capid %d:%x\n", capid, *cap); |
| @@ -2170,12 +2170,12 @@ static int bnx2x_dcbnl_get_numtcs(struct net_device *netdev, int tcid, u8 *num) | |||
| 2170 | break; | 2170 | break; |
| 2171 | default: | 2171 | default: |
| 2172 | BNX2X_ERR("Non valid TC-ID\n"); | 2172 | BNX2X_ERR("Non valid TC-ID\n"); |
| 2173 | rval = -EINVAL; | 2173 | rval = 1; |
| 2174 | break; | 2174 | break; |
| 2175 | } | 2175 | } |
| 2176 | } else { | 2176 | } else { |
| 2177 | DP(BNX2X_MSG_DCB, "DCB disabled\n"); | 2177 | DP(BNX2X_MSG_DCB, "DCB disabled\n"); |
| 2178 | rval = -EINVAL; | 2178 | rval = 1; |
| 2179 | } | 2179 | } |
| 2180 | 2180 | ||
| 2181 | return rval; | 2181 | return rval; |
| @@ -2188,7 +2188,7 @@ static int bnx2x_dcbnl_set_numtcs(struct net_device *netdev, int tcid, u8 num) | |||
| 2188 | return -EINVAL; | 2188 | return -EINVAL; |
| 2189 | } | 2189 | } |
| 2190 | 2190 | ||
| 2191 | static u8 bnx2x_dcbnl_get_pfc_state(struct net_device *netdev) | 2191 | static u8 bnx2x_dcbnl_get_pfc_state(struct net_device *netdev) |
| 2192 | { | 2192 | { |
| 2193 | struct bnx2x *bp = netdev_priv(netdev); | 2193 | struct bnx2x *bp = netdev_priv(netdev); |
| 2194 | DP(BNX2X_MSG_DCB, "state = %d\n", bp->dcbx_local_feat.pfc.enabled); | 2194 | DP(BNX2X_MSG_DCB, "state = %d\n", bp->dcbx_local_feat.pfc.enabled); |
| @@ -2390,12 +2390,12 @@ static u8 bnx2x_dcbnl_get_featcfg(struct net_device *netdev, int featid, | |||
| 2390 | break; | 2390 | break; |
| 2391 | default: | 2391 | default: |
| 2392 | BNX2X_ERR("Non valid featrue-ID\n"); | 2392 | BNX2X_ERR("Non valid featrue-ID\n"); |
| 2393 | rval = -EINVAL; | 2393 | rval = 1; |
| 2394 | break; | 2394 | break; |
| 2395 | } | 2395 | } |
| 2396 | } else { | 2396 | } else { |
| 2397 | DP(BNX2X_MSG_DCB, "DCB disabled\n"); | 2397 | DP(BNX2X_MSG_DCB, "DCB disabled\n"); |
| 2398 | rval = -EINVAL; | 2398 | rval = 1; |
| 2399 | } | 2399 | } |
| 2400 | 2400 | ||
| 2401 | return rval; | 2401 | return rval; |
| @@ -2431,12 +2431,12 @@ static u8 bnx2x_dcbnl_set_featcfg(struct net_device *netdev, int featid, | |||
| 2431 | break; | 2431 | break; |
| 2432 | default: | 2432 | default: |
| 2433 | BNX2X_ERR("Non valid featrue-ID\n"); | 2433 | BNX2X_ERR("Non valid featrue-ID\n"); |
| 2434 | rval = -EINVAL; | 2434 | rval = 1; |
| 2435 | break; | 2435 | break; |
| 2436 | } | 2436 | } |
| 2437 | } else { | 2437 | } else { |
| 2438 | DP(BNX2X_MSG_DCB, "dcbnl call not valid\n"); | 2438 | DP(BNX2X_MSG_DCB, "dcbnl call not valid\n"); |
| 2439 | rval = -EINVAL; | 2439 | rval = 1; |
| 2440 | } | 2440 | } |
| 2441 | 2441 | ||
| 2442 | return rval; | 2442 | return rval; |
diff --git a/drivers/net/ethernet/freescale/fec.c b/drivers/net/ethernet/freescale/fec.c index e3f39372ce25..911d0253dbb2 100644 --- a/drivers/net/ethernet/freescale/fec.c +++ b/drivers/net/ethernet/freescale/fec.c | |||
| @@ -1332,7 +1332,7 @@ static int fec_enet_ioctl(struct net_device *ndev, struct ifreq *rq, int cmd) | |||
| 1332 | static void fec_enet_free_buffers(struct net_device *ndev) | 1332 | static void fec_enet_free_buffers(struct net_device *ndev) |
| 1333 | { | 1333 | { |
| 1334 | struct fec_enet_private *fep = netdev_priv(ndev); | 1334 | struct fec_enet_private *fep = netdev_priv(ndev); |
| 1335 | int i; | 1335 | unsigned int i; |
| 1336 | struct sk_buff *skb; | 1336 | struct sk_buff *skb; |
| 1337 | struct bufdesc *bdp; | 1337 | struct bufdesc *bdp; |
| 1338 | 1338 | ||
| @@ -1356,7 +1356,7 @@ static void fec_enet_free_buffers(struct net_device *ndev) | |||
| 1356 | static int fec_enet_alloc_buffers(struct net_device *ndev) | 1356 | static int fec_enet_alloc_buffers(struct net_device *ndev) |
| 1357 | { | 1357 | { |
| 1358 | struct fec_enet_private *fep = netdev_priv(ndev); | 1358 | struct fec_enet_private *fep = netdev_priv(ndev); |
| 1359 | int i; | 1359 | unsigned int i; |
| 1360 | struct sk_buff *skb; | 1360 | struct sk_buff *skb; |
| 1361 | struct bufdesc *bdp; | 1361 | struct bufdesc *bdp; |
| 1362 | 1362 | ||
| @@ -1598,7 +1598,7 @@ static int fec_enet_init(struct net_device *ndev) | |||
| 1598 | struct fec_enet_private *fep = netdev_priv(ndev); | 1598 | struct fec_enet_private *fep = netdev_priv(ndev); |
| 1599 | struct bufdesc *cbd_base; | 1599 | struct bufdesc *cbd_base; |
| 1600 | struct bufdesc *bdp; | 1600 | struct bufdesc *bdp; |
| 1601 | int i; | 1601 | unsigned int i; |
| 1602 | 1602 | ||
| 1603 | /* Allocate memory for buffer descriptors. */ | 1603 | /* Allocate memory for buffer descriptors. */ |
| 1604 | cbd_base = dma_alloc_coherent(NULL, PAGE_SIZE, &fep->bd_dma, | 1604 | cbd_base = dma_alloc_coherent(NULL, PAGE_SIZE, &fep->bd_dma, |
diff --git a/drivers/net/ethernet/freescale/fec_ptp.c b/drivers/net/ethernet/freescale/fec_ptp.c index 1f17ca0f2201..0d8df400a479 100644 --- a/drivers/net/ethernet/freescale/fec_ptp.c +++ b/drivers/net/ethernet/freescale/fec_ptp.c | |||
| @@ -128,6 +128,7 @@ void fec_ptp_start_cyclecounter(struct net_device *ndev) | |||
| 128 | 128 | ||
| 129 | spin_unlock_irqrestore(&fep->tmreg_lock, flags); | 129 | spin_unlock_irqrestore(&fep->tmreg_lock, flags); |
| 130 | } | 130 | } |
| 131 | EXPORT_SYMBOL(fec_ptp_start_cyclecounter); | ||
| 131 | 132 | ||
| 132 | /** | 133 | /** |
| 133 | * fec_ptp_adjfreq - adjust ptp cycle frequency | 134 | * fec_ptp_adjfreq - adjust ptp cycle frequency |
| @@ -318,6 +319,7 @@ int fec_ptp_ioctl(struct net_device *ndev, struct ifreq *ifr, int cmd) | |||
| 318 | return copy_to_user(ifr->ifr_data, &config, sizeof(config)) ? | 319 | return copy_to_user(ifr->ifr_data, &config, sizeof(config)) ? |
| 319 | -EFAULT : 0; | 320 | -EFAULT : 0; |
| 320 | } | 321 | } |
| 322 | EXPORT_SYMBOL(fec_ptp_ioctl); | ||
| 321 | 323 | ||
| 322 | /** | 324 | /** |
| 323 | * fec_time_keep - call timecounter_read every second to avoid timer overrun | 325 | * fec_time_keep - call timecounter_read every second to avoid timer overrun |
| @@ -383,3 +385,4 @@ void fec_ptp_init(struct net_device *ndev, struct platform_device *pdev) | |||
| 383 | pr_info("registered PHC device on %s\n", ndev->name); | 385 | pr_info("registered PHC device on %s\n", ndev->name); |
| 384 | } | 386 | } |
| 385 | } | 387 | } |
| 388 | EXPORT_SYMBOL(fec_ptp_init); | ||
diff --git a/drivers/net/ethernet/intel/igb/e1000_82575.c b/drivers/net/ethernet/intel/igb/e1000_82575.c index b64542acfa34..12b1d8480808 100644 --- a/drivers/net/ethernet/intel/igb/e1000_82575.c +++ b/drivers/net/ethernet/intel/igb/e1000_82575.c | |||
| @@ -1818,27 +1818,32 @@ out: | |||
| 1818 | **/ | 1818 | **/ |
| 1819 | void igb_vmdq_set_anti_spoofing_pf(struct e1000_hw *hw, bool enable, int pf) | 1819 | void igb_vmdq_set_anti_spoofing_pf(struct e1000_hw *hw, bool enable, int pf) |
| 1820 | { | 1820 | { |
| 1821 | u32 dtxswc; | 1821 | u32 reg_val, reg_offset; |
| 1822 | 1822 | ||
| 1823 | switch (hw->mac.type) { | 1823 | switch (hw->mac.type) { |
| 1824 | case e1000_82576: | 1824 | case e1000_82576: |
| 1825 | reg_offset = E1000_DTXSWC; | ||
| 1826 | break; | ||
| 1825 | case e1000_i350: | 1827 | case e1000_i350: |
| 1826 | dtxswc = rd32(E1000_DTXSWC); | 1828 | reg_offset = E1000_TXSWC; |
| 1827 | if (enable) { | ||
| 1828 | dtxswc |= (E1000_DTXSWC_MAC_SPOOF_MASK | | ||
| 1829 | E1000_DTXSWC_VLAN_SPOOF_MASK); | ||
| 1830 | /* The PF can spoof - it has to in order to | ||
| 1831 | * support emulation mode NICs */ | ||
| 1832 | dtxswc ^= (1 << pf | 1 << (pf + MAX_NUM_VFS)); | ||
| 1833 | } else { | ||
| 1834 | dtxswc &= ~(E1000_DTXSWC_MAC_SPOOF_MASK | | ||
| 1835 | E1000_DTXSWC_VLAN_SPOOF_MASK); | ||
| 1836 | } | ||
| 1837 | wr32(E1000_DTXSWC, dtxswc); | ||
| 1838 | break; | 1829 | break; |
| 1839 | default: | 1830 | default: |
| 1840 | break; | 1831 | return; |
| 1832 | } | ||
| 1833 | |||
| 1834 | reg_val = rd32(reg_offset); | ||
| 1835 | if (enable) { | ||
| 1836 | reg_val |= (E1000_DTXSWC_MAC_SPOOF_MASK | | ||
| 1837 | E1000_DTXSWC_VLAN_SPOOF_MASK); | ||
| 1838 | /* The PF can spoof - it has to in order to | ||
| 1839 | * support emulation mode NICs | ||
| 1840 | */ | ||
| 1841 | reg_val ^= (1 << pf | 1 << (pf + MAX_NUM_VFS)); | ||
| 1842 | } else { | ||
| 1843 | reg_val &= ~(E1000_DTXSWC_MAC_SPOOF_MASK | | ||
| 1844 | E1000_DTXSWC_VLAN_SPOOF_MASK); | ||
| 1841 | } | 1845 | } |
| 1846 | wr32(reg_offset, reg_val); | ||
| 1842 | } | 1847 | } |
| 1843 | 1848 | ||
| 1844 | /** | 1849 | /** |
diff --git a/drivers/net/ethernet/intel/igb/igb_hwmon.c b/drivers/net/ethernet/intel/igb/igb_hwmon.c index 4623502054d5..0478a1abe541 100644 --- a/drivers/net/ethernet/intel/igb/igb_hwmon.c +++ b/drivers/net/ethernet/intel/igb/igb_hwmon.c | |||
| @@ -39,7 +39,7 @@ | |||
| 39 | #include <linux/pci.h> | 39 | #include <linux/pci.h> |
| 40 | 40 | ||
| 41 | #ifdef CONFIG_IGB_HWMON | 41 | #ifdef CONFIG_IGB_HWMON |
| 42 | struct i2c_board_info i350_sensor_info = { | 42 | static struct i2c_board_info i350_sensor_info = { |
| 43 | I2C_BOARD_INFO("i350bb", (0Xf8 >> 1)), | 43 | I2C_BOARD_INFO("i350bb", (0Xf8 >> 1)), |
| 44 | }; | 44 | }; |
| 45 | 45 | ||
diff --git a/drivers/net/ethernet/intel/igb/igb_main.c b/drivers/net/ethernet/intel/igb/igb_main.c index 4dbd62968c7a..8496adfc6a68 100644 --- a/drivers/net/ethernet/intel/igb/igb_main.c +++ b/drivers/net/ethernet/intel/igb/igb_main.c | |||
| @@ -2542,8 +2542,8 @@ static void igb_probe_vfs(struct igb_adapter *adapter) | |||
| 2542 | if ((hw->mac.type == e1000_i210) || (hw->mac.type == e1000_i211)) | 2542 | if ((hw->mac.type == e1000_i210) || (hw->mac.type == e1000_i211)) |
| 2543 | return; | 2543 | return; |
| 2544 | 2544 | ||
| 2545 | igb_enable_sriov(pdev, max_vfs); | ||
| 2546 | pci_sriov_set_totalvfs(pdev, 7); | 2545 | pci_sriov_set_totalvfs(pdev, 7); |
| 2546 | igb_enable_sriov(pdev, max_vfs); | ||
| 2547 | 2547 | ||
| 2548 | #endif /* CONFIG_PCI_IOV */ | 2548 | #endif /* CONFIG_PCI_IOV */ |
| 2549 | } | 2549 | } |
| @@ -2652,7 +2652,7 @@ static int igb_sw_init(struct igb_adapter *adapter) | |||
| 2652 | if (max_vfs > 7) { | 2652 | if (max_vfs > 7) { |
| 2653 | dev_warn(&pdev->dev, | 2653 | dev_warn(&pdev->dev, |
| 2654 | "Maximum of 7 VFs per PF, using max\n"); | 2654 | "Maximum of 7 VFs per PF, using max\n"); |
| 2655 | adapter->vfs_allocated_count = 7; | 2655 | max_vfs = adapter->vfs_allocated_count = 7; |
| 2656 | } else | 2656 | } else |
| 2657 | adapter->vfs_allocated_count = max_vfs; | 2657 | adapter->vfs_allocated_count = max_vfs; |
| 2658 | if (adapter->vfs_allocated_count) | 2658 | if (adapter->vfs_allocated_count) |
diff --git a/drivers/net/ethernet/intel/igb/igb_ptp.c b/drivers/net/ethernet/intel/igb/igb_ptp.c index 0987822359f0..0a237507ee85 100644 --- a/drivers/net/ethernet/intel/igb/igb_ptp.c +++ b/drivers/net/ethernet/intel/igb/igb_ptp.c | |||
| @@ -740,7 +740,7 @@ void igb_ptp_init(struct igb_adapter *adapter) | |||
| 740 | case e1000_82576: | 740 | case e1000_82576: |
| 741 | snprintf(adapter->ptp_caps.name, 16, "%pm", netdev->dev_addr); | 741 | snprintf(adapter->ptp_caps.name, 16, "%pm", netdev->dev_addr); |
| 742 | adapter->ptp_caps.owner = THIS_MODULE; | 742 | adapter->ptp_caps.owner = THIS_MODULE; |
| 743 | adapter->ptp_caps.max_adj = 1000000000; | 743 | adapter->ptp_caps.max_adj = 999999881; |
| 744 | adapter->ptp_caps.n_ext_ts = 0; | 744 | adapter->ptp_caps.n_ext_ts = 0; |
| 745 | adapter->ptp_caps.pps = 0; | 745 | adapter->ptp_caps.pps = 0; |
| 746 | adapter->ptp_caps.adjfreq = igb_ptp_adjfreq_82576; | 746 | adapter->ptp_caps.adjfreq = igb_ptp_adjfreq_82576; |
diff --git a/drivers/net/ethernet/intel/ixgbevf/ixgbevf_main.c b/drivers/net/ethernet/intel/ixgbevf/ixgbevf_main.c index c3db6cd69b68..2b6cb5ca48ee 100644 --- a/drivers/net/ethernet/intel/ixgbevf/ixgbevf_main.c +++ b/drivers/net/ethernet/intel/ixgbevf/ixgbevf_main.c | |||
| @@ -944,9 +944,17 @@ free_queue_irqs: | |||
| 944 | free_irq(adapter->msix_entries[vector].vector, | 944 | free_irq(adapter->msix_entries[vector].vector, |
| 945 | adapter->q_vector[vector]); | 945 | adapter->q_vector[vector]); |
| 946 | } | 946 | } |
| 947 | pci_disable_msix(adapter->pdev); | 947 | /* This failure is non-recoverable - it indicates the system is |
| 948 | kfree(adapter->msix_entries); | 948 | * out of MSIX vector resources and the VF driver cannot run |
| 949 | adapter->msix_entries = NULL; | 949 | * without them. Set the number of msix vectors to zero |
| 950 | * indicating that not enough can be allocated. The error | ||
| 951 | * will be returned to the user indicating device open failed. | ||
| 952 | * Any further attempts to force the driver to open will also | ||
| 953 | * fail. The only way to recover is to unload the driver and | ||
| 954 | * reload it again. If the system has recovered some MSIX | ||
| 955 | * vectors then it may succeed. | ||
| 956 | */ | ||
| 957 | adapter->num_msix_vectors = 0; | ||
| 950 | return err; | 958 | return err; |
| 951 | } | 959 | } |
| 952 | 960 | ||
| @@ -2572,6 +2580,15 @@ static int ixgbevf_open(struct net_device *netdev) | |||
| 2572 | struct ixgbe_hw *hw = &adapter->hw; | 2580 | struct ixgbe_hw *hw = &adapter->hw; |
| 2573 | int err; | 2581 | int err; |
| 2574 | 2582 | ||
| 2583 | /* A previous failure to open the device because of a lack of | ||
| 2584 | * available MSIX vector resources may have reset the number | ||
| 2585 | * of msix vectors variable to zero. The only way to recover | ||
| 2586 | * is to unload/reload the driver and hope that the system has | ||
| 2587 | * been able to recover some MSIX vector resources. | ||
| 2588 | */ | ||
| 2589 | if (!adapter->num_msix_vectors) | ||
| 2590 | return -ENOMEM; | ||
| 2591 | |||
| 2575 | /* disallow open during test */ | 2592 | /* disallow open during test */ |
| 2576 | if (test_bit(__IXGBEVF_TESTING, &adapter->state)) | 2593 | if (test_bit(__IXGBEVF_TESTING, &adapter->state)) |
| 2577 | return -EBUSY; | 2594 | return -EBUSY; |
| @@ -2628,7 +2645,6 @@ static int ixgbevf_open(struct net_device *netdev) | |||
| 2628 | 2645 | ||
| 2629 | err_req_irq: | 2646 | err_req_irq: |
| 2630 | ixgbevf_down(adapter); | 2647 | ixgbevf_down(adapter); |
| 2631 | ixgbevf_free_irq(adapter); | ||
| 2632 | err_setup_rx: | 2648 | err_setup_rx: |
| 2633 | ixgbevf_free_all_rx_resources(adapter); | 2649 | ixgbevf_free_all_rx_resources(adapter); |
| 2634 | err_setup_tx: | 2650 | err_setup_tx: |
diff --git a/drivers/net/ethernet/lantiq_etop.c b/drivers/net/ethernet/lantiq_etop.c index 6a2127489af7..bfdb06860397 100644 --- a/drivers/net/ethernet/lantiq_etop.c +++ b/drivers/net/ethernet/lantiq_etop.c | |||
| @@ -769,7 +769,7 @@ ltq_etop_probe(struct platform_device *pdev) | |||
| 769 | return 0; | 769 | return 0; |
| 770 | 770 | ||
| 771 | err_free: | 771 | err_free: |
| 772 | kfree(dev); | 772 | free_netdev(dev); |
| 773 | err_out: | 773 | err_out: |
| 774 | return err; | 774 | return err; |
| 775 | } | 775 | } |
diff --git a/drivers/net/ethernet/mellanox/mlx4/en_netdev.c b/drivers/net/ethernet/mellanox/mlx4/en_netdev.c index 995d4b6d5c1e..f278b10ef714 100644 --- a/drivers/net/ethernet/mellanox/mlx4/en_netdev.c +++ b/drivers/net/ethernet/mellanox/mlx4/en_netdev.c | |||
| @@ -1637,6 +1637,17 @@ void mlx4_en_stop_port(struct net_device *dev, int detach) | |||
| 1637 | /* Flush multicast filter */ | 1637 | /* Flush multicast filter */ |
| 1638 | mlx4_SET_MCAST_FLTR(mdev->dev, priv->port, 0, 1, MLX4_MCAST_CONFIG); | 1638 | mlx4_SET_MCAST_FLTR(mdev->dev, priv->port, 0, 1, MLX4_MCAST_CONFIG); |
| 1639 | 1639 | ||
| 1640 | /* Remove flow steering rules for the port*/ | ||
| 1641 | if (mdev->dev->caps.steering_mode == | ||
| 1642 | MLX4_STEERING_MODE_DEVICE_MANAGED) { | ||
| 1643 | ASSERT_RTNL(); | ||
| 1644 | list_for_each_entry_safe(flow, tmp_flow, | ||
| 1645 | &priv->ethtool_list, list) { | ||
| 1646 | mlx4_flow_detach(mdev->dev, flow->id); | ||
| 1647 | list_del(&flow->list); | ||
| 1648 | } | ||
| 1649 | } | ||
| 1650 | |||
| 1640 | mlx4_en_destroy_drop_qp(priv); | 1651 | mlx4_en_destroy_drop_qp(priv); |
| 1641 | 1652 | ||
| 1642 | /* Free TX Rings */ | 1653 | /* Free TX Rings */ |
| @@ -1657,17 +1668,6 @@ void mlx4_en_stop_port(struct net_device *dev, int detach) | |||
| 1657 | if (!(mdev->dev->caps.flags2 & MLX4_DEV_CAP_FLAGS2_REASSIGN_MAC_EN)) | 1668 | if (!(mdev->dev->caps.flags2 & MLX4_DEV_CAP_FLAGS2_REASSIGN_MAC_EN)) |
| 1658 | mdev->mac_removed[priv->port] = 1; | 1669 | mdev->mac_removed[priv->port] = 1; |
| 1659 | 1670 | ||
| 1660 | /* Remove flow steering rules for the port*/ | ||
| 1661 | if (mdev->dev->caps.steering_mode == | ||
| 1662 | MLX4_STEERING_MODE_DEVICE_MANAGED) { | ||
| 1663 | ASSERT_RTNL(); | ||
| 1664 | list_for_each_entry_safe(flow, tmp_flow, | ||
| 1665 | &priv->ethtool_list, list) { | ||
| 1666 | mlx4_flow_detach(mdev->dev, flow->id); | ||
| 1667 | list_del(&flow->list); | ||
| 1668 | } | ||
| 1669 | } | ||
| 1670 | |||
| 1671 | /* Free RX Rings */ | 1671 | /* Free RX Rings */ |
| 1672 | for (i = 0; i < priv->rx_ring_num; i++) { | 1672 | for (i = 0; i < priv->rx_ring_num; i++) { |
| 1673 | mlx4_en_deactivate_rx_ring(priv, &priv->rx_ring[i]); | 1673 | mlx4_en_deactivate_rx_ring(priv, &priv->rx_ring[i]); |
diff --git a/drivers/net/ethernet/mellanox/mlx4/eq.c b/drivers/net/ethernet/mellanox/mlx4/eq.c index 251ae2f93116..8e3123a1df88 100644 --- a/drivers/net/ethernet/mellanox/mlx4/eq.c +++ b/drivers/net/ethernet/mellanox/mlx4/eq.c | |||
| @@ -771,7 +771,7 @@ int mlx4_MAP_EQ_wrapper(struct mlx4_dev *dev, int slave, | |||
| 771 | struct mlx4_slave_event_eq_info *event_eq = | 771 | struct mlx4_slave_event_eq_info *event_eq = |
| 772 | priv->mfunc.master.slave_state[slave].event_eq; | 772 | priv->mfunc.master.slave_state[slave].event_eq; |
| 773 | u32 in_modifier = vhcr->in_modifier; | 773 | u32 in_modifier = vhcr->in_modifier; |
| 774 | u32 eqn = in_modifier & 0x1FF; | 774 | u32 eqn = in_modifier & 0x3FF; |
| 775 | u64 in_param = vhcr->in_param; | 775 | u64 in_param = vhcr->in_param; |
| 776 | int err = 0; | 776 | int err = 0; |
| 777 | int i; | 777 | int i; |
diff --git a/drivers/net/ethernet/mellanox/mlx4/resource_tracker.c b/drivers/net/ethernet/mellanox/mlx4/resource_tracker.c index 2995687f1aee..1391b52f443a 100644 --- a/drivers/net/ethernet/mellanox/mlx4/resource_tracker.c +++ b/drivers/net/ethernet/mellanox/mlx4/resource_tracker.c | |||
| @@ -99,6 +99,7 @@ struct res_qp { | |||
| 99 | struct list_head mcg_list; | 99 | struct list_head mcg_list; |
| 100 | spinlock_t mcg_spl; | 100 | spinlock_t mcg_spl; |
| 101 | int local_qpn; | 101 | int local_qpn; |
| 102 | atomic_t ref_count; | ||
| 102 | }; | 103 | }; |
| 103 | 104 | ||
| 104 | enum res_mtt_states { | 105 | enum res_mtt_states { |
| @@ -197,6 +198,7 @@ enum res_fs_rule_states { | |||
| 197 | 198 | ||
| 198 | struct res_fs_rule { | 199 | struct res_fs_rule { |
| 199 | struct res_common com; | 200 | struct res_common com; |
| 201 | int qpn; | ||
| 200 | }; | 202 | }; |
| 201 | 203 | ||
| 202 | static void *res_tracker_lookup(struct rb_root *root, u64 res_id) | 204 | static void *res_tracker_lookup(struct rb_root *root, u64 res_id) |
| @@ -355,7 +357,7 @@ static int mpt_mask(struct mlx4_dev *dev) | |||
| 355 | return dev->caps.num_mpts - 1; | 357 | return dev->caps.num_mpts - 1; |
| 356 | } | 358 | } |
| 357 | 359 | ||
| 358 | static void *find_res(struct mlx4_dev *dev, int res_id, | 360 | static void *find_res(struct mlx4_dev *dev, u64 res_id, |
| 359 | enum mlx4_resource type) | 361 | enum mlx4_resource type) |
| 360 | { | 362 | { |
| 361 | struct mlx4_priv *priv = mlx4_priv(dev); | 363 | struct mlx4_priv *priv = mlx4_priv(dev); |
| @@ -447,6 +449,7 @@ static struct res_common *alloc_qp_tr(int id) | |||
| 447 | ret->local_qpn = id; | 449 | ret->local_qpn = id; |
| 448 | INIT_LIST_HEAD(&ret->mcg_list); | 450 | INIT_LIST_HEAD(&ret->mcg_list); |
| 449 | spin_lock_init(&ret->mcg_spl); | 451 | spin_lock_init(&ret->mcg_spl); |
| 452 | atomic_set(&ret->ref_count, 0); | ||
| 450 | 453 | ||
| 451 | return &ret->com; | 454 | return &ret->com; |
| 452 | } | 455 | } |
| @@ -554,7 +557,7 @@ static struct res_common *alloc_xrcdn_tr(int id) | |||
| 554 | return &ret->com; | 557 | return &ret->com; |
| 555 | } | 558 | } |
| 556 | 559 | ||
| 557 | static struct res_common *alloc_fs_rule_tr(u64 id) | 560 | static struct res_common *alloc_fs_rule_tr(u64 id, int qpn) |
| 558 | { | 561 | { |
| 559 | struct res_fs_rule *ret; | 562 | struct res_fs_rule *ret; |
| 560 | 563 | ||
| @@ -564,7 +567,7 @@ static struct res_common *alloc_fs_rule_tr(u64 id) | |||
| 564 | 567 | ||
| 565 | ret->com.res_id = id; | 568 | ret->com.res_id = id; |
| 566 | ret->com.state = RES_FS_RULE_ALLOCATED; | 569 | ret->com.state = RES_FS_RULE_ALLOCATED; |
| 567 | 570 | ret->qpn = qpn; | |
| 568 | return &ret->com; | 571 | return &ret->com; |
| 569 | } | 572 | } |
| 570 | 573 | ||
| @@ -602,7 +605,7 @@ static struct res_common *alloc_tr(u64 id, enum mlx4_resource type, int slave, | |||
| 602 | ret = alloc_xrcdn_tr(id); | 605 | ret = alloc_xrcdn_tr(id); |
| 603 | break; | 606 | break; |
| 604 | case RES_FS_RULE: | 607 | case RES_FS_RULE: |
| 605 | ret = alloc_fs_rule_tr(id); | 608 | ret = alloc_fs_rule_tr(id, extra); |
| 606 | break; | 609 | break; |
| 607 | default: | 610 | default: |
| 608 | return NULL; | 611 | return NULL; |
| @@ -671,10 +674,14 @@ undo: | |||
| 671 | 674 | ||
| 672 | static int remove_qp_ok(struct res_qp *res) | 675 | static int remove_qp_ok(struct res_qp *res) |
| 673 | { | 676 | { |
| 674 | if (res->com.state == RES_QP_BUSY) | 677 | if (res->com.state == RES_QP_BUSY || atomic_read(&res->ref_count) || |
| 678 | !list_empty(&res->mcg_list)) { | ||
| 679 | pr_err("resource tracker: fail to remove qp, state %d, ref_count %d\n", | ||
| 680 | res->com.state, atomic_read(&res->ref_count)); | ||
| 675 | return -EBUSY; | 681 | return -EBUSY; |
| 676 | else if (res->com.state != RES_QP_RESERVED) | 682 | } else if (res->com.state != RES_QP_RESERVED) { |
| 677 | return -EPERM; | 683 | return -EPERM; |
| 684 | } | ||
| 678 | 685 | ||
| 679 | return 0; | 686 | return 0; |
| 680 | } | 687 | } |
| @@ -3124,6 +3131,7 @@ int mlx4_QP_FLOW_STEERING_ATTACH_wrapper(struct mlx4_dev *dev, int slave, | |||
| 3124 | struct list_head *rlist = &tracker->slave_list[slave].res_list[RES_MAC]; | 3131 | struct list_head *rlist = &tracker->slave_list[slave].res_list[RES_MAC]; |
| 3125 | int err; | 3132 | int err; |
| 3126 | int qpn; | 3133 | int qpn; |
| 3134 | struct res_qp *rqp; | ||
| 3127 | struct mlx4_net_trans_rule_hw_ctrl *ctrl; | 3135 | struct mlx4_net_trans_rule_hw_ctrl *ctrl; |
| 3128 | struct _rule_hw *rule_header; | 3136 | struct _rule_hw *rule_header; |
| 3129 | int header_id; | 3137 | int header_id; |
| @@ -3134,7 +3142,7 @@ int mlx4_QP_FLOW_STEERING_ATTACH_wrapper(struct mlx4_dev *dev, int slave, | |||
| 3134 | 3142 | ||
| 3135 | ctrl = (struct mlx4_net_trans_rule_hw_ctrl *)inbox->buf; | 3143 | ctrl = (struct mlx4_net_trans_rule_hw_ctrl *)inbox->buf; |
| 3136 | qpn = be32_to_cpu(ctrl->qpn) & 0xffffff; | 3144 | qpn = be32_to_cpu(ctrl->qpn) & 0xffffff; |
| 3137 | err = get_res(dev, slave, qpn, RES_QP, NULL); | 3145 | err = get_res(dev, slave, qpn, RES_QP, &rqp); |
| 3138 | if (err) { | 3146 | if (err) { |
| 3139 | pr_err("Steering rule with qpn 0x%x rejected.\n", qpn); | 3147 | pr_err("Steering rule with qpn 0x%x rejected.\n", qpn); |
| 3140 | return err; | 3148 | return err; |
| @@ -3175,14 +3183,16 @@ int mlx4_QP_FLOW_STEERING_ATTACH_wrapper(struct mlx4_dev *dev, int slave, | |||
| 3175 | if (err) | 3183 | if (err) |
| 3176 | goto err_put; | 3184 | goto err_put; |
| 3177 | 3185 | ||
| 3178 | err = add_res_range(dev, slave, vhcr->out_param, 1, RES_FS_RULE, 0); | 3186 | err = add_res_range(dev, slave, vhcr->out_param, 1, RES_FS_RULE, qpn); |
| 3179 | if (err) { | 3187 | if (err) { |
| 3180 | mlx4_err(dev, "Fail to add flow steering resources.\n "); | 3188 | mlx4_err(dev, "Fail to add flow steering resources.\n "); |
| 3181 | /* detach rule*/ | 3189 | /* detach rule*/ |
| 3182 | mlx4_cmd(dev, vhcr->out_param, 0, 0, | 3190 | mlx4_cmd(dev, vhcr->out_param, 0, 0, |
| 3183 | MLX4_QP_FLOW_STEERING_DETACH, MLX4_CMD_TIME_CLASS_A, | 3191 | MLX4_QP_FLOW_STEERING_DETACH, MLX4_CMD_TIME_CLASS_A, |
| 3184 | MLX4_CMD_NATIVE); | 3192 | MLX4_CMD_NATIVE); |
| 3193 | goto err_put; | ||
| 3185 | } | 3194 | } |
| 3195 | atomic_inc(&rqp->ref_count); | ||
| 3186 | err_put: | 3196 | err_put: |
| 3187 | put_res(dev, slave, qpn, RES_QP); | 3197 | put_res(dev, slave, qpn, RES_QP); |
| 3188 | return err; | 3198 | return err; |
| @@ -3195,20 +3205,35 @@ int mlx4_QP_FLOW_STEERING_DETACH_wrapper(struct mlx4_dev *dev, int slave, | |||
| 3195 | struct mlx4_cmd_info *cmd) | 3205 | struct mlx4_cmd_info *cmd) |
| 3196 | { | 3206 | { |
| 3197 | int err; | 3207 | int err; |
| 3208 | struct res_qp *rqp; | ||
| 3209 | struct res_fs_rule *rrule; | ||
| 3198 | 3210 | ||
| 3199 | if (dev->caps.steering_mode != | 3211 | if (dev->caps.steering_mode != |
| 3200 | MLX4_STEERING_MODE_DEVICE_MANAGED) | 3212 | MLX4_STEERING_MODE_DEVICE_MANAGED) |
| 3201 | return -EOPNOTSUPP; | 3213 | return -EOPNOTSUPP; |
| 3202 | 3214 | ||
| 3215 | err = get_res(dev, slave, vhcr->in_param, RES_FS_RULE, &rrule); | ||
| 3216 | if (err) | ||
| 3217 | return err; | ||
| 3218 | /* Release the rule form busy state before removal */ | ||
| 3219 | put_res(dev, slave, vhcr->in_param, RES_FS_RULE); | ||
| 3220 | err = get_res(dev, slave, rrule->qpn, RES_QP, &rqp); | ||
| 3221 | if (err) | ||
| 3222 | return err; | ||
| 3223 | |||
| 3203 | err = rem_res_range(dev, slave, vhcr->in_param, 1, RES_FS_RULE, 0); | 3224 | err = rem_res_range(dev, slave, vhcr->in_param, 1, RES_FS_RULE, 0); |
| 3204 | if (err) { | 3225 | if (err) { |
| 3205 | mlx4_err(dev, "Fail to remove flow steering resources.\n "); | 3226 | mlx4_err(dev, "Fail to remove flow steering resources.\n "); |
| 3206 | return err; | 3227 | goto out; |
| 3207 | } | 3228 | } |
| 3208 | 3229 | ||
| 3209 | err = mlx4_cmd(dev, vhcr->in_param, 0, 0, | 3230 | err = mlx4_cmd(dev, vhcr->in_param, 0, 0, |
| 3210 | MLX4_QP_FLOW_STEERING_DETACH, MLX4_CMD_TIME_CLASS_A, | 3231 | MLX4_QP_FLOW_STEERING_DETACH, MLX4_CMD_TIME_CLASS_A, |
| 3211 | MLX4_CMD_NATIVE); | 3232 | MLX4_CMD_NATIVE); |
| 3233 | if (!err) | ||
| 3234 | atomic_dec(&rqp->ref_count); | ||
| 3235 | out: | ||
| 3236 | put_res(dev, slave, rrule->qpn, RES_QP); | ||
| 3212 | return err; | 3237 | return err; |
| 3213 | } | 3238 | } |
| 3214 | 3239 | ||
| @@ -3806,6 +3831,7 @@ void mlx4_delete_all_resources_for_slave(struct mlx4_dev *dev, int slave) | |||
| 3806 | mutex_lock(&priv->mfunc.master.res_tracker.slave_list[slave].mutex); | 3831 | mutex_lock(&priv->mfunc.master.res_tracker.slave_list[slave].mutex); |
| 3807 | /*VLAN*/ | 3832 | /*VLAN*/ |
| 3808 | rem_slave_macs(dev, slave); | 3833 | rem_slave_macs(dev, slave); |
| 3834 | rem_slave_fs_rule(dev, slave); | ||
| 3809 | rem_slave_qps(dev, slave); | 3835 | rem_slave_qps(dev, slave); |
| 3810 | rem_slave_srqs(dev, slave); | 3836 | rem_slave_srqs(dev, slave); |
| 3811 | rem_slave_cqs(dev, slave); | 3837 | rem_slave_cqs(dev, slave); |
| @@ -3814,6 +3840,5 @@ void mlx4_delete_all_resources_for_slave(struct mlx4_dev *dev, int slave) | |||
| 3814 | rem_slave_mtts(dev, slave); | 3840 | rem_slave_mtts(dev, slave); |
| 3815 | rem_slave_counters(dev, slave); | 3841 | rem_slave_counters(dev, slave); |
| 3816 | rem_slave_xrcdns(dev, slave); | 3842 | rem_slave_xrcdns(dev, slave); |
| 3817 | rem_slave_fs_rule(dev, slave); | ||
| 3818 | mutex_unlock(&priv->mfunc.master.res_tracker.slave_list[slave].mutex); | 3843 | mutex_unlock(&priv->mfunc.master.res_tracker.slave_list[slave].mutex); |
| 3819 | } | 3844 | } |
diff --git a/drivers/net/ethernet/nxp/lpc_eth.c b/drivers/net/ethernet/nxp/lpc_eth.c index c4122c86f829..efa29b712d5f 100644 --- a/drivers/net/ethernet/nxp/lpc_eth.c +++ b/drivers/net/ethernet/nxp/lpc_eth.c | |||
| @@ -1472,7 +1472,8 @@ static int lpc_eth_drv_probe(struct platform_device *pdev) | |||
| 1472 | } | 1472 | } |
| 1473 | platform_set_drvdata(pdev, ndev); | 1473 | platform_set_drvdata(pdev, ndev); |
| 1474 | 1474 | ||
| 1475 | if (lpc_mii_init(pldat) != 0) | 1475 | ret = lpc_mii_init(pldat); |
| 1476 | if (ret) | ||
| 1476 | goto err_out_unregister_netdev; | 1477 | goto err_out_unregister_netdev; |
| 1477 | 1478 | ||
| 1478 | netdev_info(ndev, "LPC mac at 0x%08x irq %d\n", | 1479 | netdev_info(ndev, "LPC mac at 0x%08x irq %d\n", |
diff --git a/drivers/net/ethernet/oki-semi/pch_gbe/pch_gbe_main.c b/drivers/net/ethernet/oki-semi/pch_gbe/pch_gbe_main.c index 39ab4d09faaa..73ce7dd6b954 100644 --- a/drivers/net/ethernet/oki-semi/pch_gbe/pch_gbe_main.c +++ b/drivers/net/ethernet/oki-semi/pch_gbe/pch_gbe_main.c | |||
| @@ -1726,9 +1726,9 @@ pch_gbe_clean_rx(struct pch_gbe_adapter *adapter, | |||
| 1726 | 1726 | ||
| 1727 | skb->protocol = eth_type_trans(skb, netdev); | 1727 | skb->protocol = eth_type_trans(skb, netdev); |
| 1728 | if (tcp_ip_status & PCH_GBE_RXD_ACC_STAT_TCPIPOK) | 1728 | if (tcp_ip_status & PCH_GBE_RXD_ACC_STAT_TCPIPOK) |
| 1729 | skb->ip_summed = CHECKSUM_NONE; | ||
| 1730 | else | ||
| 1731 | skb->ip_summed = CHECKSUM_UNNECESSARY; | 1729 | skb->ip_summed = CHECKSUM_UNNECESSARY; |
| 1730 | else | ||
| 1731 | skb->ip_summed = CHECKSUM_NONE; | ||
| 1732 | 1732 | ||
| 1733 | napi_gro_receive(&adapter->napi, skb); | 1733 | napi_gro_receive(&adapter->napi, skb); |
| 1734 | (*work_done)++; | 1734 | (*work_done)++; |
diff --git a/drivers/net/ethernet/renesas/sh_eth.c b/drivers/net/ethernet/renesas/sh_eth.c index 33e96176e4d8..bf5e3cf97c4d 100644 --- a/drivers/net/ethernet/renesas/sh_eth.c +++ b/drivers/net/ethernet/renesas/sh_eth.c | |||
| @@ -2220,6 +2220,7 @@ static void sh_eth_tsu_init(struct sh_eth_private *mdp) | |||
| 2220 | /* MDIO bus release function */ | 2220 | /* MDIO bus release function */ |
| 2221 | static int sh_mdio_release(struct net_device *ndev) | 2221 | static int sh_mdio_release(struct net_device *ndev) |
| 2222 | { | 2222 | { |
| 2223 | struct sh_eth_private *mdp = netdev_priv(ndev); | ||
| 2223 | struct mii_bus *bus = dev_get_drvdata(&ndev->dev); | 2224 | struct mii_bus *bus = dev_get_drvdata(&ndev->dev); |
| 2224 | 2225 | ||
| 2225 | /* unregister mdio bus */ | 2226 | /* unregister mdio bus */ |
| @@ -2234,6 +2235,9 @@ static int sh_mdio_release(struct net_device *ndev) | |||
| 2234 | /* free bitbang info */ | 2235 | /* free bitbang info */ |
| 2235 | free_mdio_bitbang(bus); | 2236 | free_mdio_bitbang(bus); |
| 2236 | 2237 | ||
| 2238 | /* free bitbang memory */ | ||
| 2239 | kfree(mdp->bitbang); | ||
| 2240 | |||
| 2237 | return 0; | 2241 | return 0; |
| 2238 | } | 2242 | } |
| 2239 | 2243 | ||
| @@ -2262,6 +2266,7 @@ static int sh_mdio_init(struct net_device *ndev, int id, | |||
| 2262 | bitbang->ctrl.ops = &bb_ops; | 2266 | bitbang->ctrl.ops = &bb_ops; |
| 2263 | 2267 | ||
| 2264 | /* MII controller setting */ | 2268 | /* MII controller setting */ |
| 2269 | mdp->bitbang = bitbang; | ||
| 2265 | mdp->mii_bus = alloc_mdio_bitbang(&bitbang->ctrl); | 2270 | mdp->mii_bus = alloc_mdio_bitbang(&bitbang->ctrl); |
| 2266 | if (!mdp->mii_bus) { | 2271 | if (!mdp->mii_bus) { |
| 2267 | ret = -ENOMEM; | 2272 | ret = -ENOMEM; |
| @@ -2441,6 +2446,11 @@ static int sh_eth_drv_probe(struct platform_device *pdev) | |||
| 2441 | } | 2446 | } |
| 2442 | mdp->tsu_addr = ioremap(rtsu->start, | 2447 | mdp->tsu_addr = ioremap(rtsu->start, |
| 2443 | resource_size(rtsu)); | 2448 | resource_size(rtsu)); |
| 2449 | if (mdp->tsu_addr == NULL) { | ||
| 2450 | ret = -ENOMEM; | ||
| 2451 | dev_err(&pdev->dev, "TSU ioremap failed.\n"); | ||
| 2452 | goto out_release; | ||
| 2453 | } | ||
| 2444 | mdp->port = devno % 2; | 2454 | mdp->port = devno % 2; |
| 2445 | ndev->features = NETIF_F_HW_VLAN_FILTER; | 2455 | ndev->features = NETIF_F_HW_VLAN_FILTER; |
| 2446 | } | 2456 | } |
diff --git a/drivers/net/ethernet/renesas/sh_eth.h b/drivers/net/ethernet/renesas/sh_eth.h index bae84fd2e73a..e6655678458e 100644 --- a/drivers/net/ethernet/renesas/sh_eth.h +++ b/drivers/net/ethernet/renesas/sh_eth.h | |||
| @@ -705,6 +705,7 @@ struct sh_eth_private { | |||
| 705 | const u16 *reg_offset; | 705 | const u16 *reg_offset; |
| 706 | void __iomem *addr; | 706 | void __iomem *addr; |
| 707 | void __iomem *tsu_addr; | 707 | void __iomem *tsu_addr; |
| 708 | struct bb_info *bitbang; | ||
| 708 | u32 num_rx_ring; | 709 | u32 num_rx_ring; |
| 709 | u32 num_tx_ring; | 710 | u32 num_tx_ring; |
| 710 | dma_addr_t rx_desc_dma; | 711 | dma_addr_t rx_desc_dma; |
diff --git a/drivers/net/ethernet/ti/cpsw.c b/drivers/net/ethernet/ti/cpsw.c index 75c48558e6fd..df32a090d08e 100644 --- a/drivers/net/ethernet/ti/cpsw.c +++ b/drivers/net/ethernet/ti/cpsw.c | |||
| @@ -1364,7 +1364,7 @@ static int cpsw_probe_dt(struct cpsw_platform_data *data, | |||
| 1364 | struct platform_device *mdio; | 1364 | struct platform_device *mdio; |
| 1365 | 1365 | ||
| 1366 | parp = of_get_property(slave_node, "phy_id", &lenp); | 1366 | parp = of_get_property(slave_node, "phy_id", &lenp); |
| 1367 | if ((parp == NULL) && (lenp != (sizeof(void *) * 2))) { | 1367 | if ((parp == NULL) || (lenp != (sizeof(void *) * 2))) { |
| 1368 | pr_err("Missing slave[%d] phy_id property\n", i); | 1368 | pr_err("Missing slave[%d] phy_id property\n", i); |
| 1369 | ret = -EINVAL; | 1369 | ret = -EINVAL; |
| 1370 | goto error_ret; | 1370 | goto error_ret; |
diff --git a/drivers/net/wireless/ath/ath9k/ar9003_calib.c b/drivers/net/wireless/ath/ath9k/ar9003_calib.c index 4cc13940c895..f76c3ca07a45 100644 --- a/drivers/net/wireless/ath/ath9k/ar9003_calib.c +++ b/drivers/net/wireless/ath/ath9k/ar9003_calib.c | |||
| @@ -1023,6 +1023,7 @@ static bool ar9003_hw_init_cal(struct ath_hw *ah, | |||
| 1023 | AR_PHY_AGC_CONTROL_FLTR_CAL | | 1023 | AR_PHY_AGC_CONTROL_FLTR_CAL | |
| 1024 | AR_PHY_AGC_CONTROL_PKDET_CAL; | 1024 | AR_PHY_AGC_CONTROL_PKDET_CAL; |
| 1025 | 1025 | ||
| 1026 | /* Use chip chainmask only for calibration */ | ||
| 1026 | ar9003_hw_set_chain_masks(ah, ah->caps.rx_chainmask, ah->caps.tx_chainmask); | 1027 | ar9003_hw_set_chain_masks(ah, ah->caps.rx_chainmask, ah->caps.tx_chainmask); |
| 1027 | 1028 | ||
| 1028 | if (rtt) { | 1029 | if (rtt) { |
| @@ -1150,6 +1151,9 @@ skip_tx_iqcal: | |||
| 1150 | ar9003_hw_rtt_disable(ah); | 1151 | ar9003_hw_rtt_disable(ah); |
| 1151 | } | 1152 | } |
| 1152 | 1153 | ||
| 1154 | /* Revert chainmask to runtime parameters */ | ||
| 1155 | ar9003_hw_set_chain_masks(ah, ah->rxchainmask, ah->txchainmask); | ||
| 1156 | |||
| 1153 | /* Initialize list pointers */ | 1157 | /* Initialize list pointers */ |
| 1154 | ah->cal_list = ah->cal_list_last = ah->cal_list_curr = NULL; | 1158 | ah->cal_list = ah->cal_list_last = ah->cal_list_curr = NULL; |
| 1155 | 1159 | ||
diff --git a/drivers/net/wireless/ath/ath9k/link.c b/drivers/net/wireless/ath/ath9k/link.c index ade3afb21f91..39c84ecf6a42 100644 --- a/drivers/net/wireless/ath/ath9k/link.c +++ b/drivers/net/wireless/ath/ath9k/link.c | |||
| @@ -28,21 +28,21 @@ void ath_tx_complete_poll_work(struct work_struct *work) | |||
| 28 | int i; | 28 | int i; |
| 29 | bool needreset = false; | 29 | bool needreset = false; |
| 30 | 30 | ||
| 31 | for (i = 0; i < ATH9K_NUM_TX_QUEUES; i++) | 31 | for (i = 0; i < IEEE80211_NUM_ACS; i++) { |
| 32 | if (ATH_TXQ_SETUP(sc, i)) { | 32 | txq = sc->tx.txq_map[i]; |
| 33 | txq = &sc->tx.txq[i]; | 33 | |
| 34 | ath_txq_lock(sc, txq); | 34 | ath_txq_lock(sc, txq); |
| 35 | if (txq->axq_depth) { | 35 | if (txq->axq_depth) { |
| 36 | if (txq->axq_tx_inprogress) { | 36 | if (txq->axq_tx_inprogress) { |
| 37 | needreset = true; | 37 | needreset = true; |
| 38 | ath_txq_unlock(sc, txq); | 38 | ath_txq_unlock(sc, txq); |
| 39 | break; | 39 | break; |
| 40 | } else { | 40 | } else { |
| 41 | txq->axq_tx_inprogress = true; | 41 | txq->axq_tx_inprogress = true; |
| 42 | } | ||
| 43 | } | 42 | } |
| 44 | ath_txq_unlock_complete(sc, txq); | ||
| 45 | } | 43 | } |
| 44 | ath_txq_unlock_complete(sc, txq); | ||
| 45 | } | ||
| 46 | 46 | ||
| 47 | if (needreset) { | 47 | if (needreset) { |
| 48 | ath_dbg(ath9k_hw_common(sc->sc_ah), RESET, | 48 | ath_dbg(ath9k_hw_common(sc->sc_ah), RESET, |
diff --git a/drivers/net/wireless/iwlegacy/3945-mac.c b/drivers/net/wireless/iwlegacy/3945-mac.c index 3630a41df50d..c353b5f19c8c 100644 --- a/drivers/net/wireless/iwlegacy/3945-mac.c +++ b/drivers/net/wireless/iwlegacy/3945-mac.c | |||
| @@ -475,6 +475,7 @@ il3945_tx_skb(struct il_priv *il, | |||
| 475 | dma_addr_t txcmd_phys; | 475 | dma_addr_t txcmd_phys; |
| 476 | int txq_id = skb_get_queue_mapping(skb); | 476 | int txq_id = skb_get_queue_mapping(skb); |
| 477 | u16 len, idx, hdr_len; | 477 | u16 len, idx, hdr_len; |
| 478 | u16 firstlen, secondlen; | ||
| 478 | u8 id; | 479 | u8 id; |
| 479 | u8 unicast; | 480 | u8 unicast; |
| 480 | u8 sta_id; | 481 | u8 sta_id; |
| @@ -589,21 +590,22 @@ il3945_tx_skb(struct il_priv *il, | |||
| 589 | len = | 590 | len = |
| 590 | sizeof(struct il3945_tx_cmd) + sizeof(struct il_cmd_header) + | 591 | sizeof(struct il3945_tx_cmd) + sizeof(struct il_cmd_header) + |
| 591 | hdr_len; | 592 | hdr_len; |
| 592 | len = (len + 3) & ~3; | 593 | firstlen = (len + 3) & ~3; |
| 593 | 594 | ||
| 594 | /* Physical address of this Tx command's header (not MAC header!), | 595 | /* Physical address of this Tx command's header (not MAC header!), |
| 595 | * within command buffer array. */ | 596 | * within command buffer array. */ |
| 596 | txcmd_phys = | 597 | txcmd_phys = |
| 597 | pci_map_single(il->pci_dev, &out_cmd->hdr, len, PCI_DMA_TODEVICE); | 598 | pci_map_single(il->pci_dev, &out_cmd->hdr, firstlen, |
| 599 | PCI_DMA_TODEVICE); | ||
| 598 | if (unlikely(pci_dma_mapping_error(il->pci_dev, txcmd_phys))) | 600 | if (unlikely(pci_dma_mapping_error(il->pci_dev, txcmd_phys))) |
| 599 | goto drop_unlock; | 601 | goto drop_unlock; |
| 600 | 602 | ||
| 601 | /* Set up TFD's 2nd entry to point directly to remainder of skb, | 603 | /* Set up TFD's 2nd entry to point directly to remainder of skb, |
| 602 | * if any (802.11 null frames have no payload). */ | 604 | * if any (802.11 null frames have no payload). */ |
| 603 | len = skb->len - hdr_len; | 605 | secondlen = skb->len - hdr_len; |
| 604 | if (len) { | 606 | if (secondlen > 0) { |
| 605 | phys_addr = | 607 | phys_addr = |
| 606 | pci_map_single(il->pci_dev, skb->data + hdr_len, len, | 608 | pci_map_single(il->pci_dev, skb->data + hdr_len, secondlen, |
| 607 | PCI_DMA_TODEVICE); | 609 | PCI_DMA_TODEVICE); |
| 608 | if (unlikely(pci_dma_mapping_error(il->pci_dev, phys_addr))) | 610 | if (unlikely(pci_dma_mapping_error(il->pci_dev, phys_addr))) |
| 609 | goto drop_unlock; | 611 | goto drop_unlock; |
| @@ -611,12 +613,12 @@ il3945_tx_skb(struct il_priv *il, | |||
| 611 | 613 | ||
| 612 | /* Add buffer containing Tx command and MAC(!) header to TFD's | 614 | /* Add buffer containing Tx command and MAC(!) header to TFD's |
| 613 | * first entry */ | 615 | * first entry */ |
| 614 | il->ops->txq_attach_buf_to_tfd(il, txq, txcmd_phys, len, 1, 0); | 616 | il->ops->txq_attach_buf_to_tfd(il, txq, txcmd_phys, firstlen, 1, 0); |
| 615 | dma_unmap_addr_set(out_meta, mapping, txcmd_phys); | 617 | dma_unmap_addr_set(out_meta, mapping, txcmd_phys); |
| 616 | dma_unmap_len_set(out_meta, len, len); | 618 | dma_unmap_len_set(out_meta, len, firstlen); |
| 617 | if (len) | 619 | if (secondlen > 0) |
| 618 | il->ops->txq_attach_buf_to_tfd(il, txq, phys_addr, len, 0, | 620 | il->ops->txq_attach_buf_to_tfd(il, txq, phys_addr, secondlen, 0, |
| 619 | U32_PAD(len)); | 621 | U32_PAD(secondlen)); |
| 620 | 622 | ||
| 621 | if (!ieee80211_has_morefrags(hdr->frame_control)) { | 623 | if (!ieee80211_has_morefrags(hdr->frame_control)) { |
| 622 | txq->need_update = 1; | 624 | txq->need_update = 1; |
diff --git a/drivers/net/wireless/mwifiex/cmdevt.c b/drivers/net/wireless/mwifiex/cmdevt.c index 20a6c5555873..b5c8b962ce12 100644 --- a/drivers/net/wireless/mwifiex/cmdevt.c +++ b/drivers/net/wireless/mwifiex/cmdevt.c | |||
| @@ -157,6 +157,20 @@ static int mwifiex_dnld_cmd_to_fw(struct mwifiex_private *priv, | |||
| 157 | return -1; | 157 | return -1; |
| 158 | } | 158 | } |
| 159 | 159 | ||
| 160 | cmd_code = le16_to_cpu(host_cmd->command); | ||
| 161 | cmd_size = le16_to_cpu(host_cmd->size); | ||
| 162 | |||
| 163 | if (adapter->hw_status == MWIFIEX_HW_STATUS_RESET && | ||
| 164 | cmd_code != HostCmd_CMD_FUNC_SHUTDOWN && | ||
| 165 | cmd_code != HostCmd_CMD_FUNC_INIT) { | ||
| 166 | dev_err(adapter->dev, | ||
| 167 | "DNLD_CMD: FW in reset state, ignore cmd %#x\n", | ||
| 168 | cmd_code); | ||
| 169 | mwifiex_complete_cmd(adapter, cmd_node); | ||
| 170 | mwifiex_insert_cmd_to_free_q(adapter, cmd_node); | ||
| 171 | return -1; | ||
| 172 | } | ||
| 173 | |||
| 160 | /* Set command sequence number */ | 174 | /* Set command sequence number */ |
| 161 | adapter->seq_num++; | 175 | adapter->seq_num++; |
| 162 | host_cmd->seq_num = cpu_to_le16(HostCmd_SET_SEQ_NO_BSS_INFO | 176 | host_cmd->seq_num = cpu_to_le16(HostCmd_SET_SEQ_NO_BSS_INFO |
| @@ -168,9 +182,6 @@ static int mwifiex_dnld_cmd_to_fw(struct mwifiex_private *priv, | |||
| 168 | adapter->curr_cmd = cmd_node; | 182 | adapter->curr_cmd = cmd_node; |
| 169 | spin_unlock_irqrestore(&adapter->mwifiex_cmd_lock, flags); | 183 | spin_unlock_irqrestore(&adapter->mwifiex_cmd_lock, flags); |
| 170 | 184 | ||
| 171 | cmd_code = le16_to_cpu(host_cmd->command); | ||
| 172 | cmd_size = le16_to_cpu(host_cmd->size); | ||
| 173 | |||
| 174 | /* Adjust skb length */ | 185 | /* Adjust skb length */ |
| 175 | if (cmd_node->cmd_skb->len > cmd_size) | 186 | if (cmd_node->cmd_skb->len > cmd_size) |
| 176 | /* | 187 | /* |
| @@ -484,8 +495,6 @@ int mwifiex_send_cmd_sync(struct mwifiex_private *priv, uint16_t cmd_no, | |||
| 484 | 495 | ||
| 485 | ret = mwifiex_send_cmd_async(priv, cmd_no, cmd_action, cmd_oid, | 496 | ret = mwifiex_send_cmd_async(priv, cmd_no, cmd_action, cmd_oid, |
| 486 | data_buf); | 497 | data_buf); |
| 487 | if (!ret) | ||
| 488 | ret = mwifiex_wait_queue_complete(adapter); | ||
| 489 | 498 | ||
| 490 | return ret; | 499 | return ret; |
| 491 | } | 500 | } |
| @@ -588,9 +597,10 @@ int mwifiex_send_cmd_async(struct mwifiex_private *priv, uint16_t cmd_no, | |||
| 588 | if (cmd_no == HostCmd_CMD_802_11_SCAN) { | 597 | if (cmd_no == HostCmd_CMD_802_11_SCAN) { |
| 589 | mwifiex_queue_scan_cmd(priv, cmd_node); | 598 | mwifiex_queue_scan_cmd(priv, cmd_node); |
| 590 | } else { | 599 | } else { |
| 591 | adapter->cmd_queued = cmd_node; | ||
| 592 | mwifiex_insert_cmd_to_pending_q(adapter, cmd_node, true); | 600 | mwifiex_insert_cmd_to_pending_q(adapter, cmd_node, true); |
| 593 | queue_work(adapter->workqueue, &adapter->main_work); | 601 | queue_work(adapter->workqueue, &adapter->main_work); |
| 602 | if (cmd_node->wait_q_enabled) | ||
| 603 | ret = mwifiex_wait_queue_complete(adapter, cmd_node); | ||
| 594 | } | 604 | } |
| 595 | 605 | ||
| 596 | return ret; | 606 | return ret; |
diff --git a/drivers/net/wireless/mwifiex/init.c b/drivers/net/wireless/mwifiex/init.c index e38aa9b3663d..0ff4c37ab42a 100644 --- a/drivers/net/wireless/mwifiex/init.c +++ b/drivers/net/wireless/mwifiex/init.c | |||
| @@ -709,6 +709,14 @@ mwifiex_shutdown_drv(struct mwifiex_adapter *adapter) | |||
| 709 | return ret; | 709 | return ret; |
| 710 | } | 710 | } |
| 711 | 711 | ||
| 712 | /* cancel current command */ | ||
| 713 | if (adapter->curr_cmd) { | ||
| 714 | dev_warn(adapter->dev, "curr_cmd is still in processing\n"); | ||
| 715 | del_timer(&adapter->cmd_timer); | ||
| 716 | mwifiex_insert_cmd_to_free_q(adapter, adapter->curr_cmd); | ||
| 717 | adapter->curr_cmd = NULL; | ||
| 718 | } | ||
| 719 | |||
| 712 | /* shut down mwifiex */ | 720 | /* shut down mwifiex */ |
| 713 | dev_dbg(adapter->dev, "info: shutdown mwifiex...\n"); | 721 | dev_dbg(adapter->dev, "info: shutdown mwifiex...\n"); |
| 714 | 722 | ||
diff --git a/drivers/net/wireless/mwifiex/main.h b/drivers/net/wireless/mwifiex/main.h index 553adfb0aa81..7035ade9af74 100644 --- a/drivers/net/wireless/mwifiex/main.h +++ b/drivers/net/wireless/mwifiex/main.h | |||
| @@ -723,7 +723,6 @@ struct mwifiex_adapter { | |||
| 723 | u16 cmd_wait_q_required; | 723 | u16 cmd_wait_q_required; |
| 724 | struct mwifiex_wait_queue cmd_wait_q; | 724 | struct mwifiex_wait_queue cmd_wait_q; |
| 725 | u8 scan_wait_q_woken; | 725 | u8 scan_wait_q_woken; |
| 726 | struct cmd_ctrl_node *cmd_queued; | ||
| 727 | spinlock_t queue_lock; /* lock for tx queues */ | 726 | spinlock_t queue_lock; /* lock for tx queues */ |
| 728 | struct completion fw_load; | 727 | struct completion fw_load; |
| 729 | u8 country_code[IEEE80211_COUNTRY_STRING_LEN]; | 728 | u8 country_code[IEEE80211_COUNTRY_STRING_LEN]; |
| @@ -1018,7 +1017,8 @@ int mwifiex_request_set_multicast_list(struct mwifiex_private *priv, | |||
| 1018 | struct mwifiex_multicast_list *mcast_list); | 1017 | struct mwifiex_multicast_list *mcast_list); |
| 1019 | int mwifiex_copy_mcast_addr(struct mwifiex_multicast_list *mlist, | 1018 | int mwifiex_copy_mcast_addr(struct mwifiex_multicast_list *mlist, |
| 1020 | struct net_device *dev); | 1019 | struct net_device *dev); |
| 1021 | int mwifiex_wait_queue_complete(struct mwifiex_adapter *adapter); | 1020 | int mwifiex_wait_queue_complete(struct mwifiex_adapter *adapter, |
| 1021 | struct cmd_ctrl_node *cmd_queued); | ||
| 1022 | int mwifiex_bss_start(struct mwifiex_private *priv, struct cfg80211_bss *bss, | 1022 | int mwifiex_bss_start(struct mwifiex_private *priv, struct cfg80211_bss *bss, |
| 1023 | struct cfg80211_ssid *req_ssid); | 1023 | struct cfg80211_ssid *req_ssid); |
| 1024 | int mwifiex_cancel_hs(struct mwifiex_private *priv, int cmd_type); | 1024 | int mwifiex_cancel_hs(struct mwifiex_private *priv, int cmd_type); |
diff --git a/drivers/net/wireless/mwifiex/scan.c b/drivers/net/wireless/mwifiex/scan.c index bb60c2754a97..d215b4d3c51b 100644 --- a/drivers/net/wireless/mwifiex/scan.c +++ b/drivers/net/wireless/mwifiex/scan.c | |||
| @@ -1388,10 +1388,13 @@ int mwifiex_scan_networks(struct mwifiex_private *priv, | |||
| 1388 | list_del(&cmd_node->list); | 1388 | list_del(&cmd_node->list); |
| 1389 | spin_unlock_irqrestore(&adapter->scan_pending_q_lock, | 1389 | spin_unlock_irqrestore(&adapter->scan_pending_q_lock, |
| 1390 | flags); | 1390 | flags); |
| 1391 | adapter->cmd_queued = cmd_node; | ||
| 1392 | mwifiex_insert_cmd_to_pending_q(adapter, cmd_node, | 1391 | mwifiex_insert_cmd_to_pending_q(adapter, cmd_node, |
| 1393 | true); | 1392 | true); |
| 1394 | queue_work(adapter->workqueue, &adapter->main_work); | 1393 | queue_work(adapter->workqueue, &adapter->main_work); |
| 1394 | |||
| 1395 | /* Perform internal scan synchronously */ | ||
| 1396 | if (!priv->scan_request) | ||
| 1397 | mwifiex_wait_queue_complete(adapter, cmd_node); | ||
| 1395 | } else { | 1398 | } else { |
| 1396 | spin_unlock_irqrestore(&adapter->scan_pending_q_lock, | 1399 | spin_unlock_irqrestore(&adapter->scan_pending_q_lock, |
| 1397 | flags); | 1400 | flags); |
| @@ -1946,9 +1949,6 @@ int mwifiex_request_scan(struct mwifiex_private *priv, | |||
| 1946 | /* Normal scan */ | 1949 | /* Normal scan */ |
| 1947 | ret = mwifiex_scan_networks(priv, NULL); | 1950 | ret = mwifiex_scan_networks(priv, NULL); |
| 1948 | 1951 | ||
| 1949 | if (!ret) | ||
| 1950 | ret = mwifiex_wait_queue_complete(priv->adapter); | ||
| 1951 | |||
| 1952 | up(&priv->async_sem); | 1952 | up(&priv->async_sem); |
| 1953 | 1953 | ||
| 1954 | return ret; | 1954 | return ret; |
diff --git a/drivers/net/wireless/mwifiex/sta_ioctl.c b/drivers/net/wireless/mwifiex/sta_ioctl.c index 9f33c92c90f5..13100f8de3db 100644 --- a/drivers/net/wireless/mwifiex/sta_ioctl.c +++ b/drivers/net/wireless/mwifiex/sta_ioctl.c | |||
| @@ -54,16 +54,10 @@ int mwifiex_copy_mcast_addr(struct mwifiex_multicast_list *mlist, | |||
| 54 | * This function waits on a cmd wait queue. It also cancels the pending | 54 | * This function waits on a cmd wait queue. It also cancels the pending |
| 55 | * request after waking up, in case of errors. | 55 | * request after waking up, in case of errors. |
| 56 | */ | 56 | */ |
| 57 | int mwifiex_wait_queue_complete(struct mwifiex_adapter *adapter) | 57 | int mwifiex_wait_queue_complete(struct mwifiex_adapter *adapter, |
| 58 | struct cmd_ctrl_node *cmd_queued) | ||
| 58 | { | 59 | { |
| 59 | int status; | 60 | int status; |
| 60 | struct cmd_ctrl_node *cmd_queued; | ||
| 61 | |||
| 62 | if (!adapter->cmd_queued) | ||
| 63 | return 0; | ||
| 64 | |||
| 65 | cmd_queued = adapter->cmd_queued; | ||
| 66 | adapter->cmd_queued = NULL; | ||
| 67 | 61 | ||
| 68 | dev_dbg(adapter->dev, "cmd pending\n"); | 62 | dev_dbg(adapter->dev, "cmd pending\n"); |
| 69 | atomic_inc(&adapter->cmd_pending); | 63 | atomic_inc(&adapter->cmd_pending); |
diff --git a/drivers/net/wireless/rtlwifi/usb.c b/drivers/net/wireless/rtlwifi/usb.c index 156b52732f3d..5847d6d0881e 100644 --- a/drivers/net/wireless/rtlwifi/usb.c +++ b/drivers/net/wireless/rtlwifi/usb.c | |||
| @@ -851,6 +851,7 @@ static void _rtl_usb_transmit(struct ieee80211_hw *hw, struct sk_buff *skb, | |||
| 851 | if (unlikely(!_urb)) { | 851 | if (unlikely(!_urb)) { |
| 852 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | 852 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, |
| 853 | "Can't allocate urb. Drop skb!\n"); | 853 | "Can't allocate urb. Drop skb!\n"); |
| 854 | kfree_skb(skb); | ||
| 854 | return; | 855 | return; |
| 855 | } | 856 | } |
| 856 | _rtl_submit_tx_urb(hw, _urb); | 857 | _rtl_submit_tx_urb(hw, _urb); |
diff --git a/drivers/pinctrl/mvebu/pinctrl-mvebu.c b/drivers/pinctrl/mvebu/pinctrl-mvebu.c index c689c04a4f52..2d2f0a43d36b 100644 --- a/drivers/pinctrl/mvebu/pinctrl-mvebu.c +++ b/drivers/pinctrl/mvebu/pinctrl-mvebu.c | |||
| @@ -620,7 +620,7 @@ int mvebu_pinctrl_probe(struct platform_device *pdev) | |||
| 620 | 620 | ||
| 621 | /* special soc specific control */ | 621 | /* special soc specific control */ |
| 622 | if (ctrl->mpp_get || ctrl->mpp_set) { | 622 | if (ctrl->mpp_get || ctrl->mpp_set) { |
| 623 | if (!ctrl->name || !ctrl->mpp_set || !ctrl->mpp_set) { | 623 | if (!ctrl->name || !ctrl->mpp_get || !ctrl->mpp_set) { |
| 624 | dev_err(&pdev->dev, "wrong soc control info\n"); | 624 | dev_err(&pdev->dev, "wrong soc control info\n"); |
| 625 | return -EINVAL; | 625 | return -EINVAL; |
| 626 | } | 626 | } |
diff --git a/drivers/pinctrl/pinconf.c b/drivers/pinctrl/pinconf.c index ac8d382a79bb..d611ecfcbf70 100644 --- a/drivers/pinctrl/pinconf.c +++ b/drivers/pinctrl/pinconf.c | |||
| @@ -622,7 +622,7 @@ static const struct file_operations pinconf_dbg_pinname_fops = { | |||
| 622 | static int pinconf_dbg_state_print(struct seq_file *s, void *d) | 622 | static int pinconf_dbg_state_print(struct seq_file *s, void *d) |
| 623 | { | 623 | { |
| 624 | if (strlen(dbg_state_name)) | 624 | if (strlen(dbg_state_name)) |
| 625 | seq_printf(s, "%s\n", dbg_pinname); | 625 | seq_printf(s, "%s\n", dbg_state_name); |
| 626 | else | 626 | else |
| 627 | seq_printf(s, "No pin state set\n"); | 627 | seq_printf(s, "No pin state set\n"); |
| 628 | return 0; | 628 | return 0; |
diff --git a/drivers/pinctrl/pinconf.h b/drivers/pinctrl/pinconf.h index e3ed8cb072a5..bfda73d64eed 100644 --- a/drivers/pinctrl/pinconf.h +++ b/drivers/pinctrl/pinconf.h | |||
| @@ -90,7 +90,7 @@ static inline void pinconf_init_device_debugfs(struct dentry *devroot, | |||
| 90 | * pin config. | 90 | * pin config. |
| 91 | */ | 91 | */ |
| 92 | 92 | ||
| 93 | #ifdef CONFIG_GENERIC_PINCONF | 93 | #if defined(CONFIG_GENERIC_PINCONF) && defined(CONFIG_DEBUG_FS) |
| 94 | 94 | ||
| 95 | void pinconf_generic_dump_pin(struct pinctrl_dev *pctldev, | 95 | void pinconf_generic_dump_pin(struct pinctrl_dev *pctldev, |
| 96 | struct seq_file *s, unsigned pin); | 96 | struct seq_file *s, unsigned pin); |
diff --git a/drivers/pinctrl/pinctrl-abx500.c b/drivers/pinctrl/pinctrl-abx500.c index caecdd373061..c542a97c82f3 100644 --- a/drivers/pinctrl/pinctrl-abx500.c +++ b/drivers/pinctrl/pinctrl-abx500.c | |||
| @@ -422,7 +422,7 @@ static u8 abx500_get_mode(struct pinctrl_dev *pctldev, struct gpio_chip *chip, | |||
| 422 | } | 422 | } |
| 423 | 423 | ||
| 424 | /* check if pin use AlternateFunction register */ | 424 | /* check if pin use AlternateFunction register */ |
| 425 | if ((af.alt_bit1 == UNUSED) && (af.alt_bit1 == UNUSED)) | 425 | if ((af.alt_bit1 == UNUSED) && (af.alt_bit2 == UNUSED)) |
| 426 | return mode; | 426 | return mode; |
| 427 | /* | 427 | /* |
| 428 | * if pin GPIOSEL bit is set and pin supports alternate function, | 428 | * if pin GPIOSEL bit is set and pin supports alternate function, |
diff --git a/drivers/pinctrl/pinmux.c b/drivers/pinctrl/pinmux.c index 1a00658b3ea0..bd83c8b01cd1 100644 --- a/drivers/pinctrl/pinmux.c +++ b/drivers/pinctrl/pinmux.c | |||
| @@ -194,6 +194,11 @@ static const char *pin_free(struct pinctrl_dev *pctldev, int pin, | |||
| 194 | } | 194 | } |
| 195 | 195 | ||
| 196 | if (!gpio_range) { | 196 | if (!gpio_range) { |
| 197 | /* | ||
| 198 | * A pin should not be freed more times than allocated. | ||
| 199 | */ | ||
| 200 | if (WARN_ON(!desc->mux_usecount)) | ||
| 201 | return NULL; | ||
| 197 | desc->mux_usecount--; | 202 | desc->mux_usecount--; |
| 198 | if (desc->mux_usecount) | 203 | if (desc->mux_usecount) |
| 199 | return NULL; | 204 | return NULL; |
diff --git a/drivers/staging/comedi/drivers/s626.c b/drivers/staging/comedi/drivers/s626.c index 81a1fe661579..71a73ec5af8d 100644 --- a/drivers/staging/comedi/drivers/s626.c +++ b/drivers/staging/comedi/drivers/s626.c | |||
| @@ -1483,7 +1483,7 @@ static int s626_ai_cmd(struct comedi_device *dev, struct comedi_subdevice *s) | |||
| 1483 | case TRIG_NONE: | 1483 | case TRIG_NONE: |
| 1484 | /* continous acquisition */ | 1484 | /* continous acquisition */ |
| 1485 | devpriv->ai_continous = 1; | 1485 | devpriv->ai_continous = 1; |
| 1486 | devpriv->ai_sample_count = 0; | 1486 | devpriv->ai_sample_count = 1; |
| 1487 | break; | 1487 | break; |
| 1488 | } | 1488 | } |
| 1489 | 1489 | ||
diff --git a/drivers/staging/zcache/Kconfig b/drivers/staging/zcache/Kconfig index 73582705e8c5..5c3714530961 100644 --- a/drivers/staging/zcache/Kconfig +++ b/drivers/staging/zcache/Kconfig | |||
| @@ -15,7 +15,7 @@ config RAMSTER | |||
| 15 | depends on CONFIGFS_FS=y && SYSFS=y && !HIGHMEM && ZCACHE=y | 15 | depends on CONFIGFS_FS=y && SYSFS=y && !HIGHMEM && ZCACHE=y |
| 16 | depends on NET | 16 | depends on NET |
| 17 | # must ensure struct page is 8-byte aligned | 17 | # must ensure struct page is 8-byte aligned |
| 18 | select HAVE_ALIGNED_STRUCT_PAGE if !64_BIT | 18 | select HAVE_ALIGNED_STRUCT_PAGE if !64BIT |
| 19 | default n | 19 | default n |
| 20 | help | 20 | help |
| 21 | RAMster allows RAM on other machines in a cluster to be utilized | 21 | RAMster allows RAM on other machines in a cluster to be utilized |
diff --git a/drivers/target/target_core_transport.c b/drivers/target/target_core_transport.c index 2030b608136d..3243ea790eab 100644 --- a/drivers/target/target_core_transport.c +++ b/drivers/target/target_core_transport.c | |||
| @@ -1139,8 +1139,10 @@ target_setup_cmd_from_cdb(struct se_cmd *cmd, unsigned char *cdb) | |||
| 1139 | return ret; | 1139 | return ret; |
| 1140 | 1140 | ||
| 1141 | ret = target_check_reservation(cmd); | 1141 | ret = target_check_reservation(cmd); |
| 1142 | if (ret) | 1142 | if (ret) { |
| 1143 | cmd->scsi_status = SAM_STAT_RESERVATION_CONFLICT; | ||
| 1143 | return ret; | 1144 | return ret; |
| 1145 | } | ||
| 1144 | 1146 | ||
| 1145 | ret = dev->transport->parse_cdb(cmd); | 1147 | ret = dev->transport->parse_cdb(cmd); |
| 1146 | if (ret) | 1148 | if (ret) |
diff --git a/drivers/tty/serial/8250/8250.c b/drivers/tty/serial/8250/8250_core.c index cf6a5383748a..35f9c96aada9 100644 --- a/drivers/tty/serial/8250/8250.c +++ b/drivers/tty/serial/8250/8250_core.c | |||
| @@ -3418,6 +3418,7 @@ MODULE_PARM_DESC(probe_rsa, "Probe I/O ports for RSA"); | |||
| 3418 | #endif | 3418 | #endif |
| 3419 | MODULE_ALIAS_CHARDEV_MAJOR(TTY_MAJOR); | 3419 | MODULE_ALIAS_CHARDEV_MAJOR(TTY_MAJOR); |
| 3420 | 3420 | ||
| 3421 | #ifdef CONFIG_SERIAL_8250_DEPRECATED_OPTIONS | ||
| 3421 | #ifndef MODULE | 3422 | #ifndef MODULE |
| 3422 | /* This module was renamed to 8250_core in 3.7. Keep the old "8250" name | 3423 | /* This module was renamed to 8250_core in 3.7. Keep the old "8250" name |
| 3423 | * working as well for the module options so we don't break people. We | 3424 | * working as well for the module options so we don't break people. We |
| @@ -3432,7 +3433,7 @@ MODULE_ALIAS_CHARDEV_MAJOR(TTY_MAJOR); | |||
| 3432 | static void __used s8250_options(void) | 3433 | static void __used s8250_options(void) |
| 3433 | { | 3434 | { |
| 3434 | #undef MODULE_PARAM_PREFIX | 3435 | #undef MODULE_PARAM_PREFIX |
| 3435 | #define MODULE_PARAM_PREFIX "8250." | 3436 | #define MODULE_PARAM_PREFIX "8250_core." |
| 3436 | 3437 | ||
| 3437 | module_param_cb(share_irqs, ¶m_ops_uint, &share_irqs, 0644); | 3438 | module_param_cb(share_irqs, ¶m_ops_uint, &share_irqs, 0644); |
| 3438 | module_param_cb(nr_uarts, ¶m_ops_uint, &nr_uarts, 0644); | 3439 | module_param_cb(nr_uarts, ¶m_ops_uint, &nr_uarts, 0644); |
| @@ -3444,5 +3445,6 @@ static void __used s8250_options(void) | |||
| 3444 | #endif | 3445 | #endif |
| 3445 | } | 3446 | } |
| 3446 | #else | 3447 | #else |
| 3447 | MODULE_ALIAS("8250"); | 3448 | MODULE_ALIAS("8250_core"); |
| 3449 | #endif | ||
| 3448 | #endif | 3450 | #endif |
diff --git a/drivers/tty/serial/8250/8250_pci.c b/drivers/tty/serial/8250/8250_pci.c index aa76825229dc..26e3a97ab157 100644 --- a/drivers/tty/serial/8250/8250_pci.c +++ b/drivers/tty/serial/8250/8250_pci.c | |||
| @@ -1554,6 +1554,7 @@ pci_wch_ch353_setup(struct serial_private *priv, | |||
| 1554 | #define PCI_DEVICE_ID_PLX_CRONYX_OMEGA 0xc001 | 1554 | #define PCI_DEVICE_ID_PLX_CRONYX_OMEGA 0xc001 |
| 1555 | #define PCI_DEVICE_ID_INTEL_PATSBURG_KT 0x1d3d | 1555 | #define PCI_DEVICE_ID_INTEL_PATSBURG_KT 0x1d3d |
| 1556 | #define PCI_VENDOR_ID_WCH 0x4348 | 1556 | #define PCI_VENDOR_ID_WCH 0x4348 |
| 1557 | #define PCI_DEVICE_ID_WCH_CH352_2S 0x3253 | ||
| 1557 | #define PCI_DEVICE_ID_WCH_CH353_4S 0x3453 | 1558 | #define PCI_DEVICE_ID_WCH_CH353_4S 0x3453 |
| 1558 | #define PCI_DEVICE_ID_WCH_CH353_2S1PF 0x5046 | 1559 | #define PCI_DEVICE_ID_WCH_CH353_2S1PF 0x5046 |
| 1559 | #define PCI_DEVICE_ID_WCH_CH353_2S1P 0x7053 | 1560 | #define PCI_DEVICE_ID_WCH_CH353_2S1P 0x7053 |
| @@ -2172,6 +2173,14 @@ static struct pci_serial_quirk pci_serial_quirks[] __refdata = { | |||
| 2172 | .subdevice = PCI_ANY_ID, | 2173 | .subdevice = PCI_ANY_ID, |
| 2173 | .setup = pci_wch_ch353_setup, | 2174 | .setup = pci_wch_ch353_setup, |
| 2174 | }, | 2175 | }, |
| 2176 | /* WCH CH352 2S card (16550 clone) */ | ||
| 2177 | { | ||
| 2178 | .vendor = PCI_VENDOR_ID_WCH, | ||
| 2179 | .device = PCI_DEVICE_ID_WCH_CH352_2S, | ||
| 2180 | .subvendor = PCI_ANY_ID, | ||
| 2181 | .subdevice = PCI_ANY_ID, | ||
| 2182 | .setup = pci_wch_ch353_setup, | ||
| 2183 | }, | ||
| 2175 | /* | 2184 | /* |
| 2176 | * ASIX devices with FIFO bug | 2185 | * ASIX devices with FIFO bug |
| 2177 | */ | 2186 | */ |
| @@ -4870,6 +4879,10 @@ static struct pci_device_id serial_pci_tbl[] = { | |||
| 4870 | PCI_ANY_ID, PCI_ANY_ID, | 4879 | PCI_ANY_ID, PCI_ANY_ID, |
| 4871 | 0, 0, pbn_b0_bt_2_115200 }, | 4880 | 0, 0, pbn_b0_bt_2_115200 }, |
| 4872 | 4881 | ||
| 4882 | { PCI_VENDOR_ID_WCH, PCI_DEVICE_ID_WCH_CH352_2S, | ||
| 4883 | PCI_ANY_ID, PCI_ANY_ID, | ||
| 4884 | 0, 0, pbn_b0_bt_2_115200 }, | ||
| 4885 | |||
| 4873 | /* | 4886 | /* |
| 4874 | * Commtech, Inc. Fastcom adapters | 4887 | * Commtech, Inc. Fastcom adapters |
| 4875 | */ | 4888 | */ |
diff --git a/drivers/tty/serial/8250/Kconfig b/drivers/tty/serial/8250/Kconfig index 2ef9537bcb2c..80fe91e64a52 100644 --- a/drivers/tty/serial/8250/Kconfig +++ b/drivers/tty/serial/8250/Kconfig | |||
| @@ -33,6 +33,23 @@ config SERIAL_8250 | |||
| 33 | Most people will say Y or M here, so that they can use serial mice, | 33 | Most people will say Y or M here, so that they can use serial mice, |
| 34 | modems and similar devices connecting to the standard serial ports. | 34 | modems and similar devices connecting to the standard serial ports. |
| 35 | 35 | ||
| 36 | config SERIAL_8250_DEPRECATED_OPTIONS | ||
| 37 | bool "Support 8250_core.* kernel options (DEPRECATED)" | ||
| 38 | depends on SERIAL_8250 | ||
| 39 | default y | ||
| 40 | ---help--- | ||
| 41 | In 3.7 we renamed 8250 to 8250_core by mistake, so now we have to | ||
| 42 | accept kernel parameters in both forms like 8250_core.nr_uarts=4 and | ||
| 43 | 8250.nr_uarts=4. We now renamed the module back to 8250, but if | ||
| 44 | anybody noticed in 3.7 and changed their userspace we still have to | ||
| 45 | keep the 8350_core.* options around until they revert the changes | ||
| 46 | they already did. | ||
| 47 | |||
| 48 | If 8250 is built as a module, this adds 8250_core alias instead. | ||
| 49 | |||
| 50 | If you did not notice yet and/or you have userspace from pre-3.7, it | ||
| 51 | is safe (and recommended) to say N here. | ||
| 52 | |||
| 36 | config SERIAL_8250_PNP | 53 | config SERIAL_8250_PNP |
| 37 | bool "8250/16550 PNP device support" if EXPERT | 54 | bool "8250/16550 PNP device support" if EXPERT |
| 38 | depends on SERIAL_8250 && PNP | 55 | depends on SERIAL_8250 && PNP |
diff --git a/drivers/tty/serial/8250/Makefile b/drivers/tty/serial/8250/Makefile index a23838a4d535..36d68d054307 100644 --- a/drivers/tty/serial/8250/Makefile +++ b/drivers/tty/serial/8250/Makefile | |||
| @@ -2,10 +2,10 @@ | |||
| 2 | # Makefile for the 8250 serial device drivers. | 2 | # Makefile for the 8250 serial device drivers. |
| 3 | # | 3 | # |
| 4 | 4 | ||
| 5 | obj-$(CONFIG_SERIAL_8250) += 8250_core.o | 5 | obj-$(CONFIG_SERIAL_8250) += 8250.o |
| 6 | 8250_core-y := 8250.o | 6 | 8250-y := 8250_core.o |
| 7 | 8250_core-$(CONFIG_SERIAL_8250_PNP) += 8250_pnp.o | 7 | 8250-$(CONFIG_SERIAL_8250_PNP) += 8250_pnp.o |
| 8 | 8250_core-$(CONFIG_SERIAL_8250_DMA) += 8250_dma.o | 8 | 8250-$(CONFIG_SERIAL_8250_DMA) += 8250_dma.o |
| 9 | obj-$(CONFIG_SERIAL_8250_GSC) += 8250_gsc.o | 9 | obj-$(CONFIG_SERIAL_8250_GSC) += 8250_gsc.o |
| 10 | obj-$(CONFIG_SERIAL_8250_PCI) += 8250_pci.o | 10 | obj-$(CONFIG_SERIAL_8250_PCI) += 8250_pci.o |
| 11 | obj-$(CONFIG_SERIAL_8250_HP300) += 8250_hp300.o | 11 | obj-$(CONFIG_SERIAL_8250_HP300) += 8250_hp300.o |
diff --git a/drivers/tty/serial/atmel_serial.c b/drivers/tty/serial/atmel_serial.c index d4a7c241b751..3467462869ce 100644 --- a/drivers/tty/serial/atmel_serial.c +++ b/drivers/tty/serial/atmel_serial.c | |||
| @@ -158,7 +158,7 @@ struct atmel_uart_port { | |||
| 158 | }; | 158 | }; |
| 159 | 159 | ||
| 160 | static struct atmel_uart_port atmel_ports[ATMEL_MAX_UART]; | 160 | static struct atmel_uart_port atmel_ports[ATMEL_MAX_UART]; |
| 161 | static unsigned long atmel_ports_in_use; | 161 | static DECLARE_BITMAP(atmel_ports_in_use, ATMEL_MAX_UART); |
| 162 | 162 | ||
| 163 | #ifdef SUPPORT_SYSRQ | 163 | #ifdef SUPPORT_SYSRQ |
| 164 | static struct console atmel_console; | 164 | static struct console atmel_console; |
| @@ -1769,15 +1769,14 @@ static int atmel_serial_probe(struct platform_device *pdev) | |||
| 1769 | if (ret < 0) | 1769 | if (ret < 0) |
| 1770 | /* port id not found in platform data nor device-tree aliases: | 1770 | /* port id not found in platform data nor device-tree aliases: |
| 1771 | * auto-enumerate it */ | 1771 | * auto-enumerate it */ |
| 1772 | ret = find_first_zero_bit(&atmel_ports_in_use, | 1772 | ret = find_first_zero_bit(atmel_ports_in_use, ATMEL_MAX_UART); |
| 1773 | sizeof(atmel_ports_in_use)); | ||
| 1774 | 1773 | ||
| 1775 | if (ret > ATMEL_MAX_UART) { | 1774 | if (ret >= ATMEL_MAX_UART) { |
| 1776 | ret = -ENODEV; | 1775 | ret = -ENODEV; |
| 1777 | goto err; | 1776 | goto err; |
| 1778 | } | 1777 | } |
| 1779 | 1778 | ||
| 1780 | if (test_and_set_bit(ret, &atmel_ports_in_use)) { | 1779 | if (test_and_set_bit(ret, atmel_ports_in_use)) { |
| 1781 | /* port already in use */ | 1780 | /* port already in use */ |
| 1782 | ret = -EBUSY; | 1781 | ret = -EBUSY; |
| 1783 | goto err; | 1782 | goto err; |
| @@ -1857,7 +1856,7 @@ static int atmel_serial_remove(struct platform_device *pdev) | |||
| 1857 | 1856 | ||
| 1858 | /* "port" is allocated statically, so we shouldn't free it */ | 1857 | /* "port" is allocated statically, so we shouldn't free it */ |
| 1859 | 1858 | ||
| 1860 | clear_bit(port->line, &atmel_ports_in_use); | 1859 | clear_bit(port->line, atmel_ports_in_use); |
| 1861 | 1860 | ||
| 1862 | clk_put(atmel_port->clk); | 1861 | clk_put(atmel_port->clk); |
| 1863 | 1862 | ||
diff --git a/drivers/tty/serial/xilinx_uartps.c b/drivers/tty/serial/xilinx_uartps.c index ba451c7209fc..f36bbba1ac8b 100644 --- a/drivers/tty/serial/xilinx_uartps.c +++ b/drivers/tty/serial/xilinx_uartps.c | |||
| @@ -578,6 +578,8 @@ static int xuartps_startup(struct uart_port *port) | |||
| 578 | /* Receive Timeout register is enabled with value of 10 */ | 578 | /* Receive Timeout register is enabled with value of 10 */ |
| 579 | xuartps_writel(10, XUARTPS_RXTOUT_OFFSET); | 579 | xuartps_writel(10, XUARTPS_RXTOUT_OFFSET); |
| 580 | 580 | ||
| 581 | /* Clear out any pending interrupts before enabling them */ | ||
| 582 | xuartps_writel(xuartps_readl(XUARTPS_ISR_OFFSET), XUARTPS_ISR_OFFSET); | ||
| 581 | 583 | ||
| 582 | /* Set the Interrupt Registers with desired interrupts */ | 584 | /* Set the Interrupt Registers with desired interrupts */ |
| 583 | xuartps_writel(XUARTPS_IXR_TXEMPTY | XUARTPS_IXR_PARITY | | 585 | xuartps_writel(XUARTPS_IXR_TXEMPTY | XUARTPS_IXR_PARITY | |
diff --git a/drivers/tty/vt/vc_screen.c b/drivers/tty/vt/vc_screen.c index e4ca345873c3..d7799deacb21 100644 --- a/drivers/tty/vt/vc_screen.c +++ b/drivers/tty/vt/vc_screen.c | |||
| @@ -93,7 +93,7 @@ vcs_poll_data_free(struct vcs_poll_data *poll) | |||
| 93 | static struct vcs_poll_data * | 93 | static struct vcs_poll_data * |
| 94 | vcs_poll_data_get(struct file *file) | 94 | vcs_poll_data_get(struct file *file) |
| 95 | { | 95 | { |
| 96 | struct vcs_poll_data *poll = file->private_data; | 96 | struct vcs_poll_data *poll = file->private_data, *kill = NULL; |
| 97 | 97 | ||
| 98 | if (poll) | 98 | if (poll) |
| 99 | return poll; | 99 | return poll; |
| @@ -122,10 +122,12 @@ vcs_poll_data_get(struct file *file) | |||
| 122 | file->private_data = poll; | 122 | file->private_data = poll; |
| 123 | } else { | 123 | } else { |
| 124 | /* someone else raced ahead of us */ | 124 | /* someone else raced ahead of us */ |
| 125 | vcs_poll_data_free(poll); | 125 | kill = poll; |
| 126 | poll = file->private_data; | 126 | poll = file->private_data; |
| 127 | } | 127 | } |
| 128 | spin_unlock(&file->f_lock); | 128 | spin_unlock(&file->f_lock); |
| 129 | if (kill) | ||
| 130 | vcs_poll_data_free(kill); | ||
| 129 | 131 | ||
| 130 | return poll; | 132 | return poll; |
| 131 | } | 133 | } |
diff --git a/drivers/usb/core/hcd.c b/drivers/usb/core/hcd.c index 99b34a30354f..f9ec44cbb82f 100644 --- a/drivers/usb/core/hcd.c +++ b/drivers/usb/core/hcd.c | |||
| @@ -2412,6 +2412,14 @@ int usb_hcd_is_primary_hcd(struct usb_hcd *hcd) | |||
| 2412 | } | 2412 | } |
| 2413 | EXPORT_SYMBOL_GPL(usb_hcd_is_primary_hcd); | 2413 | EXPORT_SYMBOL_GPL(usb_hcd_is_primary_hcd); |
| 2414 | 2414 | ||
| 2415 | int usb_hcd_find_raw_port_number(struct usb_hcd *hcd, int port1) | ||
| 2416 | { | ||
| 2417 | if (!hcd->driver->find_raw_port_number) | ||
| 2418 | return port1; | ||
| 2419 | |||
| 2420 | return hcd->driver->find_raw_port_number(hcd, port1); | ||
| 2421 | } | ||
| 2422 | |||
| 2415 | static int usb_hcd_request_irqs(struct usb_hcd *hcd, | 2423 | static int usb_hcd_request_irqs(struct usb_hcd *hcd, |
| 2416 | unsigned int irqnum, unsigned long irqflags) | 2424 | unsigned int irqnum, unsigned long irqflags) |
| 2417 | { | 2425 | { |
diff --git a/drivers/usb/core/usb-acpi.c b/drivers/usb/core/usb-acpi.c index b6f4bad3f756..255c14464bf2 100644 --- a/drivers/usb/core/usb-acpi.c +++ b/drivers/usb/core/usb-acpi.c | |||
| @@ -15,6 +15,7 @@ | |||
| 15 | #include <linux/kernel.h> | 15 | #include <linux/kernel.h> |
| 16 | #include <linux/acpi.h> | 16 | #include <linux/acpi.h> |
| 17 | #include <linux/pci.h> | 17 | #include <linux/pci.h> |
| 18 | #include <linux/usb/hcd.h> | ||
| 18 | #include <acpi/acpi_bus.h> | 19 | #include <acpi/acpi_bus.h> |
| 19 | 20 | ||
| 20 | #include "usb.h" | 21 | #include "usb.h" |
| @@ -188,8 +189,13 @@ static int usb_acpi_find_device(struct device *dev, acpi_handle *handle) | |||
| 188 | * connected to. | 189 | * connected to. |
| 189 | */ | 190 | */ |
| 190 | if (!udev->parent) { | 191 | if (!udev->parent) { |
| 191 | *handle = acpi_get_child(DEVICE_ACPI_HANDLE(&udev->dev), | 192 | struct usb_hcd *hcd = bus_to_hcd(udev->bus); |
| 193 | int raw_port_num; | ||
| 194 | |||
| 195 | raw_port_num = usb_hcd_find_raw_port_number(hcd, | ||
| 192 | port_num); | 196 | port_num); |
| 197 | *handle = acpi_get_child(DEVICE_ACPI_HANDLE(&udev->dev), | ||
| 198 | raw_port_num); | ||
| 193 | if (!*handle) | 199 | if (!*handle) |
| 194 | return -ENODEV; | 200 | return -ENODEV; |
| 195 | } else { | 201 | } else { |
diff --git a/drivers/usb/gadget/Kconfig b/drivers/usb/gadget/Kconfig index 5a0c541daf89..c7525b1cad74 100644 --- a/drivers/usb/gadget/Kconfig +++ b/drivers/usb/gadget/Kconfig | |||
| @@ -145,6 +145,7 @@ config USB_LPC32XX | |||
| 145 | tristate "LPC32XX USB Peripheral Controller" | 145 | tristate "LPC32XX USB Peripheral Controller" |
| 146 | depends on ARCH_LPC32XX | 146 | depends on ARCH_LPC32XX |
| 147 | select USB_ISP1301 | 147 | select USB_ISP1301 |
| 148 | select USB_OTG_UTILS | ||
| 148 | help | 149 | help |
| 149 | This option selects the USB device controller in the LPC32xx SoC. | 150 | This option selects the USB device controller in the LPC32xx SoC. |
| 150 | 151 | ||
diff --git a/drivers/usb/host/ehci-sched.c b/drivers/usb/host/ehci-sched.c index b476daf49f6f..010f686d8881 100644 --- a/drivers/usb/host/ehci-sched.c +++ b/drivers/usb/host/ehci-sched.c | |||
| @@ -1214,6 +1214,7 @@ itd_urb_transaction ( | |||
| 1214 | 1214 | ||
| 1215 | memset (itd, 0, sizeof *itd); | 1215 | memset (itd, 0, sizeof *itd); |
| 1216 | itd->itd_dma = itd_dma; | 1216 | itd->itd_dma = itd_dma; |
| 1217 | itd->frame = 9999; /* an invalid value */ | ||
| 1217 | list_add (&itd->itd_list, &sched->td_list); | 1218 | list_add (&itd->itd_list, &sched->td_list); |
| 1218 | } | 1219 | } |
| 1219 | spin_unlock_irqrestore (&ehci->lock, flags); | 1220 | spin_unlock_irqrestore (&ehci->lock, flags); |
| @@ -1915,6 +1916,7 @@ sitd_urb_transaction ( | |||
| 1915 | 1916 | ||
| 1916 | memset (sitd, 0, sizeof *sitd); | 1917 | memset (sitd, 0, sizeof *sitd); |
| 1917 | sitd->sitd_dma = sitd_dma; | 1918 | sitd->sitd_dma = sitd_dma; |
| 1919 | sitd->frame = 9999; /* an invalid value */ | ||
| 1918 | list_add (&sitd->sitd_list, &iso_sched->td_list); | 1920 | list_add (&sitd->sitd_list, &iso_sched->td_list); |
| 1919 | } | 1921 | } |
| 1920 | 1922 | ||
diff --git a/drivers/usb/host/xhci-mem.c b/drivers/usb/host/xhci-mem.c index 35616ffbe3ae..6dc238c592bc 100644 --- a/drivers/usb/host/xhci-mem.c +++ b/drivers/usb/host/xhci-mem.c | |||
| @@ -1022,44 +1022,24 @@ void xhci_copy_ep0_dequeue_into_input_ctx(struct xhci_hcd *xhci, | |||
| 1022 | * is attached to (or the roothub port its ancestor hub is attached to). All we | 1022 | * is attached to (or the roothub port its ancestor hub is attached to). All we |
| 1023 | * know is the index of that port under either the USB 2.0 or the USB 3.0 | 1023 | * know is the index of that port under either the USB 2.0 or the USB 3.0 |
| 1024 | * roothub, but that doesn't give us the real index into the HW port status | 1024 | * roothub, but that doesn't give us the real index into the HW port status |
| 1025 | * registers. Scan through the xHCI roothub port array, looking for the Nth | 1025 | * registers. Call xhci_find_raw_port_number() to get real index. |
| 1026 | * entry of the correct port speed. Return the port number of that entry. | ||
| 1027 | */ | 1026 | */ |
| 1028 | static u32 xhci_find_real_port_number(struct xhci_hcd *xhci, | 1027 | static u32 xhci_find_real_port_number(struct xhci_hcd *xhci, |
| 1029 | struct usb_device *udev) | 1028 | struct usb_device *udev) |
| 1030 | { | 1029 | { |
| 1031 | struct usb_device *top_dev; | 1030 | struct usb_device *top_dev; |
| 1032 | unsigned int num_similar_speed_ports; | 1031 | struct usb_hcd *hcd; |
| 1033 | unsigned int faked_port_num; | 1032 | |
| 1034 | int i; | 1033 | if (udev->speed == USB_SPEED_SUPER) |
| 1034 | hcd = xhci->shared_hcd; | ||
| 1035 | else | ||
| 1036 | hcd = xhci->main_hcd; | ||
| 1035 | 1037 | ||
| 1036 | for (top_dev = udev; top_dev->parent && top_dev->parent->parent; | 1038 | for (top_dev = udev; top_dev->parent && top_dev->parent->parent; |
| 1037 | top_dev = top_dev->parent) | 1039 | top_dev = top_dev->parent) |
| 1038 | /* Found device below root hub */; | 1040 | /* Found device below root hub */; |
| 1039 | faked_port_num = top_dev->portnum; | ||
| 1040 | for (i = 0, num_similar_speed_ports = 0; | ||
| 1041 | i < HCS_MAX_PORTS(xhci->hcs_params1); i++) { | ||
| 1042 | u8 port_speed = xhci->port_array[i]; | ||
| 1043 | |||
| 1044 | /* | ||
| 1045 | * Skip ports that don't have known speeds, or have duplicate | ||
| 1046 | * Extended Capabilities port speed entries. | ||
| 1047 | */ | ||
| 1048 | if (port_speed == 0 || port_speed == DUPLICATE_ENTRY) | ||
| 1049 | continue; | ||
| 1050 | 1041 | ||
| 1051 | /* | 1042 | return xhci_find_raw_port_number(hcd, top_dev->portnum); |
| 1052 | * USB 3.0 ports are always under a USB 3.0 hub. USB 2.0 and | ||
| 1053 | * 1.1 ports are under the USB 2.0 hub. If the port speed | ||
| 1054 | * matches the device speed, it's a similar speed port. | ||
| 1055 | */ | ||
| 1056 | if ((port_speed == 0x03) == (udev->speed == USB_SPEED_SUPER)) | ||
| 1057 | num_similar_speed_ports++; | ||
| 1058 | if (num_similar_speed_ports == faked_port_num) | ||
| 1059 | /* Roothub ports are numbered from 1 to N */ | ||
| 1060 | return i+1; | ||
| 1061 | } | ||
| 1062 | return 0; | ||
| 1063 | } | 1043 | } |
| 1064 | 1044 | ||
| 1065 | /* Setup an xHCI virtual device for a Set Address command */ | 1045 | /* Setup an xHCI virtual device for a Set Address command */ |
diff --git a/drivers/usb/host/xhci-pci.c b/drivers/usb/host/xhci-pci.c index af259e0ec172..1a30c380043c 100644 --- a/drivers/usb/host/xhci-pci.c +++ b/drivers/usb/host/xhci-pci.c | |||
| @@ -313,6 +313,7 @@ static const struct hc_driver xhci_pci_hc_driver = { | |||
| 313 | .set_usb2_hw_lpm = xhci_set_usb2_hardware_lpm, | 313 | .set_usb2_hw_lpm = xhci_set_usb2_hardware_lpm, |
| 314 | .enable_usb3_lpm_timeout = xhci_enable_usb3_lpm_timeout, | 314 | .enable_usb3_lpm_timeout = xhci_enable_usb3_lpm_timeout, |
| 315 | .disable_usb3_lpm_timeout = xhci_disable_usb3_lpm_timeout, | 315 | .disable_usb3_lpm_timeout = xhci_disable_usb3_lpm_timeout, |
| 316 | .find_raw_port_number = xhci_find_raw_port_number, | ||
| 316 | }; | 317 | }; |
| 317 | 318 | ||
| 318 | /*-------------------------------------------------------------------------*/ | 319 | /*-------------------------------------------------------------------------*/ |
diff --git a/drivers/usb/host/xhci-ring.c b/drivers/usb/host/xhci-ring.c index 882875465301..1969c001b3f9 100644 --- a/drivers/usb/host/xhci-ring.c +++ b/drivers/usb/host/xhci-ring.c | |||
| @@ -1599,14 +1599,20 @@ static void handle_port_status(struct xhci_hcd *xhci, | |||
| 1599 | max_ports = HCS_MAX_PORTS(xhci->hcs_params1); | 1599 | max_ports = HCS_MAX_PORTS(xhci->hcs_params1); |
| 1600 | if ((port_id <= 0) || (port_id > max_ports)) { | 1600 | if ((port_id <= 0) || (port_id > max_ports)) { |
| 1601 | xhci_warn(xhci, "Invalid port id %d\n", port_id); | 1601 | xhci_warn(xhci, "Invalid port id %d\n", port_id); |
| 1602 | bogus_port_status = true; | 1602 | inc_deq(xhci, xhci->event_ring); |
| 1603 | goto cleanup; | 1603 | return; |
| 1604 | } | 1604 | } |
| 1605 | 1605 | ||
| 1606 | /* Figure out which usb_hcd this port is attached to: | 1606 | /* Figure out which usb_hcd this port is attached to: |
| 1607 | * is it a USB 3.0 port or a USB 2.0/1.1 port? | 1607 | * is it a USB 3.0 port or a USB 2.0/1.1 port? |
| 1608 | */ | 1608 | */ |
| 1609 | major_revision = xhci->port_array[port_id - 1]; | 1609 | major_revision = xhci->port_array[port_id - 1]; |
| 1610 | |||
| 1611 | /* Find the right roothub. */ | ||
| 1612 | hcd = xhci_to_hcd(xhci); | ||
| 1613 | if ((major_revision == 0x03) != (hcd->speed == HCD_USB3)) | ||
| 1614 | hcd = xhci->shared_hcd; | ||
| 1615 | |||
| 1610 | if (major_revision == 0) { | 1616 | if (major_revision == 0) { |
| 1611 | xhci_warn(xhci, "Event for port %u not in " | 1617 | xhci_warn(xhci, "Event for port %u not in " |
| 1612 | "Extended Capabilities, ignoring.\n", | 1618 | "Extended Capabilities, ignoring.\n", |
| @@ -1629,10 +1635,6 @@ static void handle_port_status(struct xhci_hcd *xhci, | |||
| 1629 | * into the index into the ports on the correct split roothub, and the | 1635 | * into the index into the ports on the correct split roothub, and the |
| 1630 | * correct bus_state structure. | 1636 | * correct bus_state structure. |
| 1631 | */ | 1637 | */ |
| 1632 | /* Find the right roothub. */ | ||
| 1633 | hcd = xhci_to_hcd(xhci); | ||
| 1634 | if ((major_revision == 0x03) != (hcd->speed == HCD_USB3)) | ||
| 1635 | hcd = xhci->shared_hcd; | ||
| 1636 | bus_state = &xhci->bus_state[hcd_index(hcd)]; | 1638 | bus_state = &xhci->bus_state[hcd_index(hcd)]; |
| 1637 | if (hcd->speed == HCD_USB3) | 1639 | if (hcd->speed == HCD_USB3) |
| 1638 | port_array = xhci->usb3_ports; | 1640 | port_array = xhci->usb3_ports; |
| @@ -2027,8 +2029,8 @@ static int process_ctrl_td(struct xhci_hcd *xhci, struct xhci_td *td, | |||
| 2027 | if (event_trb != ep_ring->dequeue && | 2029 | if (event_trb != ep_ring->dequeue && |
| 2028 | event_trb != td->last_trb) | 2030 | event_trb != td->last_trb) |
| 2029 | td->urb->actual_length = | 2031 | td->urb->actual_length = |
| 2030 | td->urb->transfer_buffer_length | 2032 | td->urb->transfer_buffer_length - |
| 2031 | - TRB_LEN(le32_to_cpu(event->transfer_len)); | 2033 | EVENT_TRB_LEN(le32_to_cpu(event->transfer_len)); |
| 2032 | else | 2034 | else |
| 2033 | td->urb->actual_length = 0; | 2035 | td->urb->actual_length = 0; |
| 2034 | 2036 | ||
| @@ -2060,7 +2062,7 @@ static int process_ctrl_td(struct xhci_hcd *xhci, struct xhci_td *td, | |||
| 2060 | /* Maybe the event was for the data stage? */ | 2062 | /* Maybe the event was for the data stage? */ |
| 2061 | td->urb->actual_length = | 2063 | td->urb->actual_length = |
| 2062 | td->urb->transfer_buffer_length - | 2064 | td->urb->transfer_buffer_length - |
| 2063 | TRB_LEN(le32_to_cpu(event->transfer_len)); | 2065 | EVENT_TRB_LEN(le32_to_cpu(event->transfer_len)); |
| 2064 | xhci_dbg(xhci, "Waiting for status " | 2066 | xhci_dbg(xhci, "Waiting for status " |
| 2065 | "stage event\n"); | 2067 | "stage event\n"); |
| 2066 | return 0; | 2068 | return 0; |
| @@ -2096,7 +2098,7 @@ static int process_isoc_td(struct xhci_hcd *xhci, struct xhci_td *td, | |||
| 2096 | /* handle completion code */ | 2098 | /* handle completion code */ |
| 2097 | switch (trb_comp_code) { | 2099 | switch (trb_comp_code) { |
| 2098 | case COMP_SUCCESS: | 2100 | case COMP_SUCCESS: |
| 2099 | if (TRB_LEN(le32_to_cpu(event->transfer_len)) == 0) { | 2101 | if (EVENT_TRB_LEN(le32_to_cpu(event->transfer_len)) == 0) { |
| 2100 | frame->status = 0; | 2102 | frame->status = 0; |
| 2101 | break; | 2103 | break; |
| 2102 | } | 2104 | } |
| @@ -2141,7 +2143,7 @@ static int process_isoc_td(struct xhci_hcd *xhci, struct xhci_td *td, | |||
| 2141 | len += TRB_LEN(le32_to_cpu(cur_trb->generic.field[2])); | 2143 | len += TRB_LEN(le32_to_cpu(cur_trb->generic.field[2])); |
| 2142 | } | 2144 | } |
| 2143 | len += TRB_LEN(le32_to_cpu(cur_trb->generic.field[2])) - | 2145 | len += TRB_LEN(le32_to_cpu(cur_trb->generic.field[2])) - |
| 2144 | TRB_LEN(le32_to_cpu(event->transfer_len)); | 2146 | EVENT_TRB_LEN(le32_to_cpu(event->transfer_len)); |
| 2145 | 2147 | ||
| 2146 | if (trb_comp_code != COMP_STOP_INVAL) { | 2148 | if (trb_comp_code != COMP_STOP_INVAL) { |
| 2147 | frame->actual_length = len; | 2149 | frame->actual_length = len; |
| @@ -2199,7 +2201,7 @@ static int process_bulk_intr_td(struct xhci_hcd *xhci, struct xhci_td *td, | |||
| 2199 | case COMP_SUCCESS: | 2201 | case COMP_SUCCESS: |
| 2200 | /* Double check that the HW transferred everything. */ | 2202 | /* Double check that the HW transferred everything. */ |
| 2201 | if (event_trb != td->last_trb || | 2203 | if (event_trb != td->last_trb || |
| 2202 | TRB_LEN(le32_to_cpu(event->transfer_len)) != 0) { | 2204 | EVENT_TRB_LEN(le32_to_cpu(event->transfer_len)) != 0) { |
| 2203 | xhci_warn(xhci, "WARN Successful completion " | 2205 | xhci_warn(xhci, "WARN Successful completion " |
| 2204 | "on short TX\n"); | 2206 | "on short TX\n"); |
| 2205 | if (td->urb->transfer_flags & URB_SHORT_NOT_OK) | 2207 | if (td->urb->transfer_flags & URB_SHORT_NOT_OK) |
| @@ -2227,18 +2229,18 @@ static int process_bulk_intr_td(struct xhci_hcd *xhci, struct xhci_td *td, | |||
| 2227 | "%d bytes untransferred\n", | 2229 | "%d bytes untransferred\n", |
| 2228 | td->urb->ep->desc.bEndpointAddress, | 2230 | td->urb->ep->desc.bEndpointAddress, |
| 2229 | td->urb->transfer_buffer_length, | 2231 | td->urb->transfer_buffer_length, |
| 2230 | TRB_LEN(le32_to_cpu(event->transfer_len))); | 2232 | EVENT_TRB_LEN(le32_to_cpu(event->transfer_len))); |
| 2231 | /* Fast path - was this the last TRB in the TD for this URB? */ | 2233 | /* Fast path - was this the last TRB in the TD for this URB? */ |
| 2232 | if (event_trb == td->last_trb) { | 2234 | if (event_trb == td->last_trb) { |
| 2233 | if (TRB_LEN(le32_to_cpu(event->transfer_len)) != 0) { | 2235 | if (EVENT_TRB_LEN(le32_to_cpu(event->transfer_len)) != 0) { |
| 2234 | td->urb->actual_length = | 2236 | td->urb->actual_length = |
| 2235 | td->urb->transfer_buffer_length - | 2237 | td->urb->transfer_buffer_length - |
| 2236 | TRB_LEN(le32_to_cpu(event->transfer_len)); | 2238 | EVENT_TRB_LEN(le32_to_cpu(event->transfer_len)); |
| 2237 | if (td->urb->transfer_buffer_length < | 2239 | if (td->urb->transfer_buffer_length < |
| 2238 | td->urb->actual_length) { | 2240 | td->urb->actual_length) { |
| 2239 | xhci_warn(xhci, "HC gave bad length " | 2241 | xhci_warn(xhci, "HC gave bad length " |
| 2240 | "of %d bytes left\n", | 2242 | "of %d bytes left\n", |
| 2241 | TRB_LEN(le32_to_cpu(event->transfer_len))); | 2243 | EVENT_TRB_LEN(le32_to_cpu(event->transfer_len))); |
| 2242 | td->urb->actual_length = 0; | 2244 | td->urb->actual_length = 0; |
| 2243 | if (td->urb->transfer_flags & URB_SHORT_NOT_OK) | 2245 | if (td->urb->transfer_flags & URB_SHORT_NOT_OK) |
| 2244 | *status = -EREMOTEIO; | 2246 | *status = -EREMOTEIO; |
| @@ -2280,7 +2282,7 @@ static int process_bulk_intr_td(struct xhci_hcd *xhci, struct xhci_td *td, | |||
| 2280 | if (trb_comp_code != COMP_STOP_INVAL) | 2282 | if (trb_comp_code != COMP_STOP_INVAL) |
| 2281 | td->urb->actual_length += | 2283 | td->urb->actual_length += |
| 2282 | TRB_LEN(le32_to_cpu(cur_trb->generic.field[2])) - | 2284 | TRB_LEN(le32_to_cpu(cur_trb->generic.field[2])) - |
| 2283 | TRB_LEN(le32_to_cpu(event->transfer_len)); | 2285 | EVENT_TRB_LEN(le32_to_cpu(event->transfer_len)); |
| 2284 | } | 2286 | } |
| 2285 | 2287 | ||
| 2286 | return finish_td(xhci, td, event_trb, event, ep, status, false); | 2288 | return finish_td(xhci, td, event_trb, event, ep, status, false); |
| @@ -2368,7 +2370,7 @@ static int handle_tx_event(struct xhci_hcd *xhci, | |||
| 2368 | * transfer type | 2370 | * transfer type |
| 2369 | */ | 2371 | */ |
| 2370 | case COMP_SUCCESS: | 2372 | case COMP_SUCCESS: |
| 2371 | if (TRB_LEN(le32_to_cpu(event->transfer_len)) == 0) | 2373 | if (EVENT_TRB_LEN(le32_to_cpu(event->transfer_len)) == 0) |
| 2372 | break; | 2374 | break; |
| 2373 | if (xhci->quirks & XHCI_TRUST_TX_LENGTH) | 2375 | if (xhci->quirks & XHCI_TRUST_TX_LENGTH) |
| 2374 | trb_comp_code = COMP_SHORT_TX; | 2376 | trb_comp_code = COMP_SHORT_TX; |
| @@ -2461,14 +2463,21 @@ static int handle_tx_event(struct xhci_hcd *xhci, | |||
| 2461 | * TD list. | 2463 | * TD list. |
| 2462 | */ | 2464 | */ |
| 2463 | if (list_empty(&ep_ring->td_list)) { | 2465 | if (list_empty(&ep_ring->td_list)) { |
| 2464 | xhci_warn(xhci, "WARN Event TRB for slot %d ep %d " | 2466 | /* |
| 2465 | "with no TDs queued?\n", | 2467 | * A stopped endpoint may generate an extra completion |
| 2466 | TRB_TO_SLOT_ID(le32_to_cpu(event->flags)), | 2468 | * event if the device was suspended. Don't print |
| 2467 | ep_index); | 2469 | * warnings. |
| 2468 | xhci_dbg(xhci, "Event TRB with TRB type ID %u\n", | 2470 | */ |
| 2469 | (le32_to_cpu(event->flags) & | 2471 | if (!(trb_comp_code == COMP_STOP || |
| 2470 | TRB_TYPE_BITMASK)>>10); | 2472 | trb_comp_code == COMP_STOP_INVAL)) { |
| 2471 | xhci_print_trb_offsets(xhci, (union xhci_trb *) event); | 2473 | xhci_warn(xhci, "WARN Event TRB for slot %d ep %d with no TDs queued?\n", |
| 2474 | TRB_TO_SLOT_ID(le32_to_cpu(event->flags)), | ||
| 2475 | ep_index); | ||
| 2476 | xhci_dbg(xhci, "Event TRB with TRB type ID %u\n", | ||
| 2477 | (le32_to_cpu(event->flags) & | ||
| 2478 | TRB_TYPE_BITMASK)>>10); | ||
| 2479 | xhci_print_trb_offsets(xhci, (union xhci_trb *) event); | ||
| 2480 | } | ||
| 2472 | if (ep->skip) { | 2481 | if (ep->skip) { |
| 2473 | ep->skip = false; | 2482 | ep->skip = false; |
| 2474 | xhci_dbg(xhci, "td_list is empty while skip " | 2483 | xhci_dbg(xhci, "td_list is empty while skip " |
diff --git a/drivers/usb/host/xhci.c b/drivers/usb/host/xhci.c index 849470b18831..53b8f89a0b1c 100644 --- a/drivers/usb/host/xhci.c +++ b/drivers/usb/host/xhci.c | |||
| @@ -3779,6 +3779,28 @@ int xhci_address_device(struct usb_hcd *hcd, struct usb_device *udev) | |||
| 3779 | return 0; | 3779 | return 0; |
| 3780 | } | 3780 | } |
| 3781 | 3781 | ||
| 3782 | /* | ||
| 3783 | * Transfer the port index into real index in the HW port status | ||
| 3784 | * registers. Caculate offset between the port's PORTSC register | ||
| 3785 | * and port status base. Divide the number of per port register | ||
| 3786 | * to get the real index. The raw port number bases 1. | ||
| 3787 | */ | ||
| 3788 | int xhci_find_raw_port_number(struct usb_hcd *hcd, int port1) | ||
| 3789 | { | ||
| 3790 | struct xhci_hcd *xhci = hcd_to_xhci(hcd); | ||
| 3791 | __le32 __iomem *base_addr = &xhci->op_regs->port_status_base; | ||
| 3792 | __le32 __iomem *addr; | ||
| 3793 | int raw_port; | ||
| 3794 | |||
| 3795 | if (hcd->speed != HCD_USB3) | ||
| 3796 | addr = xhci->usb2_ports[port1 - 1]; | ||
| 3797 | else | ||
| 3798 | addr = xhci->usb3_ports[port1 - 1]; | ||
| 3799 | |||
| 3800 | raw_port = (addr - base_addr)/NUM_PORT_REGS + 1; | ||
| 3801 | return raw_port; | ||
| 3802 | } | ||
| 3803 | |||
| 3782 | #ifdef CONFIG_USB_SUSPEND | 3804 | #ifdef CONFIG_USB_SUSPEND |
| 3783 | 3805 | ||
| 3784 | /* BESL to HIRD Encoding array for USB2 LPM */ | 3806 | /* BESL to HIRD Encoding array for USB2 LPM */ |
diff --git a/drivers/usb/host/xhci.h b/drivers/usb/host/xhci.h index 2c510e4a7d4c..63582719e0fb 100644 --- a/drivers/usb/host/xhci.h +++ b/drivers/usb/host/xhci.h | |||
| @@ -972,6 +972,10 @@ struct xhci_transfer_event { | |||
| 972 | __le32 flags; | 972 | __le32 flags; |
| 973 | }; | 973 | }; |
| 974 | 974 | ||
| 975 | /* Transfer event TRB length bit mask */ | ||
| 976 | /* bits 0:23 */ | ||
| 977 | #define EVENT_TRB_LEN(p) ((p) & 0xffffff) | ||
| 978 | |||
| 975 | /** Transfer Event bit fields **/ | 979 | /** Transfer Event bit fields **/ |
| 976 | #define TRB_TO_EP_ID(p) (((p) >> 16) & 0x1f) | 980 | #define TRB_TO_EP_ID(p) (((p) >> 16) & 0x1f) |
| 977 | 981 | ||
| @@ -1829,6 +1833,7 @@ void xhci_test_and_clear_bit(struct xhci_hcd *xhci, __le32 __iomem **port_array, | |||
| 1829 | int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, u16 wIndex, | 1833 | int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, u16 wIndex, |
| 1830 | char *buf, u16 wLength); | 1834 | char *buf, u16 wLength); |
| 1831 | int xhci_hub_status_data(struct usb_hcd *hcd, char *buf); | 1835 | int xhci_hub_status_data(struct usb_hcd *hcd, char *buf); |
| 1836 | int xhci_find_raw_port_number(struct usb_hcd *hcd, int port1); | ||
| 1832 | 1837 | ||
| 1833 | #ifdef CONFIG_PM | 1838 | #ifdef CONFIG_PM |
| 1834 | int xhci_bus_suspend(struct usb_hcd *hcd); | 1839 | int xhci_bus_suspend(struct usb_hcd *hcd); |
diff --git a/drivers/usb/phy/Kconfig b/drivers/usb/phy/Kconfig index 65217a590068..90549382eba5 100644 --- a/drivers/usb/phy/Kconfig +++ b/drivers/usb/phy/Kconfig | |||
| @@ -38,6 +38,7 @@ config USB_ISP1301 | |||
| 38 | tristate "NXP ISP1301 USB transceiver support" | 38 | tristate "NXP ISP1301 USB transceiver support" |
| 39 | depends on USB || USB_GADGET | 39 | depends on USB || USB_GADGET |
| 40 | depends on I2C | 40 | depends on I2C |
| 41 | select USB_OTG_UTILS | ||
| 41 | help | 42 | help |
| 42 | Say Y here to add support for the NXP ISP1301 USB transceiver driver. | 43 | Say Y here to add support for the NXP ISP1301 USB transceiver driver. |
| 43 | This chip is typically used as USB transceiver for USB host, gadget | 44 | This chip is typically used as USB transceiver for USB host, gadget |
diff --git a/drivers/usb/serial/ftdi_sio.c b/drivers/usb/serial/ftdi_sio.c index d4809d551473..9886180e45f1 100644 --- a/drivers/usb/serial/ftdi_sio.c +++ b/drivers/usb/serial/ftdi_sio.c | |||
| @@ -640,6 +640,7 @@ static struct usb_device_id id_table_combined [] = { | |||
| 640 | { USB_DEVICE(FTDI_VID, FTDI_RM_CANVIEW_PID) }, | 640 | { USB_DEVICE(FTDI_VID, FTDI_RM_CANVIEW_PID) }, |
| 641 | { USB_DEVICE(ACTON_VID, ACTON_SPECTRAPRO_PID) }, | 641 | { USB_DEVICE(ACTON_VID, ACTON_SPECTRAPRO_PID) }, |
| 642 | { USB_DEVICE(CONTEC_VID, CONTEC_COM1USBH_PID) }, | 642 | { USB_DEVICE(CONTEC_VID, CONTEC_COM1USBH_PID) }, |
| 643 | { USB_DEVICE(MITSUBISHI_VID, MITSUBISHI_FXUSB_PID) }, | ||
| 643 | { USB_DEVICE(BANDB_VID, BANDB_USOTL4_PID) }, | 644 | { USB_DEVICE(BANDB_VID, BANDB_USOTL4_PID) }, |
| 644 | { USB_DEVICE(BANDB_VID, BANDB_USTL4_PID) }, | 645 | { USB_DEVICE(BANDB_VID, BANDB_USTL4_PID) }, |
| 645 | { USB_DEVICE(BANDB_VID, BANDB_USO9ML2_PID) }, | 646 | { USB_DEVICE(BANDB_VID, BANDB_USO9ML2_PID) }, |
diff --git a/drivers/usb/serial/ftdi_sio_ids.h b/drivers/usb/serial/ftdi_sio_ids.h index 9d359e189a64..e79861eeed4c 100644 --- a/drivers/usb/serial/ftdi_sio_ids.h +++ b/drivers/usb/serial/ftdi_sio_ids.h | |||
| @@ -584,6 +584,13 @@ | |||
| 584 | #define CONTEC_COM1USBH_PID 0x8311 /* COM-1(USB)H */ | 584 | #define CONTEC_COM1USBH_PID 0x8311 /* COM-1(USB)H */ |
| 585 | 585 | ||
| 586 | /* | 586 | /* |
| 587 | * Mitsubishi Electric Corp. (http://www.meau.com) | ||
| 588 | * Submitted by Konstantin Holoborodko | ||
| 589 | */ | ||
| 590 | #define MITSUBISHI_VID 0x06D3 | ||
| 591 | #define MITSUBISHI_FXUSB_PID 0x0284 /* USB/RS422 converters: FX-USB-AW/-BD */ | ||
| 592 | |||
| 593 | /* | ||
| 587 | * Definitions for B&B Electronics products. | 594 | * Definitions for B&B Electronics products. |
| 588 | */ | 595 | */ |
| 589 | #define BANDB_VID 0x0856 /* B&B Electronics Vendor ID */ | 596 | #define BANDB_VID 0x0856 /* B&B Electronics Vendor ID */ |
diff --git a/drivers/usb/serial/usb-serial.c b/drivers/usb/serial/usb-serial.c index 2e70efa08b77..5d9b178484fd 100644 --- a/drivers/usb/serial/usb-serial.c +++ b/drivers/usb/serial/usb-serial.c | |||
| @@ -903,6 +903,7 @@ static int usb_serial_probe(struct usb_interface *interface, | |||
| 903 | port->port.ops = &serial_port_ops; | 903 | port->port.ops = &serial_port_ops; |
| 904 | port->serial = serial; | 904 | port->serial = serial; |
| 905 | spin_lock_init(&port->lock); | 905 | spin_lock_init(&port->lock); |
| 906 | init_waitqueue_head(&port->delta_msr_wait); | ||
| 906 | /* Keep this for private driver use for the moment but | 907 | /* Keep this for private driver use for the moment but |
| 907 | should probably go away */ | 908 | should probably go away */ |
| 908 | INIT_WORK(&port->work, usb_serial_port_work); | 909 | INIT_WORK(&port->work, usb_serial_port_work); |
diff --git a/drivers/vhost/tcm_vhost.c b/drivers/vhost/tcm_vhost.c index 43fb11ee2e8d..2968b4934659 100644 --- a/drivers/vhost/tcm_vhost.c +++ b/drivers/vhost/tcm_vhost.c | |||
| @@ -60,6 +60,15 @@ enum { | |||
| 60 | VHOST_SCSI_VQ_IO = 2, | 60 | VHOST_SCSI_VQ_IO = 2, |
| 61 | }; | 61 | }; |
| 62 | 62 | ||
| 63 | /* | ||
| 64 | * VIRTIO_RING_F_EVENT_IDX seems broken. Not sure the bug is in | ||
| 65 | * kernel but disabling it helps. | ||
| 66 | * TODO: debug and remove the workaround. | ||
| 67 | */ | ||
| 68 | enum { | ||
| 69 | VHOST_SCSI_FEATURES = VHOST_FEATURES & (~VIRTIO_RING_F_EVENT_IDX) | ||
| 70 | }; | ||
| 71 | |||
| 63 | #define VHOST_SCSI_MAX_TARGET 256 | 72 | #define VHOST_SCSI_MAX_TARGET 256 |
| 64 | #define VHOST_SCSI_MAX_VQ 128 | 73 | #define VHOST_SCSI_MAX_VQ 128 |
| 65 | 74 | ||
| @@ -946,7 +955,7 @@ static void vhost_scsi_flush(struct vhost_scsi *vs) | |||
| 946 | 955 | ||
| 947 | static int vhost_scsi_set_features(struct vhost_scsi *vs, u64 features) | 956 | static int vhost_scsi_set_features(struct vhost_scsi *vs, u64 features) |
| 948 | { | 957 | { |
| 949 | if (features & ~VHOST_FEATURES) | 958 | if (features & ~VHOST_SCSI_FEATURES) |
| 950 | return -EOPNOTSUPP; | 959 | return -EOPNOTSUPP; |
| 951 | 960 | ||
| 952 | mutex_lock(&vs->dev.mutex); | 961 | mutex_lock(&vs->dev.mutex); |
| @@ -992,7 +1001,7 @@ static long vhost_scsi_ioctl(struct file *f, unsigned int ioctl, | |||
| 992 | return -EFAULT; | 1001 | return -EFAULT; |
| 993 | return 0; | 1002 | return 0; |
| 994 | case VHOST_GET_FEATURES: | 1003 | case VHOST_GET_FEATURES: |
| 995 | features = VHOST_FEATURES; | 1004 | features = VHOST_SCSI_FEATURES; |
| 996 | if (copy_to_user(featurep, &features, sizeof features)) | 1005 | if (copy_to_user(featurep, &features, sizeof features)) |
| 997 | return -EFAULT; | 1006 | return -EFAULT; |
| 998 | return 0; | 1007 | return 0; |
diff --git a/drivers/video/mxsfb.c b/drivers/video/mxsfb.c index 755556ca5b2d..45169cbaba6e 100644 --- a/drivers/video/mxsfb.c +++ b/drivers/video/mxsfb.c | |||
| @@ -169,6 +169,7 @@ struct mxsfb_info { | |||
| 169 | unsigned dotclk_delay; | 169 | unsigned dotclk_delay; |
| 170 | const struct mxsfb_devdata *devdata; | 170 | const struct mxsfb_devdata *devdata; |
| 171 | int mapped; | 171 | int mapped; |
| 172 | u32 sync; | ||
| 172 | }; | 173 | }; |
| 173 | 174 | ||
| 174 | #define mxsfb_is_v3(host) (host->devdata->ipversion == 3) | 175 | #define mxsfb_is_v3(host) (host->devdata->ipversion == 3) |
| @@ -456,9 +457,9 @@ static int mxsfb_set_par(struct fb_info *fb_info) | |||
| 456 | vdctrl0 |= VDCTRL0_HSYNC_ACT_HIGH; | 457 | vdctrl0 |= VDCTRL0_HSYNC_ACT_HIGH; |
| 457 | if (fb_info->var.sync & FB_SYNC_VERT_HIGH_ACT) | 458 | if (fb_info->var.sync & FB_SYNC_VERT_HIGH_ACT) |
| 458 | vdctrl0 |= VDCTRL0_VSYNC_ACT_HIGH; | 459 | vdctrl0 |= VDCTRL0_VSYNC_ACT_HIGH; |
| 459 | if (fb_info->var.sync & FB_SYNC_DATA_ENABLE_HIGH_ACT) | 460 | if (host->sync & MXSFB_SYNC_DATA_ENABLE_HIGH_ACT) |
| 460 | vdctrl0 |= VDCTRL0_ENABLE_ACT_HIGH; | 461 | vdctrl0 |= VDCTRL0_ENABLE_ACT_HIGH; |
| 461 | if (fb_info->var.sync & FB_SYNC_DOTCLK_FAILING_ACT) | 462 | if (host->sync & MXSFB_SYNC_DOTCLK_FAILING_ACT) |
| 462 | vdctrl0 |= VDCTRL0_DOTCLK_ACT_FAILING; | 463 | vdctrl0 |= VDCTRL0_DOTCLK_ACT_FAILING; |
| 463 | 464 | ||
| 464 | writel(vdctrl0, host->base + LCDC_VDCTRL0); | 465 | writel(vdctrl0, host->base + LCDC_VDCTRL0); |
| @@ -861,6 +862,8 @@ static int mxsfb_probe(struct platform_device *pdev) | |||
| 861 | 862 | ||
| 862 | INIT_LIST_HEAD(&fb_info->modelist); | 863 | INIT_LIST_HEAD(&fb_info->modelist); |
| 863 | 864 | ||
| 865 | host->sync = pdata->sync; | ||
| 866 | |||
| 864 | ret = mxsfb_init_fbinfo(host); | 867 | ret = mxsfb_init_fbinfo(host); |
| 865 | if (ret != 0) | 868 | if (ret != 0) |
| 866 | goto error_init_fb; | 869 | goto error_init_fb; |
diff --git a/drivers/video/omap/omapfb_main.c b/drivers/video/omap/omapfb_main.c index e31f5b33b501..d40612c31a98 100644 --- a/drivers/video/omap/omapfb_main.c +++ b/drivers/video/omap/omapfb_main.c | |||
| @@ -32,6 +32,8 @@ | |||
| 32 | 32 | ||
| 33 | #include <linux/omap-dma.h> | 33 | #include <linux/omap-dma.h> |
| 34 | 34 | ||
| 35 | #include <mach/hardware.h> | ||
| 36 | |||
| 35 | #include "omapfb.h" | 37 | #include "omapfb.h" |
| 36 | #include "lcdc.h" | 38 | #include "lcdc.h" |
| 37 | 39 | ||
diff --git a/drivers/video/omap2/displays/panel-tpo-td043mtea1.c b/drivers/video/omap2/displays/panel-tpo-td043mtea1.c index 6b6643911d29..048c98381ef6 100644 --- a/drivers/video/omap2/displays/panel-tpo-td043mtea1.c +++ b/drivers/video/omap2/displays/panel-tpo-td043mtea1.c | |||
| @@ -63,6 +63,9 @@ struct tpo_td043_device { | |||
| 63 | u32 power_on_resume:1; | 63 | u32 power_on_resume:1; |
| 64 | }; | 64 | }; |
| 65 | 65 | ||
| 66 | /* used to pass spi_device from SPI to DSS portion of the driver */ | ||
| 67 | static struct tpo_td043_device *g_tpo_td043; | ||
| 68 | |||
| 66 | static int tpo_td043_write(struct spi_device *spi, u8 addr, u8 data) | 69 | static int tpo_td043_write(struct spi_device *spi, u8 addr, u8 data) |
| 67 | { | 70 | { |
| 68 | struct spi_message m; | 71 | struct spi_message m; |
| @@ -403,7 +406,7 @@ static void tpo_td043_disable(struct omap_dss_device *dssdev) | |||
| 403 | 406 | ||
| 404 | static int tpo_td043_probe(struct omap_dss_device *dssdev) | 407 | static int tpo_td043_probe(struct omap_dss_device *dssdev) |
| 405 | { | 408 | { |
| 406 | struct tpo_td043_device *tpo_td043 = dev_get_drvdata(&dssdev->dev); | 409 | struct tpo_td043_device *tpo_td043 = g_tpo_td043; |
| 407 | int nreset_gpio = dssdev->reset_gpio; | 410 | int nreset_gpio = dssdev->reset_gpio; |
| 408 | int ret = 0; | 411 | int ret = 0; |
| 409 | 412 | ||
| @@ -440,6 +443,8 @@ static int tpo_td043_probe(struct omap_dss_device *dssdev) | |||
| 440 | if (ret) | 443 | if (ret) |
| 441 | dev_warn(&dssdev->dev, "failed to create sysfs files\n"); | 444 | dev_warn(&dssdev->dev, "failed to create sysfs files\n"); |
| 442 | 445 | ||
| 446 | dev_set_drvdata(&dssdev->dev, tpo_td043); | ||
| 447 | |||
| 443 | return 0; | 448 | return 0; |
| 444 | 449 | ||
| 445 | fail_gpio_req: | 450 | fail_gpio_req: |
| @@ -505,6 +510,9 @@ static int tpo_td043_spi_probe(struct spi_device *spi) | |||
| 505 | return -ENODEV; | 510 | return -ENODEV; |
| 506 | } | 511 | } |
| 507 | 512 | ||
| 513 | if (g_tpo_td043 != NULL) | ||
| 514 | return -EBUSY; | ||
| 515 | |||
| 508 | spi->bits_per_word = 16; | 516 | spi->bits_per_word = 16; |
| 509 | spi->mode = SPI_MODE_0; | 517 | spi->mode = SPI_MODE_0; |
| 510 | 518 | ||
| @@ -521,7 +529,7 @@ static int tpo_td043_spi_probe(struct spi_device *spi) | |||
| 521 | tpo_td043->spi = spi; | 529 | tpo_td043->spi = spi; |
| 522 | tpo_td043->nreset_gpio = dssdev->reset_gpio; | 530 | tpo_td043->nreset_gpio = dssdev->reset_gpio; |
| 523 | dev_set_drvdata(&spi->dev, tpo_td043); | 531 | dev_set_drvdata(&spi->dev, tpo_td043); |
| 524 | dev_set_drvdata(&dssdev->dev, tpo_td043); | 532 | g_tpo_td043 = tpo_td043; |
| 525 | 533 | ||
| 526 | omap_dss_register_driver(&tpo_td043_driver); | 534 | omap_dss_register_driver(&tpo_td043_driver); |
| 527 | 535 | ||
| @@ -534,6 +542,7 @@ static int tpo_td043_spi_remove(struct spi_device *spi) | |||
| 534 | 542 | ||
| 535 | omap_dss_unregister_driver(&tpo_td043_driver); | 543 | omap_dss_unregister_driver(&tpo_td043_driver); |
| 536 | kfree(tpo_td043); | 544 | kfree(tpo_td043); |
| 545 | g_tpo_td043 = NULL; | ||
| 537 | 546 | ||
| 538 | return 0; | 547 | return 0; |
| 539 | } | 548 | } |
diff --git a/drivers/video/omap2/dss/dss_features.c b/drivers/video/omap2/dss/dss_features.c index d7d66ef5cb58..7f791aeda4d2 100644 --- a/drivers/video/omap2/dss/dss_features.c +++ b/drivers/video/omap2/dss/dss_features.c | |||
| @@ -202,12 +202,10 @@ static const enum omap_dss_output_id omap3630_dss_supported_outputs[] = { | |||
| 202 | 202 | ||
| 203 | static const enum omap_dss_output_id omap4_dss_supported_outputs[] = { | 203 | static const enum omap_dss_output_id omap4_dss_supported_outputs[] = { |
| 204 | /* OMAP_DSS_CHANNEL_LCD */ | 204 | /* OMAP_DSS_CHANNEL_LCD */ |
| 205 | OMAP_DSS_OUTPUT_DPI | OMAP_DSS_OUTPUT_DBI | | 205 | OMAP_DSS_OUTPUT_DBI | OMAP_DSS_OUTPUT_DSI1, |
| 206 | OMAP_DSS_OUTPUT_DSI1, | ||
| 207 | 206 | ||
| 208 | /* OMAP_DSS_CHANNEL_DIGIT */ | 207 | /* OMAP_DSS_CHANNEL_DIGIT */ |
| 209 | OMAP_DSS_OUTPUT_VENC | OMAP_DSS_OUTPUT_HDMI | | 208 | OMAP_DSS_OUTPUT_VENC | OMAP_DSS_OUTPUT_HDMI, |
| 210 | OMAP_DSS_OUTPUT_DPI, | ||
| 211 | 209 | ||
| 212 | /* OMAP_DSS_CHANNEL_LCD2 */ | 210 | /* OMAP_DSS_CHANNEL_LCD2 */ |
| 213 | OMAP_DSS_OUTPUT_DPI | OMAP_DSS_OUTPUT_DBI | | 211 | OMAP_DSS_OUTPUT_DPI | OMAP_DSS_OUTPUT_DBI | |
diff --git a/drivers/xen/Kconfig b/drivers/xen/Kconfig index 5a32232cf7c1..67af155cf602 100644 --- a/drivers/xen/Kconfig +++ b/drivers/xen/Kconfig | |||
| @@ -182,7 +182,7 @@ config XEN_PRIVCMD | |||
| 182 | 182 | ||
| 183 | config XEN_STUB | 183 | config XEN_STUB |
| 184 | bool "Xen stub drivers" | 184 | bool "Xen stub drivers" |
| 185 | depends on XEN && X86_64 | 185 | depends on XEN && X86_64 && BROKEN |
| 186 | default n | 186 | default n |
| 187 | help | 187 | help |
| 188 | Allow kernel to install stub drivers, to reserve space for Xen drivers, | 188 | Allow kernel to install stub drivers, to reserve space for Xen drivers, |
diff --git a/drivers/xen/events.c b/drivers/xen/events.c index d17aa41a9041..aa85881d17b2 100644 --- a/drivers/xen/events.c +++ b/drivers/xen/events.c | |||
| @@ -403,11 +403,23 @@ static void unmask_evtchn(int port) | |||
| 403 | 403 | ||
| 404 | if (unlikely((cpu != cpu_from_evtchn(port)))) | 404 | if (unlikely((cpu != cpu_from_evtchn(port)))) |
| 405 | do_hypercall = 1; | 405 | do_hypercall = 1; |
| 406 | else | 406 | else { |
| 407 | /* | ||
| 408 | * Need to clear the mask before checking pending to | ||
| 409 | * avoid a race with an event becoming pending. | ||
| 410 | * | ||
| 411 | * EVTCHNOP_unmask will only trigger an upcall if the | ||
| 412 | * mask bit was set, so if a hypercall is needed | ||
| 413 | * remask the event. | ||
| 414 | */ | ||
| 415 | sync_clear_bit(port, BM(&s->evtchn_mask[0])); | ||
| 407 | evtchn_pending = sync_test_bit(port, BM(&s->evtchn_pending[0])); | 416 | evtchn_pending = sync_test_bit(port, BM(&s->evtchn_pending[0])); |
| 408 | 417 | ||
| 409 | if (unlikely(evtchn_pending && xen_hvm_domain())) | 418 | if (unlikely(evtchn_pending && xen_hvm_domain())) { |
| 410 | do_hypercall = 1; | 419 | sync_set_bit(port, BM(&s->evtchn_mask[0])); |
| 420 | do_hypercall = 1; | ||
| 421 | } | ||
| 422 | } | ||
| 411 | 423 | ||
| 412 | /* Slow path (hypercall) if this is a non-local port or if this is | 424 | /* Slow path (hypercall) if this is a non-local port or if this is |
| 413 | * an hvm domain and an event is pending (hvm domains don't have | 425 | * an hvm domain and an event is pending (hvm domains don't have |
| @@ -418,8 +430,6 @@ static void unmask_evtchn(int port) | |||
| 418 | } else { | 430 | } else { |
| 419 | struct vcpu_info *vcpu_info = __this_cpu_read(xen_vcpu); | 431 | struct vcpu_info *vcpu_info = __this_cpu_read(xen_vcpu); |
| 420 | 432 | ||
| 421 | sync_clear_bit(port, BM(&s->evtchn_mask[0])); | ||
| 422 | |||
| 423 | /* | 433 | /* |
| 424 | * The following is basically the equivalent of | 434 | * The following is basically the equivalent of |
| 425 | * 'hw_resend_irq'. Just like a real IO-APIC we 'lose | 435 | * 'hw_resend_irq'. Just like a real IO-APIC we 'lose |
diff --git a/drivers/xen/fallback.c b/drivers/xen/fallback.c index 0ef7c4d40f86..b04fb64c5a91 100644 --- a/drivers/xen/fallback.c +++ b/drivers/xen/fallback.c | |||
| @@ -44,7 +44,7 @@ int xen_event_channel_op_compat(int cmd, void *arg) | |||
| 44 | } | 44 | } |
| 45 | EXPORT_SYMBOL_GPL(xen_event_channel_op_compat); | 45 | EXPORT_SYMBOL_GPL(xen_event_channel_op_compat); |
| 46 | 46 | ||
| 47 | int HYPERVISOR_physdev_op_compat(int cmd, void *arg) | 47 | int xen_physdev_op_compat(int cmd, void *arg) |
| 48 | { | 48 | { |
| 49 | struct physdev_op op; | 49 | struct physdev_op op; |
| 50 | int rc; | 50 | int rc; |
| @@ -78,3 +78,4 @@ int HYPERVISOR_physdev_op_compat(int cmd, void *arg) | |||
| 78 | 78 | ||
| 79 | return rc; | 79 | return rc; |
| 80 | } | 80 | } |
| 81 | EXPORT_SYMBOL_GPL(xen_physdev_op_compat); | ||
diff --git a/drivers/xen/xen-acpi-processor.c b/drivers/xen/xen-acpi-processor.c index f3278a6603ca..90e34ac7e522 100644 --- a/drivers/xen/xen-acpi-processor.c +++ b/drivers/xen/xen-acpi-processor.c | |||
| @@ -505,6 +505,9 @@ static int __init xen_acpi_processor_init(void) | |||
| 505 | 505 | ||
| 506 | pr = per_cpu(processors, i); | 506 | pr = per_cpu(processors, i); |
| 507 | perf = per_cpu_ptr(acpi_perf_data, i); | 507 | perf = per_cpu_ptr(acpi_perf_data, i); |
| 508 | if (!pr) | ||
| 509 | continue; | ||
| 510 | |||
| 508 | pr->performance = perf; | 511 | pr->performance = perf; |
| 509 | rc = acpi_processor_get_performance_info(pr); | 512 | rc = acpi_processor_get_performance_info(pr); |
| 510 | if (rc) | 513 | if (rc) |
diff --git a/drivers/xen/xen-pciback/pci_stub.c b/drivers/xen/xen-pciback/pci_stub.c index 9204126f1560..a2278ba7fb27 100644 --- a/drivers/xen/xen-pciback/pci_stub.c +++ b/drivers/xen/xen-pciback/pci_stub.c | |||
| @@ -17,6 +17,7 @@ | |||
| 17 | #include <xen/events.h> | 17 | #include <xen/events.h> |
| 18 | #include <asm/xen/pci.h> | 18 | #include <asm/xen/pci.h> |
| 19 | #include <asm/xen/hypervisor.h> | 19 | #include <asm/xen/hypervisor.h> |
| 20 | #include <xen/interface/physdev.h> | ||
| 20 | #include "pciback.h" | 21 | #include "pciback.h" |
| 21 | #include "conf_space.h" | 22 | #include "conf_space.h" |
| 22 | #include "conf_space_quirks.h" | 23 | #include "conf_space_quirks.h" |
| @@ -85,37 +86,52 @@ static struct pcistub_device *pcistub_device_alloc(struct pci_dev *dev) | |||
| 85 | static void pcistub_device_release(struct kref *kref) | 86 | static void pcistub_device_release(struct kref *kref) |
| 86 | { | 87 | { |
| 87 | struct pcistub_device *psdev; | 88 | struct pcistub_device *psdev; |
| 89 | struct pci_dev *dev; | ||
| 88 | struct xen_pcibk_dev_data *dev_data; | 90 | struct xen_pcibk_dev_data *dev_data; |
| 89 | 91 | ||
| 90 | psdev = container_of(kref, struct pcistub_device, kref); | 92 | psdev = container_of(kref, struct pcistub_device, kref); |
| 91 | dev_data = pci_get_drvdata(psdev->dev); | 93 | dev = psdev->dev; |
| 94 | dev_data = pci_get_drvdata(dev); | ||
| 92 | 95 | ||
| 93 | dev_dbg(&psdev->dev->dev, "pcistub_device_release\n"); | 96 | dev_dbg(&dev->dev, "pcistub_device_release\n"); |
| 94 | 97 | ||
| 95 | xen_unregister_device_domain_owner(psdev->dev); | 98 | xen_unregister_device_domain_owner(dev); |
| 96 | 99 | ||
| 97 | /* Call the reset function which does not take lock as this | 100 | /* Call the reset function which does not take lock as this |
| 98 | * is called from "unbind" which takes a device_lock mutex. | 101 | * is called from "unbind" which takes a device_lock mutex. |
| 99 | */ | 102 | */ |
| 100 | __pci_reset_function_locked(psdev->dev); | 103 | __pci_reset_function_locked(dev); |
| 101 | if (pci_load_and_free_saved_state(psdev->dev, | 104 | if (pci_load_and_free_saved_state(dev, &dev_data->pci_saved_state)) |
| 102 | &dev_data->pci_saved_state)) { | 105 | dev_dbg(&dev->dev, "Could not reload PCI state\n"); |
| 103 | dev_dbg(&psdev->dev->dev, "Could not reload PCI state\n"); | 106 | else |
| 104 | } else | 107 | pci_restore_state(dev); |
| 105 | pci_restore_state(psdev->dev); | 108 | |
| 109 | if (pci_find_capability(dev, PCI_CAP_ID_MSIX)) { | ||
| 110 | struct physdev_pci_device ppdev = { | ||
| 111 | .seg = pci_domain_nr(dev->bus), | ||
| 112 | .bus = dev->bus->number, | ||
| 113 | .devfn = dev->devfn | ||
| 114 | }; | ||
| 115 | int err = HYPERVISOR_physdev_op(PHYSDEVOP_release_msix, | ||
| 116 | &ppdev); | ||
| 117 | |||
| 118 | if (err) | ||
| 119 | dev_warn(&dev->dev, "MSI-X release failed (%d)\n", | ||
| 120 | err); | ||
| 121 | } | ||
| 106 | 122 | ||
| 107 | /* Disable the device */ | 123 | /* Disable the device */ |
| 108 | xen_pcibk_reset_device(psdev->dev); | 124 | xen_pcibk_reset_device(dev); |
| 109 | 125 | ||
| 110 | kfree(dev_data); | 126 | kfree(dev_data); |
| 111 | pci_set_drvdata(psdev->dev, NULL); | 127 | pci_set_drvdata(dev, NULL); |
| 112 | 128 | ||
| 113 | /* Clean-up the device */ | 129 | /* Clean-up the device */ |
| 114 | xen_pcibk_config_free_dyn_fields(psdev->dev); | 130 | xen_pcibk_config_free_dyn_fields(dev); |
| 115 | xen_pcibk_config_free_dev(psdev->dev); | 131 | xen_pcibk_config_free_dev(dev); |
| 116 | 132 | ||
| 117 | psdev->dev->dev_flags &= ~PCI_DEV_FLAGS_ASSIGNED; | 133 | dev->dev_flags &= ~PCI_DEV_FLAGS_ASSIGNED; |
| 118 | pci_dev_put(psdev->dev); | 134 | pci_dev_put(dev); |
| 119 | 135 | ||
| 120 | kfree(psdev); | 136 | kfree(psdev); |
| 121 | } | 137 | } |
| @@ -355,6 +371,19 @@ static int pcistub_init_device(struct pci_dev *dev) | |||
| 355 | if (err) | 371 | if (err) |
| 356 | goto config_release; | 372 | goto config_release; |
| 357 | 373 | ||
| 374 | if (pci_find_capability(dev, PCI_CAP_ID_MSIX)) { | ||
| 375 | struct physdev_pci_device ppdev = { | ||
| 376 | .seg = pci_domain_nr(dev->bus), | ||
| 377 | .bus = dev->bus->number, | ||
| 378 | .devfn = dev->devfn | ||
| 379 | }; | ||
| 380 | |||
| 381 | err = HYPERVISOR_physdev_op(PHYSDEVOP_prepare_msix, &ppdev); | ||
| 382 | if (err) | ||
| 383 | dev_err(&dev->dev, "MSI-X preparation failed (%d)\n", | ||
| 384 | err); | ||
| 385 | } | ||
| 386 | |||
| 358 | /* We need the device active to save the state. */ | 387 | /* We need the device active to save the state. */ |
| 359 | dev_dbg(&dev->dev, "save state of device\n"); | 388 | dev_dbg(&dev->dev, "save state of device\n"); |
| 360 | pci_save_state(dev); | 389 | pci_save_state(dev); |
