diff options
Diffstat (limited to 'drivers')
318 files changed, 6179 insertions, 2121 deletions
diff --git a/drivers/ata/ahci.c b/drivers/ata/ahci.c index 328826381a2..b8d96ce37fc 100644 --- a/drivers/ata/ahci.c +++ b/drivers/ata/ahci.c | |||
| @@ -260,6 +260,7 @@ static const struct pci_device_id ahci_pci_tbl[] = { | |||
| 260 | { PCI_VDEVICE(INTEL, 0x1d02), board_ahci }, /* PBG AHCI */ | 260 | { PCI_VDEVICE(INTEL, 0x1d02), board_ahci }, /* PBG AHCI */ |
| 261 | { PCI_VDEVICE(INTEL, 0x1d04), board_ahci }, /* PBG RAID */ | 261 | { PCI_VDEVICE(INTEL, 0x1d04), board_ahci }, /* PBG RAID */ |
| 262 | { PCI_VDEVICE(INTEL, 0x1d06), board_ahci }, /* PBG RAID */ | 262 | { PCI_VDEVICE(INTEL, 0x1d06), board_ahci }, /* PBG RAID */ |
| 263 | { PCI_VDEVICE(INTEL, 0x2323), board_ahci }, /* DH89xxCC AHCI */ | ||
| 263 | 264 | ||
| 264 | /* JMicron 360/1/3/5/6, match class to avoid IDE function */ | 265 | /* JMicron 360/1/3/5/6, match class to avoid IDE function */ |
| 265 | { PCI_VENDOR_ID_JMICRON, PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID, | 266 | { PCI_VENDOR_ID_JMICRON, PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID, |
| @@ -379,6 +380,8 @@ static const struct pci_device_id ahci_pci_tbl[] = { | |||
| 379 | { PCI_VDEVICE(MARVELL, 0x6145), board_ahci_mv }, /* 6145 */ | 380 | { PCI_VDEVICE(MARVELL, 0x6145), board_ahci_mv }, /* 6145 */ |
| 380 | { PCI_VDEVICE(MARVELL, 0x6121), board_ahci_mv }, /* 6121 */ | 381 | { PCI_VDEVICE(MARVELL, 0x6121), board_ahci_mv }, /* 6121 */ |
| 381 | { PCI_DEVICE(0x1b4b, 0x9123), | 382 | { PCI_DEVICE(0x1b4b, 0x9123), |
| 383 | .class = PCI_CLASS_STORAGE_SATA_AHCI, | ||
| 384 | .class_mask = 0xffffff, | ||
| 382 | .driver_data = board_ahci_yes_fbs }, /* 88se9128 */ | 385 | .driver_data = board_ahci_yes_fbs }, /* 88se9128 */ |
| 383 | 386 | ||
| 384 | /* Promise */ | 387 | /* Promise */ |
diff --git a/drivers/ata/libata-core.c b/drivers/ata/libata-core.c index a31fe96f7de..d4e52e21485 100644 --- a/drivers/ata/libata-core.c +++ b/drivers/ata/libata-core.c | |||
| @@ -4138,6 +4138,7 @@ static const struct ata_blacklist_entry ata_device_blacklist [] = { | |||
| 4138 | * device and controller are SATA. | 4138 | * device and controller are SATA. |
| 4139 | */ | 4139 | */ |
| 4140 | { "PIONEER DVD-RW DVRTD08", "1.00", ATA_HORKAGE_NOSETXFER }, | 4140 | { "PIONEER DVD-RW DVRTD08", "1.00", ATA_HORKAGE_NOSETXFER }, |
| 4141 | { "PIONEER DVD-RW DVR-212D", "1.28", ATA_HORKAGE_NOSETXFER }, | ||
| 4141 | 4142 | ||
| 4142 | /* End Marker */ | 4143 | /* End Marker */ |
| 4143 | { } | 4144 | { } |
diff --git a/drivers/ata/libata-scsi.c b/drivers/ata/libata-scsi.c index 5defc74973d..600f6353ecf 100644 --- a/drivers/ata/libata-scsi.c +++ b/drivers/ata/libata-scsi.c | |||
| @@ -1099,9 +1099,9 @@ static int ata_scsi_dev_config(struct scsi_device *sdev, | |||
| 1099 | struct request_queue *q = sdev->request_queue; | 1099 | struct request_queue *q = sdev->request_queue; |
| 1100 | void *buf; | 1100 | void *buf; |
| 1101 | 1101 | ||
| 1102 | /* set the min alignment and padding */ | 1102 | sdev->sector_size = ATA_SECT_SIZE; |
| 1103 | blk_queue_update_dma_alignment(sdev->request_queue, | 1103 | |
| 1104 | ATA_DMA_PAD_SZ - 1); | 1104 | /* set DMA padding */ |
| 1105 | blk_queue_update_dma_pad(sdev->request_queue, | 1105 | blk_queue_update_dma_pad(sdev->request_queue, |
| 1106 | ATA_DMA_PAD_SZ - 1); | 1106 | ATA_DMA_PAD_SZ - 1); |
| 1107 | 1107 | ||
| @@ -1115,13 +1115,25 @@ static int ata_scsi_dev_config(struct scsi_device *sdev, | |||
| 1115 | 1115 | ||
| 1116 | blk_queue_dma_drain(q, atapi_drain_needed, buf, ATAPI_MAX_DRAIN); | 1116 | blk_queue_dma_drain(q, atapi_drain_needed, buf, ATAPI_MAX_DRAIN); |
| 1117 | } else { | 1117 | } else { |
| 1118 | /* ATA devices must be sector aligned */ | ||
| 1119 | sdev->sector_size = ata_id_logical_sector_size(dev->id); | 1118 | sdev->sector_size = ata_id_logical_sector_size(dev->id); |
| 1120 | blk_queue_update_dma_alignment(sdev->request_queue, | ||
| 1121 | sdev->sector_size - 1); | ||
| 1122 | sdev->manage_start_stop = 1; | 1119 | sdev->manage_start_stop = 1; |
| 1123 | } | 1120 | } |
| 1124 | 1121 | ||
| 1122 | /* | ||
| 1123 | * ata_pio_sectors() expects buffer for each sector to not cross | ||
| 1124 | * page boundary. Enforce it by requiring buffers to be sector | ||
| 1125 | * aligned, which works iff sector_size is not larger than | ||
| 1126 | * PAGE_SIZE. ATAPI devices also need the alignment as | ||
| 1127 | * IDENTIFY_PACKET is executed as ATA_PROT_PIO. | ||
| 1128 | */ | ||
| 1129 | if (sdev->sector_size > PAGE_SIZE) | ||
| 1130 | ata_dev_printk(dev, KERN_WARNING, | ||
| 1131 | "sector_size=%u > PAGE_SIZE, PIO may malfunction\n", | ||
| 1132 | sdev->sector_size); | ||
| 1133 | |||
| 1134 | blk_queue_update_dma_alignment(sdev->request_queue, | ||
| 1135 | sdev->sector_size - 1); | ||
| 1136 | |||
| 1125 | if (dev->flags & ATA_DFLAG_AN) | 1137 | if (dev->flags & ATA_DFLAG_AN) |
| 1126 | set_bit(SDEV_EVT_MEDIA_CHANGE, sdev->supported_events); | 1138 | set_bit(SDEV_EVT_MEDIA_CHANGE, sdev->supported_events); |
| 1127 | 1139 | ||
diff --git a/drivers/ata/pata_hpt366.c b/drivers/ata/pata_hpt366.c index d7e57db36bc..538ec38ba99 100644 --- a/drivers/ata/pata_hpt366.c +++ b/drivers/ata/pata_hpt366.c | |||
| @@ -25,7 +25,7 @@ | |||
| 25 | #include <linux/libata.h> | 25 | #include <linux/libata.h> |
| 26 | 26 | ||
| 27 | #define DRV_NAME "pata_hpt366" | 27 | #define DRV_NAME "pata_hpt366" |
| 28 | #define DRV_VERSION "0.6.9" | 28 | #define DRV_VERSION "0.6.10" |
| 29 | 29 | ||
| 30 | struct hpt_clock { | 30 | struct hpt_clock { |
| 31 | u8 xfer_mode; | 31 | u8 xfer_mode; |
| @@ -160,8 +160,8 @@ static int hpt_dma_blacklisted(const struct ata_device *dev, char *modestr, | |||
| 160 | 160 | ||
| 161 | while (list[i] != NULL) { | 161 | while (list[i] != NULL) { |
| 162 | if (!strcmp(list[i], model_num)) { | 162 | if (!strcmp(list[i], model_num)) { |
| 163 | printk(KERN_WARNING DRV_NAME ": %s is not supported for %s.\n", | 163 | pr_warning(DRV_NAME ": %s is not supported for %s.\n", |
| 164 | modestr, list[i]); | 164 | modestr, list[i]); |
| 165 | return 1; | 165 | return 1; |
| 166 | } | 166 | } |
| 167 | i++; | 167 | i++; |
diff --git a/drivers/ata/pata_hpt37x.c b/drivers/ata/pata_hpt37x.c index efdd18bc866..4c5b5183225 100644 --- a/drivers/ata/pata_hpt37x.c +++ b/drivers/ata/pata_hpt37x.c | |||
| @@ -24,7 +24,7 @@ | |||
| 24 | #include <linux/libata.h> | 24 | #include <linux/libata.h> |
| 25 | 25 | ||
| 26 | #define DRV_NAME "pata_hpt37x" | 26 | #define DRV_NAME "pata_hpt37x" |
| 27 | #define DRV_VERSION "0.6.18" | 27 | #define DRV_VERSION "0.6.22" |
| 28 | 28 | ||
| 29 | struct hpt_clock { | 29 | struct hpt_clock { |
| 30 | u8 xfer_speed; | 30 | u8 xfer_speed; |
| @@ -229,8 +229,8 @@ static int hpt_dma_blacklisted(const struct ata_device *dev, char *modestr, | |||
| 229 | 229 | ||
| 230 | while (list[i] != NULL) { | 230 | while (list[i] != NULL) { |
| 231 | if (!strcmp(list[i], model_num)) { | 231 | if (!strcmp(list[i], model_num)) { |
| 232 | printk(KERN_WARNING DRV_NAME ": %s is not supported for %s.\n", | 232 | pr_warning(DRV_NAME ": %s is not supported for %s.\n", |
| 233 | modestr, list[i]); | 233 | modestr, list[i]); |
| 234 | return 1; | 234 | return 1; |
| 235 | } | 235 | } |
| 236 | i++; | 236 | i++; |
| @@ -642,7 +642,6 @@ static struct ata_port_operations hpt372_port_ops = { | |||
| 642 | static struct ata_port_operations hpt374_fn1_port_ops = { | 642 | static struct ata_port_operations hpt374_fn1_port_ops = { |
| 643 | .inherits = &hpt372_port_ops, | 643 | .inherits = &hpt372_port_ops, |
| 644 | .cable_detect = hpt374_fn1_cable_detect, | 644 | .cable_detect = hpt374_fn1_cable_detect, |
| 645 | .prereset = hpt37x_pre_reset, | ||
| 646 | }; | 645 | }; |
| 647 | 646 | ||
| 648 | /** | 647 | /** |
| @@ -803,7 +802,7 @@ static int hpt37x_init_one(struct pci_dev *dev, const struct pci_device_id *id) | |||
| 803 | .udma_mask = ATA_UDMA6, | 802 | .udma_mask = ATA_UDMA6, |
| 804 | .port_ops = &hpt302_port_ops | 803 | .port_ops = &hpt302_port_ops |
| 805 | }; | 804 | }; |
| 806 | /* HPT374 - UDMA100, function 1 uses different prereset method */ | 805 | /* HPT374 - UDMA100, function 1 uses different cable_detect method */ |
| 807 | static const struct ata_port_info info_hpt374_fn0 = { | 806 | static const struct ata_port_info info_hpt374_fn0 = { |
| 808 | .flags = ATA_FLAG_SLAVE_POSS, | 807 | .flags = ATA_FLAG_SLAVE_POSS, |
| 809 | .pio_mask = ATA_PIO4, | 808 | .pio_mask = ATA_PIO4, |
| @@ -838,7 +837,8 @@ static int hpt37x_init_one(struct pci_dev *dev, const struct pci_device_id *id) | |||
| 838 | if (rc) | 837 | if (rc) |
| 839 | return rc; | 838 | return rc; |
| 840 | 839 | ||
| 841 | if (dev->device == PCI_DEVICE_ID_TTI_HPT366) { | 840 | switch (dev->device) { |
| 841 | case PCI_DEVICE_ID_TTI_HPT366: | ||
| 842 | /* May be a later chip in disguise. Check */ | 842 | /* May be a later chip in disguise. Check */ |
| 843 | /* Older chips are in the HPT366 driver. Ignore them */ | 843 | /* Older chips are in the HPT366 driver. Ignore them */ |
| 844 | if (rev < 3) | 844 | if (rev < 3) |
| @@ -863,54 +863,50 @@ static int hpt37x_init_one(struct pci_dev *dev, const struct pci_device_id *id) | |||
| 863 | chip_table = &hpt372; | 863 | chip_table = &hpt372; |
| 864 | break; | 864 | break; |
| 865 | default: | 865 | default: |
| 866 | printk(KERN_ERR "pata_hpt37x: Unknown HPT366 subtype, " | 866 | pr_err(DRV_NAME ": Unknown HPT366 subtype, " |
| 867 | "please report (%d).\n", rev); | 867 | "please report (%d).\n", rev); |
| 868 | return -ENODEV; | 868 | return -ENODEV; |
| 869 | } | 869 | } |
| 870 | } else { | 870 | break; |
| 871 | switch (dev->device) { | 871 | case PCI_DEVICE_ID_TTI_HPT372: |
| 872 | case PCI_DEVICE_ID_TTI_HPT372: | 872 | /* 372N if rev >= 2 */ |
| 873 | /* 372N if rev >= 2 */ | 873 | if (rev >= 2) |
| 874 | if (rev >= 2) | 874 | return -ENODEV; |
| 875 | return -ENODEV; | 875 | ppi[0] = &info_hpt372; |
| 876 | ppi[0] = &info_hpt372; | 876 | chip_table = &hpt372a; |
| 877 | chip_table = &hpt372a; | 877 | break; |
| 878 | break; | 878 | case PCI_DEVICE_ID_TTI_HPT302: |
| 879 | case PCI_DEVICE_ID_TTI_HPT302: | 879 | /* 302N if rev > 1 */ |
| 880 | /* 302N if rev > 1 */ | 880 | if (rev > 1) |
| 881 | if (rev > 1) | 881 | return -ENODEV; |
| 882 | return -ENODEV; | 882 | ppi[0] = &info_hpt302; |
| 883 | ppi[0] = &info_hpt302; | 883 | /* Check this */ |
| 884 | /* Check this */ | 884 | chip_table = &hpt302; |
| 885 | chip_table = &hpt302; | 885 | break; |
| 886 | break; | 886 | case PCI_DEVICE_ID_TTI_HPT371: |
| 887 | case PCI_DEVICE_ID_TTI_HPT371: | 887 | if (rev > 1) |
| 888 | if (rev > 1) | 888 | return -ENODEV; |
| 889 | return -ENODEV; | 889 | ppi[0] = &info_hpt302; |
| 890 | ppi[0] = &info_hpt302; | 890 | chip_table = &hpt371; |
| 891 | chip_table = &hpt371; | 891 | /* |
| 892 | /* | 892 | * Single channel device, master is not present but the BIOS |
| 893 | * Single channel device, master is not present | 893 | * (or us for non x86) must mark it absent |
| 894 | * but the BIOS (or us for non x86) must mark it | 894 | */ |
| 895 | * absent | 895 | pci_read_config_byte(dev, 0x50, &mcr1); |
| 896 | */ | 896 | mcr1 &= ~0x04; |
| 897 | pci_read_config_byte(dev, 0x50, &mcr1); | 897 | pci_write_config_byte(dev, 0x50, mcr1); |
| 898 | mcr1 &= ~0x04; | 898 | break; |
| 899 | pci_write_config_byte(dev, 0x50, mcr1); | 899 | case PCI_DEVICE_ID_TTI_HPT374: |
| 900 | break; | 900 | chip_table = &hpt374; |
| 901 | case PCI_DEVICE_ID_TTI_HPT374: | 901 | if (!(PCI_FUNC(dev->devfn) & 1)) |
| 902 | chip_table = &hpt374; | 902 | *ppi = &info_hpt374_fn0; |
| 903 | if (!(PCI_FUNC(dev->devfn) & 1)) | 903 | else |
| 904 | *ppi = &info_hpt374_fn0; | 904 | *ppi = &info_hpt374_fn1; |
| 905 | else | 905 | break; |
| 906 | *ppi = &info_hpt374_fn1; | 906 | default: |
| 907 | break; | 907 | pr_err(DRV_NAME ": PCI table is bogus, please report (%d).\n", |
| 908 | default: | 908 | dev->device); |
| 909 | printk(KERN_ERR | 909 | return -ENODEV; |
| 910 | "pata_hpt37x: PCI table is bogus, please report (%d).\n", | ||
| 911 | dev->device); | ||
| 912 | return -ENODEV; | ||
| 913 | } | ||
| 914 | } | 910 | } |
| 915 | /* Ok so this is a chip we support */ | 911 | /* Ok so this is a chip we support */ |
| 916 | 912 | ||
| @@ -957,8 +953,7 @@ static int hpt37x_init_one(struct pci_dev *dev, const struct pci_device_id *id) | |||
| 957 | u8 sr; | 953 | u8 sr; |
| 958 | u32 total = 0; | 954 | u32 total = 0; |
| 959 | 955 | ||
| 960 | printk(KERN_WARNING | 956 | pr_warning(DRV_NAME ": BIOS has not set timing clocks.\n"); |
| 961 | "pata_hpt37x: BIOS has not set timing clocks.\n"); | ||
| 962 | 957 | ||
| 963 | /* This is the process the HPT371 BIOS is reported to use */ | 958 | /* This is the process the HPT371 BIOS is reported to use */ |
| 964 | for (i = 0; i < 128; i++) { | 959 | for (i = 0; i < 128; i++) { |
| @@ -1014,7 +1009,7 @@ static int hpt37x_init_one(struct pci_dev *dev, const struct pci_device_id *id) | |||
| 1014 | (f_high << 16) | f_low | 0x100); | 1009 | (f_high << 16) | f_low | 0x100); |
| 1015 | } | 1010 | } |
| 1016 | if (adjust == 8) { | 1011 | if (adjust == 8) { |
| 1017 | printk(KERN_ERR "pata_hpt37x: DPLL did not stabilize!\n"); | 1012 | pr_err(DRV_NAME ": DPLL did not stabilize!\n"); |
| 1018 | return -ENODEV; | 1013 | return -ENODEV; |
| 1019 | } | 1014 | } |
| 1020 | if (dpll == 3) | 1015 | if (dpll == 3) |
| @@ -1022,8 +1017,8 @@ static int hpt37x_init_one(struct pci_dev *dev, const struct pci_device_id *id) | |||
| 1022 | else | 1017 | else |
| 1023 | private_data = (void *)hpt37x_timings_50; | 1018 | private_data = (void *)hpt37x_timings_50; |
| 1024 | 1019 | ||
| 1025 | printk(KERN_INFO "pata_hpt37x: bus clock %dMHz, using %dMHz DPLL.\n", | 1020 | pr_info(DRV_NAME ": bus clock %dMHz, using %dMHz DPLL.\n", |
| 1026 | MHz[clock_slot], MHz[dpll]); | 1021 | MHz[clock_slot], MHz[dpll]); |
| 1027 | } else { | 1022 | } else { |
| 1028 | private_data = (void *)chip_table->clocks[clock_slot]; | 1023 | private_data = (void *)chip_table->clocks[clock_slot]; |
| 1029 | /* | 1024 | /* |
| @@ -1036,8 +1031,9 @@ static int hpt37x_init_one(struct pci_dev *dev, const struct pci_device_id *id) | |||
| 1036 | ppi[0] = &info_hpt370_33; | 1031 | ppi[0] = &info_hpt370_33; |
| 1037 | if (clock_slot < 2 && ppi[0] == &info_hpt370a) | 1032 | if (clock_slot < 2 && ppi[0] == &info_hpt370a) |
| 1038 | ppi[0] = &info_hpt370a_33; | 1033 | ppi[0] = &info_hpt370a_33; |
| 1039 | printk(KERN_INFO "pata_hpt37x: %s using %dMHz bus clock.\n", | 1034 | |
| 1040 | chip_table->name, MHz[clock_slot]); | 1035 | pr_info(DRV_NAME ": %s using %dMHz bus clock.\n", |
| 1036 | chip_table->name, MHz[clock_slot]); | ||
| 1041 | } | 1037 | } |
| 1042 | 1038 | ||
| 1043 | /* Now kick off ATA set up */ | 1039 | /* Now kick off ATA set up */ |
diff --git a/drivers/ata/pata_hpt3x2n.c b/drivers/ata/pata_hpt3x2n.c index d2239bbdb79..eca68caf5f4 100644 --- a/drivers/ata/pata_hpt3x2n.c +++ b/drivers/ata/pata_hpt3x2n.c | |||
| @@ -25,7 +25,7 @@ | |||
| 25 | #include <linux/libata.h> | 25 | #include <linux/libata.h> |
| 26 | 26 | ||
| 27 | #define DRV_NAME "pata_hpt3x2n" | 27 | #define DRV_NAME "pata_hpt3x2n" |
| 28 | #define DRV_VERSION "0.3.13" | 28 | #define DRV_VERSION "0.3.14" |
| 29 | 29 | ||
| 30 | enum { | 30 | enum { |
| 31 | HPT_PCI_FAST = (1 << 31), | 31 | HPT_PCI_FAST = (1 << 31), |
| @@ -418,7 +418,7 @@ static int hpt3x2n_pci_clock(struct pci_dev *pdev) | |||
| 418 | u16 sr; | 418 | u16 sr; |
| 419 | u32 total = 0; | 419 | u32 total = 0; |
| 420 | 420 | ||
| 421 | printk(KERN_WARNING "pata_hpt3x2n: BIOS clock data not set.\n"); | 421 | pr_warning(DRV_NAME ": BIOS clock data not set.\n"); |
| 422 | 422 | ||
| 423 | /* This is the process the HPT371 BIOS is reported to use */ | 423 | /* This is the process the HPT371 BIOS is reported to use */ |
| 424 | for (i = 0; i < 128; i++) { | 424 | for (i = 0; i < 128; i++) { |
| @@ -528,8 +528,7 @@ hpt372n: | |||
| 528 | ppi[0] = &info_hpt372n; | 528 | ppi[0] = &info_hpt372n; |
| 529 | break; | 529 | break; |
| 530 | default: | 530 | default: |
| 531 | printk(KERN_ERR | 531 | pr_err(DRV_NAME ": PCI table is bogus, please report (%d).\n", |
| 532 | "pata_hpt3x2n: PCI table is bogus please report (%d).\n", | ||
| 533 | dev->device); | 532 | dev->device); |
| 534 | return -ENODEV; | 533 | return -ENODEV; |
| 535 | } | 534 | } |
| @@ -579,12 +578,11 @@ hpt372n: | |||
| 579 | pci_write_config_dword(dev, 0x5C, (f_high << 16) | f_low); | 578 | pci_write_config_dword(dev, 0x5C, (f_high << 16) | f_low); |
| 580 | } | 579 | } |
| 581 | if (adjust == 8) { | 580 | if (adjust == 8) { |
| 582 | printk(KERN_ERR "pata_hpt3x2n: DPLL did not stabilize!\n"); | 581 | pr_err(DRV_NAME ": DPLL did not stabilize!\n"); |
| 583 | return -ENODEV; | 582 | return -ENODEV; |
| 584 | } | 583 | } |
| 585 | 584 | ||
| 586 | printk(KERN_INFO "pata_hpt37x: bus clock %dMHz, using 66MHz DPLL.\n", | 585 | pr_info(DRV_NAME ": bus clock %dMHz, using 66MHz DPLL.\n", pci_mhz); |
| 587 | pci_mhz); | ||
| 588 | 586 | ||
| 589 | /* | 587 | /* |
| 590 | * Set our private data up. We only need a few flags | 588 | * Set our private data up. We only need a few flags |
diff --git a/drivers/ata/pata_mpc52xx.c b/drivers/ata/pata_mpc52xx.c index 8cc536e49a0..d7d8026cde9 100644 --- a/drivers/ata/pata_mpc52xx.c +++ b/drivers/ata/pata_mpc52xx.c | |||
| @@ -610,7 +610,7 @@ static struct scsi_host_template mpc52xx_ata_sht = { | |||
| 610 | }; | 610 | }; |
| 611 | 611 | ||
| 612 | static struct ata_port_operations mpc52xx_ata_port_ops = { | 612 | static struct ata_port_operations mpc52xx_ata_port_ops = { |
| 613 | .inherits = &ata_sff_port_ops, | 613 | .inherits = &ata_bmdma_port_ops, |
| 614 | .sff_dev_select = mpc52xx_ata_dev_select, | 614 | .sff_dev_select = mpc52xx_ata_dev_select, |
| 615 | .set_piomode = mpc52xx_ata_set_piomode, | 615 | .set_piomode = mpc52xx_ata_set_piomode, |
| 616 | .set_dmamode = mpc52xx_ata_set_dmamode, | 616 | .set_dmamode = mpc52xx_ata_set_dmamode, |
diff --git a/drivers/atm/idt77105.c b/drivers/atm/idt77105.c index bca9cb89a11..487a5473985 100644 --- a/drivers/atm/idt77105.c +++ b/drivers/atm/idt77105.c | |||
| @@ -151,7 +151,7 @@ static int fetch_stats(struct atm_dev *dev,struct idt77105_stats __user *arg,int | |||
| 151 | spin_unlock_irqrestore(&idt77105_priv_lock, flags); | 151 | spin_unlock_irqrestore(&idt77105_priv_lock, flags); |
| 152 | if (arg == NULL) | 152 | if (arg == NULL) |
| 153 | return 0; | 153 | return 0; |
| 154 | return copy_to_user(arg, &PRIV(dev)->stats, | 154 | return copy_to_user(arg, &stats, |
| 155 | sizeof(struct idt77105_stats)) ? -EFAULT : 0; | 155 | sizeof(struct idt77105_stats)) ? -EFAULT : 0; |
| 156 | } | 156 | } |
| 157 | 157 | ||
diff --git a/drivers/base/power/runtime.c b/drivers/base/power/runtime.c index 656493a5e07..42615b419df 100644 --- a/drivers/base/power/runtime.c +++ b/drivers/base/power/runtime.c | |||
| @@ -407,12 +407,15 @@ static int rpm_suspend(struct device *dev, int rpmflags) | |||
| 407 | goto out; | 407 | goto out; |
| 408 | } | 408 | } |
| 409 | 409 | ||
| 410 | /* Maybe the parent is now able to suspend. */ | ||
| 410 | if (parent && !parent->power.ignore_children && !dev->power.irq_safe) { | 411 | if (parent && !parent->power.ignore_children && !dev->power.irq_safe) { |
| 411 | spin_unlock_irq(&dev->power.lock); | 412 | spin_unlock(&dev->power.lock); |
| 412 | 413 | ||
| 413 | pm_request_idle(parent); | 414 | spin_lock(&parent->power.lock); |
| 415 | rpm_idle(parent, RPM_ASYNC); | ||
| 416 | spin_unlock(&parent->power.lock); | ||
| 414 | 417 | ||
| 415 | spin_lock_irq(&dev->power.lock); | 418 | spin_lock(&dev->power.lock); |
| 416 | } | 419 | } |
| 417 | 420 | ||
| 418 | out: | 421 | out: |
diff --git a/drivers/bluetooth/ath3k.c b/drivers/bluetooth/ath3k.c index 949ed09c636..a126e614601 100644 --- a/drivers/bluetooth/ath3k.c +++ b/drivers/bluetooth/ath3k.c | |||
| @@ -47,46 +47,40 @@ MODULE_DEVICE_TABLE(usb, ath3k_table); | |||
| 47 | #define USB_REQ_DFU_DNLOAD 1 | 47 | #define USB_REQ_DFU_DNLOAD 1 |
| 48 | #define BULK_SIZE 4096 | 48 | #define BULK_SIZE 4096 |
| 49 | 49 | ||
| 50 | struct ath3k_data { | 50 | static int ath3k_load_firmware(struct usb_device *udev, |
| 51 | struct usb_device *udev; | 51 | const struct firmware *firmware) |
| 52 | u8 *fw_data; | ||
| 53 | u32 fw_size; | ||
| 54 | u32 fw_sent; | ||
| 55 | }; | ||
| 56 | |||
| 57 | static int ath3k_load_firmware(struct ath3k_data *data, | ||
| 58 | unsigned char *firmware, | ||
| 59 | int count) | ||
| 60 | { | 52 | { |
| 61 | u8 *send_buf; | 53 | u8 *send_buf; |
| 62 | int err, pipe, len, size, sent = 0; | 54 | int err, pipe, len, size, sent = 0; |
| 55 | int count = firmware->size; | ||
| 63 | 56 | ||
| 64 | BT_DBG("ath3k %p udev %p", data, data->udev); | 57 | BT_DBG("udev %p", udev); |
| 65 | 58 | ||
| 66 | pipe = usb_sndctrlpipe(data->udev, 0); | 59 | pipe = usb_sndctrlpipe(udev, 0); |
| 67 | 60 | ||
| 68 | if ((usb_control_msg(data->udev, pipe, | 61 | send_buf = kmalloc(BULK_SIZE, GFP_ATOMIC); |
| 62 | if (!send_buf) { | ||
| 63 | BT_ERR("Can't allocate memory chunk for firmware"); | ||
| 64 | return -ENOMEM; | ||
| 65 | } | ||
| 66 | |||
| 67 | memcpy(send_buf, firmware->data, 20); | ||
| 68 | if ((err = usb_control_msg(udev, pipe, | ||
| 69 | USB_REQ_DFU_DNLOAD, | 69 | USB_REQ_DFU_DNLOAD, |
| 70 | USB_TYPE_VENDOR, 0, 0, | 70 | USB_TYPE_VENDOR, 0, 0, |
| 71 | firmware, 20, USB_CTRL_SET_TIMEOUT)) < 0) { | 71 | send_buf, 20, USB_CTRL_SET_TIMEOUT)) < 0) { |
| 72 | BT_ERR("Can't change to loading configuration err"); | 72 | BT_ERR("Can't change to loading configuration err"); |
| 73 | return -EBUSY; | 73 | goto error; |
| 74 | } | 74 | } |
| 75 | sent += 20; | 75 | sent += 20; |
| 76 | count -= 20; | 76 | count -= 20; |
| 77 | 77 | ||
| 78 | send_buf = kmalloc(BULK_SIZE, GFP_ATOMIC); | ||
| 79 | if (!send_buf) { | ||
| 80 | BT_ERR("Can't allocate memory chunk for firmware"); | ||
| 81 | return -ENOMEM; | ||
| 82 | } | ||
| 83 | |||
| 84 | while (count) { | 78 | while (count) { |
| 85 | size = min_t(uint, count, BULK_SIZE); | 79 | size = min_t(uint, count, BULK_SIZE); |
| 86 | pipe = usb_sndbulkpipe(data->udev, 0x02); | 80 | pipe = usb_sndbulkpipe(udev, 0x02); |
| 87 | memcpy(send_buf, firmware + sent, size); | 81 | memcpy(send_buf, firmware->data + sent, size); |
| 88 | 82 | ||
| 89 | err = usb_bulk_msg(data->udev, pipe, send_buf, size, | 83 | err = usb_bulk_msg(udev, pipe, send_buf, size, |
| 90 | &len, 3000); | 84 | &len, 3000); |
| 91 | 85 | ||
| 92 | if (err || (len != size)) { | 86 | if (err || (len != size)) { |
| @@ -112,57 +106,28 @@ static int ath3k_probe(struct usb_interface *intf, | |||
| 112 | { | 106 | { |
| 113 | const struct firmware *firmware; | 107 | const struct firmware *firmware; |
| 114 | struct usb_device *udev = interface_to_usbdev(intf); | 108 | struct usb_device *udev = interface_to_usbdev(intf); |
| 115 | struct ath3k_data *data; | ||
| 116 | int size; | ||
| 117 | 109 | ||
| 118 | BT_DBG("intf %p id %p", intf, id); | 110 | BT_DBG("intf %p id %p", intf, id); |
| 119 | 111 | ||
| 120 | if (intf->cur_altsetting->desc.bInterfaceNumber != 0) | 112 | if (intf->cur_altsetting->desc.bInterfaceNumber != 0) |
| 121 | return -ENODEV; | 113 | return -ENODEV; |
| 122 | 114 | ||
| 123 | data = kzalloc(sizeof(*data), GFP_KERNEL); | ||
| 124 | if (!data) | ||
| 125 | return -ENOMEM; | ||
| 126 | |||
| 127 | data->udev = udev; | ||
| 128 | |||
| 129 | if (request_firmware(&firmware, "ath3k-1.fw", &udev->dev) < 0) { | 115 | if (request_firmware(&firmware, "ath3k-1.fw", &udev->dev) < 0) { |
| 130 | kfree(data); | ||
| 131 | return -EIO; | 116 | return -EIO; |
| 132 | } | 117 | } |
| 133 | 118 | ||
| 134 | size = max_t(uint, firmware->size, 4096); | 119 | if (ath3k_load_firmware(udev, firmware)) { |
| 135 | data->fw_data = kmalloc(size, GFP_KERNEL); | ||
| 136 | if (!data->fw_data) { | ||
| 137 | release_firmware(firmware); | 120 | release_firmware(firmware); |
| 138 | kfree(data); | ||
| 139 | return -ENOMEM; | ||
| 140 | } | ||
| 141 | |||
| 142 | memcpy(data->fw_data, firmware->data, firmware->size); | ||
| 143 | data->fw_size = firmware->size; | ||
| 144 | data->fw_sent = 0; | ||
| 145 | release_firmware(firmware); | ||
| 146 | |||
| 147 | usb_set_intfdata(intf, data); | ||
| 148 | if (ath3k_load_firmware(data, data->fw_data, data->fw_size)) { | ||
| 149 | usb_set_intfdata(intf, NULL); | ||
| 150 | kfree(data->fw_data); | ||
| 151 | kfree(data); | ||
| 152 | return -EIO; | 121 | return -EIO; |
| 153 | } | 122 | } |
| 123 | release_firmware(firmware); | ||
| 154 | 124 | ||
| 155 | return 0; | 125 | return 0; |
| 156 | } | 126 | } |
| 157 | 127 | ||
| 158 | static void ath3k_disconnect(struct usb_interface *intf) | 128 | static void ath3k_disconnect(struct usb_interface *intf) |
| 159 | { | 129 | { |
| 160 | struct ath3k_data *data = usb_get_intfdata(intf); | ||
| 161 | |||
| 162 | BT_DBG("ath3k_disconnect intf %p", intf); | 130 | BT_DBG("ath3k_disconnect intf %p", intf); |
| 163 | |||
| 164 | kfree(data->fw_data); | ||
| 165 | kfree(data); | ||
| 166 | } | 131 | } |
| 167 | 132 | ||
| 168 | static struct usb_driver ath3k_driver = { | 133 | static struct usb_driver ath3k_driver = { |
diff --git a/drivers/char/agp/Kconfig b/drivers/char/agp/Kconfig index fcd867d923b..d8b1b576556 100644 --- a/drivers/char/agp/Kconfig +++ b/drivers/char/agp/Kconfig | |||
| @@ -50,7 +50,7 @@ config AGP_ATI | |||
| 50 | 50 | ||
| 51 | config AGP_AMD | 51 | config AGP_AMD |
| 52 | tristate "AMD Irongate, 761, and 762 chipset support" | 52 | tristate "AMD Irongate, 761, and 762 chipset support" |
| 53 | depends on AGP && (X86_32 || ALPHA) | 53 | depends on AGP && X86_32 |
| 54 | help | 54 | help |
| 55 | This option gives you AGP support for the GLX component of | 55 | This option gives you AGP support for the GLX component of |
| 56 | X on AMD Irongate, 761, and 762 chipsets. | 56 | X on AMD Irongate, 761, and 762 chipsets. |
diff --git a/drivers/char/agp/amd-k7-agp.c b/drivers/char/agp/amd-k7-agp.c index b1b4362bc64..45681c0ff3b 100644 --- a/drivers/char/agp/amd-k7-agp.c +++ b/drivers/char/agp/amd-k7-agp.c | |||
| @@ -41,22 +41,8 @@ static int amd_create_page_map(struct amd_page_map *page_map) | |||
| 41 | if (page_map->real == NULL) | 41 | if (page_map->real == NULL) |
| 42 | return -ENOMEM; | 42 | return -ENOMEM; |
| 43 | 43 | ||
| 44 | #ifndef CONFIG_X86 | ||
| 45 | SetPageReserved(virt_to_page(page_map->real)); | ||
| 46 | global_cache_flush(); | ||
| 47 | page_map->remapped = ioremap_nocache(virt_to_phys(page_map->real), | ||
| 48 | PAGE_SIZE); | ||
| 49 | if (page_map->remapped == NULL) { | ||
| 50 | ClearPageReserved(virt_to_page(page_map->real)); | ||
| 51 | free_page((unsigned long) page_map->real); | ||
| 52 | page_map->real = NULL; | ||
| 53 | return -ENOMEM; | ||
| 54 | } | ||
| 55 | global_cache_flush(); | ||
| 56 | #else | ||
| 57 | set_memory_uc((unsigned long)page_map->real, 1); | 44 | set_memory_uc((unsigned long)page_map->real, 1); |
| 58 | page_map->remapped = page_map->real; | 45 | page_map->remapped = page_map->real; |
| 59 | #endif | ||
| 60 | 46 | ||
| 61 | for (i = 0; i < PAGE_SIZE / sizeof(unsigned long); i++) { | 47 | for (i = 0; i < PAGE_SIZE / sizeof(unsigned long); i++) { |
| 62 | writel(agp_bridge->scratch_page, page_map->remapped+i); | 48 | writel(agp_bridge->scratch_page, page_map->remapped+i); |
| @@ -68,12 +54,7 @@ static int amd_create_page_map(struct amd_page_map *page_map) | |||
| 68 | 54 | ||
| 69 | static void amd_free_page_map(struct amd_page_map *page_map) | 55 | static void amd_free_page_map(struct amd_page_map *page_map) |
| 70 | { | 56 | { |
| 71 | #ifndef CONFIG_X86 | ||
| 72 | iounmap(page_map->remapped); | ||
| 73 | ClearPageReserved(virt_to_page(page_map->real)); | ||
| 74 | #else | ||
| 75 | set_memory_wb((unsigned long)page_map->real, 1); | 57 | set_memory_wb((unsigned long)page_map->real, 1); |
| 76 | #endif | ||
| 77 | free_page((unsigned long) page_map->real); | 58 | free_page((unsigned long) page_map->real); |
| 78 | } | 59 | } |
| 79 | 60 | ||
diff --git a/drivers/char/agp/intel-agp.c b/drivers/char/agp/intel-agp.c index 857df10c042..b0a0dccc98c 100644 --- a/drivers/char/agp/intel-agp.c +++ b/drivers/char/agp/intel-agp.c | |||
| @@ -774,20 +774,14 @@ static int __devinit agp_intel_probe(struct pci_dev *pdev, | |||
| 774 | dev_info(&pdev->dev, "Intel %s Chipset\n", intel_agp_chipsets[i].name); | 774 | dev_info(&pdev->dev, "Intel %s Chipset\n", intel_agp_chipsets[i].name); |
| 775 | 775 | ||
| 776 | /* | 776 | /* |
| 777 | * If the device has not been properly setup, the following will catch | ||
| 778 | * the problem and should stop the system from crashing. | ||
| 779 | * 20030610 - hamish@zot.org | ||
| 780 | */ | ||
| 781 | if (pci_enable_device(pdev)) { | ||
| 782 | dev_err(&pdev->dev, "can't enable PCI device\n"); | ||
| 783 | agp_put_bridge(bridge); | ||
| 784 | return -ENODEV; | ||
| 785 | } | ||
| 786 | |||
| 787 | /* | ||
| 788 | * The following fixes the case where the BIOS has "forgotten" to | 777 | * The following fixes the case where the BIOS has "forgotten" to |
| 789 | * provide an address range for the GART. | 778 | * provide an address range for the GART. |
| 790 | * 20030610 - hamish@zot.org | 779 | * 20030610 - hamish@zot.org |
| 780 | * This happens before pci_enable_device() intentionally; | ||
| 781 | * calling pci_enable_device() before assigning the resource | ||
| 782 | * will result in the GART being disabled on machines with such | ||
| 783 | * BIOSs (the GART ends up with a BAR starting at 0, which | ||
| 784 | * conflicts a lot of other devices). | ||
| 791 | */ | 785 | */ |
| 792 | r = &pdev->resource[0]; | 786 | r = &pdev->resource[0]; |
| 793 | if (!r->start && r->end) { | 787 | if (!r->start && r->end) { |
| @@ -798,6 +792,17 @@ static int __devinit agp_intel_probe(struct pci_dev *pdev, | |||
| 798 | } | 792 | } |
| 799 | } | 793 | } |
| 800 | 794 | ||
| 795 | /* | ||
| 796 | * If the device has not been properly setup, the following will catch | ||
| 797 | * the problem and should stop the system from crashing. | ||
| 798 | * 20030610 - hamish@zot.org | ||
| 799 | */ | ||
| 800 | if (pci_enable_device(pdev)) { | ||
| 801 | dev_err(&pdev->dev, "can't enable PCI device\n"); | ||
| 802 | agp_put_bridge(bridge); | ||
| 803 | return -ENODEV; | ||
| 804 | } | ||
| 805 | |||
| 801 | /* Fill in the mode register */ | 806 | /* Fill in the mode register */ |
| 802 | if (cap_ptr) { | 807 | if (cap_ptr) { |
| 803 | pci_read_config_dword(pdev, | 808 | pci_read_config_dword(pdev, |
diff --git a/drivers/char/agp/intel-gtt.c b/drivers/char/agp/intel-gtt.c index 826ab0939a1..fab3d3265ad 100644 --- a/drivers/char/agp/intel-gtt.c +++ b/drivers/char/agp/intel-gtt.c | |||
| @@ -68,6 +68,7 @@ static struct _intel_private { | |||
| 68 | phys_addr_t gma_bus_addr; | 68 | phys_addr_t gma_bus_addr; |
| 69 | u32 PGETBL_save; | 69 | u32 PGETBL_save; |
| 70 | u32 __iomem *gtt; /* I915G */ | 70 | u32 __iomem *gtt; /* I915G */ |
| 71 | bool clear_fake_agp; /* on first access via agp, fill with scratch */ | ||
| 71 | int num_dcache_entries; | 72 | int num_dcache_entries; |
| 72 | union { | 73 | union { |
| 73 | void __iomem *i9xx_flush_page; | 74 | void __iomem *i9xx_flush_page; |
| @@ -869,21 +870,12 @@ static int intel_fake_agp_free_gatt_table(struct agp_bridge_data *bridge) | |||
| 869 | 870 | ||
| 870 | static int intel_fake_agp_configure(void) | 871 | static int intel_fake_agp_configure(void) |
| 871 | { | 872 | { |
| 872 | int i; | ||
| 873 | |||
| 874 | if (!intel_enable_gtt()) | 873 | if (!intel_enable_gtt()) |
| 875 | return -EIO; | 874 | return -EIO; |
| 876 | 875 | ||
| 876 | intel_private.clear_fake_agp = true; | ||
| 877 | agp_bridge->gart_bus_addr = intel_private.gma_bus_addr; | 877 | agp_bridge->gart_bus_addr = intel_private.gma_bus_addr; |
| 878 | 878 | ||
| 879 | for (i = 0; i < intel_private.base.gtt_total_entries; i++) { | ||
| 880 | intel_private.driver->write_entry(intel_private.scratch_page_dma, | ||
| 881 | i, 0); | ||
| 882 | } | ||
| 883 | readl(intel_private.gtt+i-1); /* PCI Posting. */ | ||
| 884 | |||
| 885 | global_cache_flush(); | ||
| 886 | |||
| 887 | return 0; | 879 | return 0; |
| 888 | } | 880 | } |
| 889 | 881 | ||
| @@ -945,6 +937,13 @@ static int intel_fake_agp_insert_entries(struct agp_memory *mem, | |||
| 945 | { | 937 | { |
| 946 | int ret = -EINVAL; | 938 | int ret = -EINVAL; |
| 947 | 939 | ||
| 940 | if (intel_private.clear_fake_agp) { | ||
| 941 | int start = intel_private.base.stolen_size / PAGE_SIZE; | ||
| 942 | int end = intel_private.base.gtt_mappable_entries; | ||
| 943 | intel_gtt_clear_range(start, end - start); | ||
| 944 | intel_private.clear_fake_agp = false; | ||
| 945 | } | ||
| 946 | |||
| 948 | if (INTEL_GTT_GEN == 1 && type == AGP_DCACHE_MEMORY) | 947 | if (INTEL_GTT_GEN == 1 && type == AGP_DCACHE_MEMORY) |
| 949 | return i810_insert_dcache_entries(mem, pg_start, type); | 948 | return i810_insert_dcache_entries(mem, pg_start, type); |
| 950 | 949 | ||
diff --git a/drivers/char/bfin_jtag_comm.c b/drivers/char/bfin_jtag_comm.c index e397df3ad98..16402445f2b 100644 --- a/drivers/char/bfin_jtag_comm.c +++ b/drivers/char/bfin_jtag_comm.c | |||
| @@ -183,16 +183,16 @@ bfin_jc_circ_write(const unsigned char *buf, int count) | |||
| 183 | } | 183 | } |
| 184 | 184 | ||
| 185 | #ifndef CONFIG_BFIN_JTAG_COMM_CONSOLE | 185 | #ifndef CONFIG_BFIN_JTAG_COMM_CONSOLE |
| 186 | # define acquire_console_sem() | 186 | # define console_lock() |
| 187 | # define release_console_sem() | 187 | # define console_unlock() |
| 188 | #endif | 188 | #endif |
| 189 | static int | 189 | static int |
| 190 | bfin_jc_write(struct tty_struct *tty, const unsigned char *buf, int count) | 190 | bfin_jc_write(struct tty_struct *tty, const unsigned char *buf, int count) |
| 191 | { | 191 | { |
| 192 | int i; | 192 | int i; |
| 193 | acquire_console_sem(); | 193 | console_lock(); |
| 194 | i = bfin_jc_circ_write(buf, count); | 194 | i = bfin_jc_circ_write(buf, count); |
| 195 | release_console_sem(); | 195 | console_unlock(); |
| 196 | wake_up_process(bfin_jc_kthread); | 196 | wake_up_process(bfin_jc_kthread); |
| 197 | return i; | 197 | return i; |
| 198 | } | 198 | } |
diff --git a/drivers/char/tpm/tpm.c b/drivers/char/tpm/tpm.c index 1f46f1cd922..36e0fa161c2 100644 --- a/drivers/char/tpm/tpm.c +++ b/drivers/char/tpm/tpm.c | |||
| @@ -364,12 +364,14 @@ unsigned long tpm_calc_ordinal_duration(struct tpm_chip *chip, | |||
| 364 | tpm_protected_ordinal_duration[ordinal & | 364 | tpm_protected_ordinal_duration[ordinal & |
| 365 | TPM_PROTECTED_ORDINAL_MASK]; | 365 | TPM_PROTECTED_ORDINAL_MASK]; |
| 366 | 366 | ||
| 367 | if (duration_idx != TPM_UNDEFINED) | 367 | if (duration_idx != TPM_UNDEFINED) { |
| 368 | duration = chip->vendor.duration[duration_idx]; | 368 | duration = chip->vendor.duration[duration_idx]; |
| 369 | if (duration <= 0) | 369 | /* if duration is 0, it's because chip->vendor.duration wasn't */ |
| 370 | /* filled yet, so we set the lowest timeout just to give enough */ | ||
| 371 | /* time for tpm_get_timeouts() to succeed */ | ||
| 372 | return (duration <= 0 ? HZ : duration); | ||
| 373 | } else | ||
| 370 | return 2 * 60 * HZ; | 374 | return 2 * 60 * HZ; |
| 371 | else | ||
| 372 | return duration; | ||
| 373 | } | 375 | } |
| 374 | EXPORT_SYMBOL_GPL(tpm_calc_ordinal_duration); | 376 | EXPORT_SYMBOL_GPL(tpm_calc_ordinal_duration); |
| 375 | 377 | ||
diff --git a/drivers/char/tpm/tpm_tis.c b/drivers/char/tpm/tpm_tis.c index c17a305ecb2..dd21df55689 100644 --- a/drivers/char/tpm/tpm_tis.c +++ b/drivers/char/tpm/tpm_tis.c | |||
| @@ -493,9 +493,6 @@ static int tpm_tis_init(struct device *dev, resource_size_t start, | |||
| 493 | "1.2 TPM (device-id 0x%X, rev-id %d)\n", | 493 | "1.2 TPM (device-id 0x%X, rev-id %d)\n", |
| 494 | vendor >> 16, ioread8(chip->vendor.iobase + TPM_RID(0))); | 494 | vendor >> 16, ioread8(chip->vendor.iobase + TPM_RID(0))); |
| 495 | 495 | ||
| 496 | if (is_itpm(to_pnp_dev(dev))) | ||
| 497 | itpm = 1; | ||
| 498 | |||
| 499 | if (itpm) | 496 | if (itpm) |
| 500 | dev_info(dev, "Intel iTPM workaround enabled\n"); | 497 | dev_info(dev, "Intel iTPM workaround enabled\n"); |
| 501 | 498 | ||
| @@ -637,6 +634,9 @@ static int __devinit tpm_tis_pnp_init(struct pnp_dev *pnp_dev, | |||
| 637 | else | 634 | else |
| 638 | interrupts = 0; | 635 | interrupts = 0; |
| 639 | 636 | ||
| 637 | if (is_itpm(pnp_dev)) | ||
| 638 | itpm = 1; | ||
| 639 | |||
| 640 | return tpm_tis_init(&pnp_dev->dev, start, len, irq); | 640 | return tpm_tis_init(&pnp_dev->dev, start, len, irq); |
| 641 | } | 641 | } |
| 642 | 642 | ||
diff --git a/drivers/clocksource/acpi_pm.c b/drivers/clocksource/acpi_pm.c index cfb0f527841..effe7974aa9 100644 --- a/drivers/clocksource/acpi_pm.c +++ b/drivers/clocksource/acpi_pm.c | |||
| @@ -202,17 +202,21 @@ static int __init init_acpi_pm_clocksource(void) | |||
| 202 | printk(KERN_INFO "PM-Timer had inconsistent results:" | 202 | printk(KERN_INFO "PM-Timer had inconsistent results:" |
| 203 | " 0x%#llx, 0x%#llx - aborting.\n", | 203 | " 0x%#llx, 0x%#llx - aborting.\n", |
| 204 | value1, value2); | 204 | value1, value2); |
| 205 | pmtmr_ioport = 0; | ||
| 205 | return -EINVAL; | 206 | return -EINVAL; |
| 206 | } | 207 | } |
| 207 | if (i == ACPI_PM_READ_CHECKS) { | 208 | if (i == ACPI_PM_READ_CHECKS) { |
| 208 | printk(KERN_INFO "PM-Timer failed consistency check " | 209 | printk(KERN_INFO "PM-Timer failed consistency check " |
| 209 | " (0x%#llx) - aborting.\n", value1); | 210 | " (0x%#llx) - aborting.\n", value1); |
| 211 | pmtmr_ioport = 0; | ||
| 210 | return -ENODEV; | 212 | return -ENODEV; |
| 211 | } | 213 | } |
| 212 | } | 214 | } |
| 213 | 215 | ||
| 214 | if (verify_pmtmr_rate() != 0) | 216 | if (verify_pmtmr_rate() != 0){ |
| 217 | pmtmr_ioport = 0; | ||
| 215 | return -ENODEV; | 218 | return -ENODEV; |
| 219 | } | ||
| 216 | 220 | ||
| 217 | return clocksource_register_hz(&clocksource_acpi_pm, | 221 | return clocksource_register_hz(&clocksource_acpi_pm, |
| 218 | PMTMR_TICKS_PER_SEC); | 222 | PMTMR_TICKS_PER_SEC); |
diff --git a/drivers/clocksource/tcb_clksrc.c b/drivers/clocksource/tcb_clksrc.c index 01b886e6882..79c47e88d5d 100644 --- a/drivers/clocksource/tcb_clksrc.c +++ b/drivers/clocksource/tcb_clksrc.c | |||
| @@ -196,9 +196,9 @@ static void __init setup_clkevents(struct atmel_tc *tc, int clk32k_divisor_idx) | |||
| 196 | clkevt.clkevt.min_delta_ns = clockevent_delta2ns(1, &clkevt.clkevt) + 1; | 196 | clkevt.clkevt.min_delta_ns = clockevent_delta2ns(1, &clkevt.clkevt) + 1; |
| 197 | clkevt.clkevt.cpumask = cpumask_of(0); | 197 | clkevt.clkevt.cpumask = cpumask_of(0); |
| 198 | 198 | ||
| 199 | setup_irq(irq, &tc_irqaction); | ||
| 200 | |||
| 201 | clockevents_register_device(&clkevt.clkevt); | 199 | clockevents_register_device(&clkevt.clkevt); |
| 200 | |||
| 201 | setup_irq(irq, &tc_irqaction); | ||
| 202 | } | 202 | } |
| 203 | 203 | ||
| 204 | #else /* !CONFIG_GENERIC_CLOCKEVENTS */ | 204 | #else /* !CONFIG_GENERIC_CLOCKEVENTS */ |
diff --git a/drivers/gpio/langwell_gpio.c b/drivers/gpio/langwell_gpio.c index d81cc748e77..54d70a47afc 100644 --- a/drivers/gpio/langwell_gpio.c +++ b/drivers/gpio/langwell_gpio.c | |||
| @@ -187,7 +187,7 @@ MODULE_DEVICE_TABLE(pci, lnw_gpio_ids); | |||
| 187 | 187 | ||
| 188 | static void lnw_irq_handler(unsigned irq, struct irq_desc *desc) | 188 | static void lnw_irq_handler(unsigned irq, struct irq_desc *desc) |
| 189 | { | 189 | { |
| 190 | struct lnw_gpio *lnw = (struct lnw_gpio *)get_irq_data(irq); | 190 | struct lnw_gpio *lnw = get_irq_data(irq); |
| 191 | u32 base, gpio; | 191 | u32 base, gpio; |
| 192 | void __iomem *gedr; | 192 | void __iomem *gedr; |
| 193 | u32 gedr_v; | 193 | u32 gedr_v; |
| @@ -206,7 +206,12 @@ static void lnw_irq_handler(unsigned irq, struct irq_desc *desc) | |||
| 206 | /* clear the edge detect status bit */ | 206 | /* clear the edge detect status bit */ |
| 207 | writel(gedr_v, gedr); | 207 | writel(gedr_v, gedr); |
| 208 | } | 208 | } |
| 209 | desc->chip->eoi(irq); | 209 | |
| 210 | if (desc->chip->irq_eoi) | ||
| 211 | desc->chip->irq_eoi(irq_get_irq_data(irq)); | ||
| 212 | else | ||
| 213 | dev_warn(lnw->chip.dev, "missing EOI handler for irq %d\n", irq); | ||
| 214 | |||
| 210 | } | 215 | } |
| 211 | 216 | ||
| 212 | static int __devinit lnw_gpio_probe(struct pci_dev *pdev, | 217 | static int __devinit lnw_gpio_probe(struct pci_dev *pdev, |
diff --git a/drivers/gpu/drm/Kconfig b/drivers/gpu/drm/Kconfig index bea966f8ac8..0902d446003 100644 --- a/drivers/gpu/drm/Kconfig +++ b/drivers/gpu/drm/Kconfig | |||
| @@ -100,7 +100,10 @@ config DRM_I830 | |||
| 100 | config DRM_I915 | 100 | config DRM_I915 |
| 101 | tristate "i915 driver" | 101 | tristate "i915 driver" |
| 102 | depends on AGP_INTEL | 102 | depends on AGP_INTEL |
| 103 | # we need shmfs for the swappable backing store, and in particular | ||
| 104 | # the shmem_readpage() which depends upon tmpfs | ||
| 103 | select SHMEM | 105 | select SHMEM |
| 106 | select TMPFS | ||
| 104 | select DRM_KMS_HELPER | 107 | select DRM_KMS_HELPER |
| 105 | select FB_CFB_FILLRECT | 108 | select FB_CFB_FILLRECT |
| 106 | select FB_CFB_COPYAREA | 109 | select FB_CFB_COPYAREA |
diff --git a/drivers/gpu/drm/drm_crtc.c b/drivers/gpu/drm/drm_crtc.c index 2baa6708e44..654faa803dc 100644 --- a/drivers/gpu/drm/drm_crtc.c +++ b/drivers/gpu/drm/drm_crtc.c | |||
| @@ -2674,3 +2674,23 @@ out: | |||
| 2674 | mutex_unlock(&dev->mode_config.mutex); | 2674 | mutex_unlock(&dev->mode_config.mutex); |
| 2675 | return ret; | 2675 | return ret; |
| 2676 | } | 2676 | } |
| 2677 | |||
| 2678 | void drm_mode_config_reset(struct drm_device *dev) | ||
| 2679 | { | ||
| 2680 | struct drm_crtc *crtc; | ||
| 2681 | struct drm_encoder *encoder; | ||
| 2682 | struct drm_connector *connector; | ||
| 2683 | |||
| 2684 | list_for_each_entry(crtc, &dev->mode_config.crtc_list, head) | ||
| 2685 | if (crtc->funcs->reset) | ||
| 2686 | crtc->funcs->reset(crtc); | ||
| 2687 | |||
| 2688 | list_for_each_entry(encoder, &dev->mode_config.encoder_list, head) | ||
| 2689 | if (encoder->funcs->reset) | ||
| 2690 | encoder->funcs->reset(encoder); | ||
| 2691 | |||
| 2692 | list_for_each_entry(connector, &dev->mode_config.connector_list, head) | ||
| 2693 | if (connector->funcs->reset) | ||
| 2694 | connector->funcs->reset(connector); | ||
| 2695 | } | ||
| 2696 | EXPORT_SYMBOL(drm_mode_config_reset); | ||
diff --git a/drivers/gpu/drm/drm_crtc_helper.c b/drivers/gpu/drm/drm_crtc_helper.c index 952b3d4fb2a..92369655dca 100644 --- a/drivers/gpu/drm/drm_crtc_helper.c +++ b/drivers/gpu/drm/drm_crtc_helper.c | |||
| @@ -343,13 +343,12 @@ bool drm_crtc_helper_set_mode(struct drm_crtc *crtc, | |||
| 343 | struct drm_encoder *encoder; | 343 | struct drm_encoder *encoder; |
| 344 | bool ret = true; | 344 | bool ret = true; |
| 345 | 345 | ||
| 346 | adjusted_mode = drm_mode_duplicate(dev, mode); | ||
| 347 | |||
| 348 | crtc->enabled = drm_helper_crtc_in_use(crtc); | 346 | crtc->enabled = drm_helper_crtc_in_use(crtc); |
| 349 | |||
| 350 | if (!crtc->enabled) | 347 | if (!crtc->enabled) |
| 351 | return true; | 348 | return true; |
| 352 | 349 | ||
| 350 | adjusted_mode = drm_mode_duplicate(dev, mode); | ||
| 351 | |||
| 353 | saved_hwmode = crtc->hwmode; | 352 | saved_hwmode = crtc->hwmode; |
| 354 | saved_mode = crtc->mode; | 353 | saved_mode = crtc->mode; |
| 355 | saved_x = crtc->x; | 354 | saved_x = crtc->x; |
| @@ -437,10 +436,9 @@ bool drm_crtc_helper_set_mode(struct drm_crtc *crtc, | |||
| 437 | */ | 436 | */ |
| 438 | drm_calc_timestamping_constants(crtc); | 437 | drm_calc_timestamping_constants(crtc); |
| 439 | 438 | ||
| 440 | /* XXX free adjustedmode */ | ||
| 441 | drm_mode_destroy(dev, adjusted_mode); | ||
| 442 | /* FIXME: add subpixel order */ | 439 | /* FIXME: add subpixel order */ |
| 443 | done: | 440 | done: |
| 441 | drm_mode_destroy(dev, adjusted_mode); | ||
| 444 | if (!ret) { | 442 | if (!ret) { |
| 445 | crtc->hwmode = saved_hwmode; | 443 | crtc->hwmode = saved_hwmode; |
| 446 | crtc->mode = saved_mode; | 444 | crtc->mode = saved_mode; |
| @@ -497,14 +495,17 @@ int drm_crtc_helper_set_config(struct drm_mode_set *set) | |||
| 497 | 495 | ||
| 498 | crtc_funcs = set->crtc->helper_private; | 496 | crtc_funcs = set->crtc->helper_private; |
| 499 | 497 | ||
| 498 | if (!set->mode) | ||
| 499 | set->fb = NULL; | ||
| 500 | |||
| 500 | if (set->fb) { | 501 | if (set->fb) { |
| 501 | DRM_DEBUG_KMS("[CRTC:%d] [FB:%d] #connectors=%d (x y) (%i %i)\n", | 502 | DRM_DEBUG_KMS("[CRTC:%d] [FB:%d] #connectors=%d (x y) (%i %i)\n", |
| 502 | set->crtc->base.id, set->fb->base.id, | 503 | set->crtc->base.id, set->fb->base.id, |
| 503 | (int)set->num_connectors, set->x, set->y); | 504 | (int)set->num_connectors, set->x, set->y); |
| 504 | } else { | 505 | } else { |
| 505 | DRM_DEBUG_KMS("[CRTC:%d] [NOFB] #connectors=%d (x y) (%i %i)\n", | 506 | DRM_DEBUG_KMS("[CRTC:%d] [NOFB]\n", set->crtc->base.id); |
| 506 | set->crtc->base.id, (int)set->num_connectors, | 507 | set->mode = NULL; |
| 507 | set->x, set->y); | 508 | set->num_connectors = 0; |
| 508 | } | 509 | } |
| 509 | 510 | ||
| 510 | dev = set->crtc->dev; | 511 | dev = set->crtc->dev; |
| @@ -649,8 +650,8 @@ int drm_crtc_helper_set_config(struct drm_mode_set *set) | |||
| 649 | mode_changed = true; | 650 | mode_changed = true; |
| 650 | 651 | ||
| 651 | if (mode_changed) { | 652 | if (mode_changed) { |
| 652 | set->crtc->enabled = (set->mode != NULL); | 653 | set->crtc->enabled = drm_helper_crtc_in_use(set->crtc); |
| 653 | if (set->mode != NULL) { | 654 | if (set->crtc->enabled) { |
| 654 | DRM_DEBUG_KMS("attempting to set mode from" | 655 | DRM_DEBUG_KMS("attempting to set mode from" |
| 655 | " userspace\n"); | 656 | " userspace\n"); |
| 656 | drm_mode_debug_printmodeline(set->mode); | 657 | drm_mode_debug_printmodeline(set->mode); |
| @@ -665,6 +666,12 @@ int drm_crtc_helper_set_config(struct drm_mode_set *set) | |||
| 665 | ret = -EINVAL; | 666 | ret = -EINVAL; |
| 666 | goto fail; | 667 | goto fail; |
| 667 | } | 668 | } |
| 669 | DRM_DEBUG_KMS("Setting connector DPMS state to on\n"); | ||
| 670 | for (i = 0; i < set->num_connectors; i++) { | ||
| 671 | DRM_DEBUG_KMS("\t[CONNECTOR:%d:%s] set DPMS on\n", set->connectors[i]->base.id, | ||
| 672 | drm_get_connector_name(set->connectors[i])); | ||
| 673 | set->connectors[i]->dpms = DRM_MODE_DPMS_ON; | ||
| 674 | } | ||
| 668 | } | 675 | } |
| 669 | drm_helper_disable_unused_functions(dev); | 676 | drm_helper_disable_unused_functions(dev); |
| 670 | } else if (fb_changed) { | 677 | } else if (fb_changed) { |
| @@ -681,12 +688,6 @@ int drm_crtc_helper_set_config(struct drm_mode_set *set) | |||
| 681 | goto fail; | 688 | goto fail; |
| 682 | } | 689 | } |
| 683 | } | 690 | } |
| 684 | DRM_DEBUG_KMS("Setting connector DPMS state to on\n"); | ||
| 685 | for (i = 0; i < set->num_connectors; i++) { | ||
| 686 | DRM_DEBUG_KMS("\t[CONNECTOR:%d:%s] set DPMS on\n", set->connectors[i]->base.id, | ||
| 687 | drm_get_connector_name(set->connectors[i])); | ||
| 688 | set->connectors[i]->dpms = DRM_MODE_DPMS_ON; | ||
| 689 | } | ||
| 690 | 691 | ||
| 691 | kfree(save_connectors); | 692 | kfree(save_connectors); |
| 692 | kfree(save_encoders); | 693 | kfree(save_encoders); |
diff --git a/drivers/gpu/drm/drm_irq.c b/drivers/gpu/drm/drm_irq.c index 0054e957203..3dadfa2a852 100644 --- a/drivers/gpu/drm/drm_irq.c +++ b/drivers/gpu/drm/drm_irq.c | |||
| @@ -1250,7 +1250,7 @@ void drm_handle_vblank_events(struct drm_device *dev, int crtc) | |||
| 1250 | * Drivers should call this routine in their vblank interrupt handlers to | 1250 | * Drivers should call this routine in their vblank interrupt handlers to |
| 1251 | * update the vblank counter and send any signals that may be pending. | 1251 | * update the vblank counter and send any signals that may be pending. |
| 1252 | */ | 1252 | */ |
| 1253 | void drm_handle_vblank(struct drm_device *dev, int crtc) | 1253 | bool drm_handle_vblank(struct drm_device *dev, int crtc) |
| 1254 | { | 1254 | { |
| 1255 | u32 vblcount; | 1255 | u32 vblcount; |
| 1256 | s64 diff_ns; | 1256 | s64 diff_ns; |
| @@ -1258,7 +1258,7 @@ void drm_handle_vblank(struct drm_device *dev, int crtc) | |||
| 1258 | unsigned long irqflags; | 1258 | unsigned long irqflags; |
| 1259 | 1259 | ||
| 1260 | if (!dev->num_crtcs) | 1260 | if (!dev->num_crtcs) |
| 1261 | return; | 1261 | return false; |
| 1262 | 1262 | ||
| 1263 | /* Need timestamp lock to prevent concurrent execution with | 1263 | /* Need timestamp lock to prevent concurrent execution with |
| 1264 | * vblank enable/disable, as this would cause inconsistent | 1264 | * vblank enable/disable, as this would cause inconsistent |
| @@ -1269,7 +1269,7 @@ void drm_handle_vblank(struct drm_device *dev, int crtc) | |||
| 1269 | /* Vblank irq handling disabled. Nothing to do. */ | 1269 | /* Vblank irq handling disabled. Nothing to do. */ |
| 1270 | if (!dev->vblank_enabled[crtc]) { | 1270 | if (!dev->vblank_enabled[crtc]) { |
| 1271 | spin_unlock_irqrestore(&dev->vblank_time_lock, irqflags); | 1271 | spin_unlock_irqrestore(&dev->vblank_time_lock, irqflags); |
| 1272 | return; | 1272 | return false; |
| 1273 | } | 1273 | } |
| 1274 | 1274 | ||
| 1275 | /* Fetch corresponding timestamp for this vblank interval from | 1275 | /* Fetch corresponding timestamp for this vblank interval from |
| @@ -1311,5 +1311,6 @@ void drm_handle_vblank(struct drm_device *dev, int crtc) | |||
| 1311 | drm_handle_vblank_events(dev, crtc); | 1311 | drm_handle_vblank_events(dev, crtc); |
| 1312 | 1312 | ||
| 1313 | spin_unlock_irqrestore(&dev->vblank_time_lock, irqflags); | 1313 | spin_unlock_irqrestore(&dev->vblank_time_lock, irqflags); |
| 1314 | return true; | ||
| 1314 | } | 1315 | } |
| 1315 | EXPORT_SYMBOL(drm_handle_vblank); | 1316 | EXPORT_SYMBOL(drm_handle_vblank); |
diff --git a/drivers/gpu/drm/i915/i915_dma.c b/drivers/gpu/drm/i915/i915_dma.c index 844f3c972b0..17bd766f208 100644 --- a/drivers/gpu/drm/i915/i915_dma.c +++ b/drivers/gpu/drm/i915/i915_dma.c | |||
| @@ -152,7 +152,7 @@ static int i915_initialize(struct drm_device * dev, drm_i915_init_t * init) | |||
| 152 | { | 152 | { |
| 153 | drm_i915_private_t *dev_priv = dev->dev_private; | 153 | drm_i915_private_t *dev_priv = dev->dev_private; |
| 154 | struct drm_i915_master_private *master_priv = dev->primary->master->driver_priv; | 154 | struct drm_i915_master_private *master_priv = dev->primary->master->driver_priv; |
| 155 | struct intel_ring_buffer *ring = LP_RING(dev_priv); | 155 | int ret; |
| 156 | 156 | ||
| 157 | master_priv->sarea = drm_getsarea(dev); | 157 | master_priv->sarea = drm_getsarea(dev); |
| 158 | if (master_priv->sarea) { | 158 | if (master_priv->sarea) { |
| @@ -163,33 +163,22 @@ static int i915_initialize(struct drm_device * dev, drm_i915_init_t * init) | |||
| 163 | } | 163 | } |
| 164 | 164 | ||
| 165 | if (init->ring_size != 0) { | 165 | if (init->ring_size != 0) { |
| 166 | if (ring->obj != NULL) { | 166 | if (LP_RING(dev_priv)->obj != NULL) { |
| 167 | i915_dma_cleanup(dev); | 167 | i915_dma_cleanup(dev); |
| 168 | DRM_ERROR("Client tried to initialize ringbuffer in " | 168 | DRM_ERROR("Client tried to initialize ringbuffer in " |
| 169 | "GEM mode\n"); | 169 | "GEM mode\n"); |
| 170 | return -EINVAL; | 170 | return -EINVAL; |
| 171 | } | 171 | } |
| 172 | 172 | ||
| 173 | ring->size = init->ring_size; | 173 | ret = intel_render_ring_init_dri(dev, |
| 174 | 174 | init->ring_start, | |
| 175 | ring->map.offset = init->ring_start; | 175 | init->ring_size); |
| 176 | ring->map.size = init->ring_size; | 176 | if (ret) { |
| 177 | ring->map.type = 0; | ||
| 178 | ring->map.flags = 0; | ||
| 179 | ring->map.mtrr = 0; | ||
| 180 | |||
| 181 | drm_core_ioremap_wc(&ring->map, dev); | ||
| 182 | |||
| 183 | if (ring->map.handle == NULL) { | ||
| 184 | i915_dma_cleanup(dev); | 177 | i915_dma_cleanup(dev); |
| 185 | DRM_ERROR("can not ioremap virtual address for" | 178 | return ret; |
| 186 | " ring buffer\n"); | ||
| 187 | return -ENOMEM; | ||
| 188 | } | 179 | } |
| 189 | } | 180 | } |
| 190 | 181 | ||
| 191 | ring->virtual_start = ring->map.handle; | ||
| 192 | |||
| 193 | dev_priv->cpp = init->cpp; | 182 | dev_priv->cpp = init->cpp; |
| 194 | dev_priv->back_offset = init->back_offset; | 183 | dev_priv->back_offset = init->back_offset; |
| 195 | dev_priv->front_offset = init->front_offset; | 184 | dev_priv->front_offset = init->front_offset; |
| @@ -1226,9 +1215,15 @@ static int i915_load_modeset_init(struct drm_device *dev) | |||
| 1226 | if (ret) | 1215 | if (ret) |
| 1227 | DRM_INFO("failed to find VBIOS tables\n"); | 1216 | DRM_INFO("failed to find VBIOS tables\n"); |
| 1228 | 1217 | ||
| 1229 | /* if we have > 1 VGA cards, then disable the radeon VGA resources */ | 1218 | /* If we have > 1 VGA cards, then we need to arbitrate access |
| 1219 | * to the common VGA resources. | ||
| 1220 | * | ||
| 1221 | * If we are a secondary display controller (!PCI_DISPLAY_CLASS_VGA), | ||
| 1222 | * then we do not take part in VGA arbitration and the | ||
| 1223 | * vga_client_register() fails with -ENODEV. | ||
| 1224 | */ | ||
| 1230 | ret = vga_client_register(dev->pdev, dev, NULL, i915_vga_set_decode); | 1225 | ret = vga_client_register(dev->pdev, dev, NULL, i915_vga_set_decode); |
| 1231 | if (ret) | 1226 | if (ret && ret != -ENODEV) |
| 1232 | goto cleanup_ringbuffer; | 1227 | goto cleanup_ringbuffer; |
| 1233 | 1228 | ||
| 1234 | intel_register_dsm_handler(); | 1229 | intel_register_dsm_handler(); |
diff --git a/drivers/gpu/drm/i915/i915_drv.c b/drivers/gpu/drm/i915/i915_drv.c index 72fea2bcfc4..cfb56d0ff36 100644 --- a/drivers/gpu/drm/i915/i915_drv.c +++ b/drivers/gpu/drm/i915/i915_drv.c | |||
| @@ -60,7 +60,7 @@ extern int intel_agp_enabled; | |||
| 60 | 60 | ||
| 61 | #define INTEL_VGA_DEVICE(id, info) { \ | 61 | #define INTEL_VGA_DEVICE(id, info) { \ |
| 62 | .class = PCI_CLASS_DISPLAY_VGA << 8, \ | 62 | .class = PCI_CLASS_DISPLAY_VGA << 8, \ |
| 63 | .class_mask = 0xffff00, \ | 63 | .class_mask = 0xff0000, \ |
| 64 | .vendor = 0x8086, \ | 64 | .vendor = 0x8086, \ |
| 65 | .device = id, \ | 65 | .device = id, \ |
| 66 | .subvendor = PCI_ANY_ID, \ | 66 | .subvendor = PCI_ANY_ID, \ |
| @@ -354,6 +354,7 @@ static int i915_drm_thaw(struct drm_device *dev) | |||
| 354 | error = i915_gem_init_ringbuffer(dev); | 354 | error = i915_gem_init_ringbuffer(dev); |
| 355 | mutex_unlock(&dev->struct_mutex); | 355 | mutex_unlock(&dev->struct_mutex); |
| 356 | 356 | ||
| 357 | drm_mode_config_reset(dev); | ||
| 357 | drm_irq_install(dev); | 358 | drm_irq_install(dev); |
| 358 | 359 | ||
| 359 | /* Resume the modeset for every activated CRTC */ | 360 | /* Resume the modeset for every activated CRTC */ |
| @@ -542,6 +543,7 @@ int i915_reset(struct drm_device *dev, u8 flags) | |||
| 542 | 543 | ||
| 543 | mutex_unlock(&dev->struct_mutex); | 544 | mutex_unlock(&dev->struct_mutex); |
| 544 | drm_irq_uninstall(dev); | 545 | drm_irq_uninstall(dev); |
| 546 | drm_mode_config_reset(dev); | ||
| 545 | drm_irq_install(dev); | 547 | drm_irq_install(dev); |
| 546 | mutex_lock(&dev->struct_mutex); | 548 | mutex_lock(&dev->struct_mutex); |
| 547 | } | 549 | } |
| @@ -566,6 +568,14 @@ int i915_reset(struct drm_device *dev, u8 flags) | |||
| 566 | static int __devinit | 568 | static int __devinit |
| 567 | i915_pci_probe(struct pci_dev *pdev, const struct pci_device_id *ent) | 569 | i915_pci_probe(struct pci_dev *pdev, const struct pci_device_id *ent) |
| 568 | { | 570 | { |
| 571 | /* Only bind to function 0 of the device. Early generations | ||
| 572 | * used function 1 as a placeholder for multi-head. This causes | ||
| 573 | * us confusion instead, especially on the systems where both | ||
| 574 | * functions have the same PCI-ID! | ||
| 575 | */ | ||
| 576 | if (PCI_FUNC(pdev->devfn)) | ||
| 577 | return -ENODEV; | ||
| 578 | |||
| 569 | return drm_get_pci_dev(pdev, ent, &driver); | 579 | return drm_get_pci_dev(pdev, ent, &driver); |
| 570 | } | 580 | } |
| 571 | 581 | ||
| @@ -752,6 +762,9 @@ static int __init i915_init(void) | |||
| 752 | driver.driver_features &= ~DRIVER_MODESET; | 762 | driver.driver_features &= ~DRIVER_MODESET; |
| 753 | #endif | 763 | #endif |
| 754 | 764 | ||
| 765 | if (!(driver.driver_features & DRIVER_MODESET)) | ||
| 766 | driver.get_vblank_timestamp = NULL; | ||
| 767 | |||
| 755 | return drm_init(&driver); | 768 | return drm_init(&driver); |
| 756 | } | 769 | } |
| 757 | 770 | ||
diff --git a/drivers/gpu/drm/i915/i915_drv.h b/drivers/gpu/drm/i915/i915_drv.h index 5969f46ac2d..a0149c619cd 100644 --- a/drivers/gpu/drm/i915/i915_drv.h +++ b/drivers/gpu/drm/i915/i915_drv.h | |||
| @@ -543,8 +543,11 @@ typedef struct drm_i915_private { | |||
| 543 | /** List of all objects in gtt_space. Used to restore gtt | 543 | /** List of all objects in gtt_space. Used to restore gtt |
| 544 | * mappings on resume */ | 544 | * mappings on resume */ |
| 545 | struct list_head gtt_list; | 545 | struct list_head gtt_list; |
| 546 | /** End of mappable part of GTT */ | 546 | |
| 547 | /** Usable portion of the GTT for GEM */ | ||
| 548 | unsigned long gtt_start; | ||
| 547 | unsigned long gtt_mappable_end; | 549 | unsigned long gtt_mappable_end; |
| 550 | unsigned long gtt_end; | ||
| 548 | 551 | ||
| 549 | struct io_mapping *gtt_mapping; | 552 | struct io_mapping *gtt_mapping; |
| 550 | int gtt_mtrr; | 553 | int gtt_mtrr; |
diff --git a/drivers/gpu/drm/i915/i915_gem.c b/drivers/gpu/drm/i915/i915_gem.c index 3dfc848ff75..cf4f74c7c6f 100644 --- a/drivers/gpu/drm/i915/i915_gem.c +++ b/drivers/gpu/drm/i915/i915_gem.c | |||
| @@ -140,12 +140,16 @@ void i915_gem_do_init(struct drm_device *dev, | |||
| 140 | { | 140 | { |
| 141 | drm_i915_private_t *dev_priv = dev->dev_private; | 141 | drm_i915_private_t *dev_priv = dev->dev_private; |
| 142 | 142 | ||
| 143 | drm_mm_init(&dev_priv->mm.gtt_space, start, | 143 | drm_mm_init(&dev_priv->mm.gtt_space, start, end - start); |
| 144 | end - start); | ||
| 145 | 144 | ||
| 145 | dev_priv->mm.gtt_start = start; | ||
| 146 | dev_priv->mm.gtt_mappable_end = mappable_end; | ||
| 147 | dev_priv->mm.gtt_end = end; | ||
| 146 | dev_priv->mm.gtt_total = end - start; | 148 | dev_priv->mm.gtt_total = end - start; |
| 147 | dev_priv->mm.mappable_gtt_total = min(end, mappable_end) - start; | 149 | dev_priv->mm.mappable_gtt_total = min(end, mappable_end) - start; |
| 148 | dev_priv->mm.gtt_mappable_end = mappable_end; | 150 | |
| 151 | /* Take over this portion of the GTT */ | ||
| 152 | intel_gtt_clear_range(start / PAGE_SIZE, (end-start) / PAGE_SIZE); | ||
| 149 | } | 153 | } |
| 150 | 154 | ||
| 151 | int | 155 | int |
| @@ -1857,7 +1861,7 @@ i915_gem_retire_requests_ring(struct drm_device *dev, | |||
| 1857 | 1861 | ||
| 1858 | seqno = ring->get_seqno(ring); | 1862 | seqno = ring->get_seqno(ring); |
| 1859 | 1863 | ||
| 1860 | for (i = 0; i < I915_NUM_RINGS; i++) | 1864 | for (i = 0; i < ARRAY_SIZE(ring->sync_seqno); i++) |
| 1861 | if (seqno >= ring->sync_seqno[i]) | 1865 | if (seqno >= ring->sync_seqno[i]) |
| 1862 | ring->sync_seqno[i] = 0; | 1866 | ring->sync_seqno[i] = 0; |
| 1863 | 1867 | ||
diff --git a/drivers/gpu/drm/i915/i915_gem_execbuffer.c b/drivers/gpu/drm/i915/i915_gem_execbuffer.c index dcfdf4151b6..d2f445e825f 100644 --- a/drivers/gpu/drm/i915/i915_gem_execbuffer.c +++ b/drivers/gpu/drm/i915/i915_gem_execbuffer.c | |||
| @@ -1175,7 +1175,7 @@ i915_gem_do_execbuffer(struct drm_device *dev, void *data, | |||
| 1175 | goto err; | 1175 | goto err; |
| 1176 | 1176 | ||
| 1177 | seqno = i915_gem_next_request_seqno(dev, ring); | 1177 | seqno = i915_gem_next_request_seqno(dev, ring); |
| 1178 | for (i = 0; i < I915_NUM_RINGS-1; i++) { | 1178 | for (i = 0; i < ARRAY_SIZE(ring->sync_seqno); i++) { |
| 1179 | if (seqno < ring->sync_seqno[i]) { | 1179 | if (seqno < ring->sync_seqno[i]) { |
| 1180 | /* The GPU can not handle its semaphore value wrapping, | 1180 | /* The GPU can not handle its semaphore value wrapping, |
| 1181 | * so every billion or so execbuffers, we need to stall | 1181 | * so every billion or so execbuffers, we need to stall |
diff --git a/drivers/gpu/drm/i915/i915_gem_gtt.c b/drivers/gpu/drm/i915/i915_gem_gtt.c index 70433ae50ac..b0abdc64aa9 100644 --- a/drivers/gpu/drm/i915/i915_gem_gtt.c +++ b/drivers/gpu/drm/i915/i915_gem_gtt.c | |||
| @@ -34,6 +34,10 @@ void i915_gem_restore_gtt_mappings(struct drm_device *dev) | |||
| 34 | struct drm_i915_private *dev_priv = dev->dev_private; | 34 | struct drm_i915_private *dev_priv = dev->dev_private; |
| 35 | struct drm_i915_gem_object *obj; | 35 | struct drm_i915_gem_object *obj; |
| 36 | 36 | ||
| 37 | /* First fill our portion of the GTT with scratch pages */ | ||
| 38 | intel_gtt_clear_range(dev_priv->mm.gtt_start / PAGE_SIZE, | ||
| 39 | (dev_priv->mm.gtt_end - dev_priv->mm.gtt_start) / PAGE_SIZE); | ||
| 40 | |||
| 37 | list_for_each_entry(obj, &dev_priv->mm.gtt_list, gtt_list) { | 41 | list_for_each_entry(obj, &dev_priv->mm.gtt_list, gtt_list) { |
| 38 | i915_gem_clflush_object(obj); | 42 | i915_gem_clflush_object(obj); |
| 39 | 43 | ||
diff --git a/drivers/gpu/drm/i915/i915_irq.c b/drivers/gpu/drm/i915/i915_irq.c index b8e509ae065..97f946dcc1a 100644 --- a/drivers/gpu/drm/i915/i915_irq.c +++ b/drivers/gpu/drm/i915/i915_irq.c | |||
| @@ -274,24 +274,35 @@ int i915_get_crtc_scanoutpos(struct drm_device *dev, int pipe, | |||
| 274 | return ret; | 274 | return ret; |
| 275 | } | 275 | } |
| 276 | 276 | ||
| 277 | int i915_get_vblank_timestamp(struct drm_device *dev, int crtc, | 277 | int i915_get_vblank_timestamp(struct drm_device *dev, int pipe, |
| 278 | int *max_error, | 278 | int *max_error, |
| 279 | struct timeval *vblank_time, | 279 | struct timeval *vblank_time, |
| 280 | unsigned flags) | 280 | unsigned flags) |
| 281 | { | 281 | { |
| 282 | struct drm_crtc *drmcrtc; | 282 | struct drm_i915_private *dev_priv = dev->dev_private; |
| 283 | struct drm_crtc *crtc; | ||
| 283 | 284 | ||
| 284 | if (crtc < 0 || crtc >= dev->num_crtcs) { | 285 | if (pipe < 0 || pipe >= dev_priv->num_pipe) { |
| 285 | DRM_ERROR("Invalid crtc %d\n", crtc); | 286 | DRM_ERROR("Invalid crtc %d\n", pipe); |
| 286 | return -EINVAL; | 287 | return -EINVAL; |
| 287 | } | 288 | } |
| 288 | 289 | ||
| 289 | /* Get drm_crtc to timestamp: */ | 290 | /* Get drm_crtc to timestamp: */ |
| 290 | drmcrtc = intel_get_crtc_for_pipe(dev, crtc); | 291 | crtc = intel_get_crtc_for_pipe(dev, pipe); |
| 292 | if (crtc == NULL) { | ||
| 293 | DRM_ERROR("Invalid crtc %d\n", pipe); | ||
| 294 | return -EINVAL; | ||
| 295 | } | ||
| 296 | |||
| 297 | if (!crtc->enabled) { | ||
| 298 | DRM_DEBUG_KMS("crtc %d is disabled\n", pipe); | ||
| 299 | return -EBUSY; | ||
| 300 | } | ||
| 291 | 301 | ||
| 292 | /* Helper routine in DRM core does all the work: */ | 302 | /* Helper routine in DRM core does all the work: */ |
| 293 | return drm_calc_vbltimestamp_from_scanoutpos(dev, crtc, max_error, | 303 | return drm_calc_vbltimestamp_from_scanoutpos(dev, pipe, max_error, |
| 294 | vblank_time, flags, drmcrtc); | 304 | vblank_time, flags, |
| 305 | crtc); | ||
| 295 | } | 306 | } |
| 296 | 307 | ||
| 297 | /* | 308 | /* |
| @@ -348,8 +359,12 @@ static void notify_ring(struct drm_device *dev, | |||
| 348 | struct intel_ring_buffer *ring) | 359 | struct intel_ring_buffer *ring) |
| 349 | { | 360 | { |
| 350 | struct drm_i915_private *dev_priv = dev->dev_private; | 361 | struct drm_i915_private *dev_priv = dev->dev_private; |
| 351 | u32 seqno = ring->get_seqno(ring); | 362 | u32 seqno; |
| 352 | 363 | ||
| 364 | if (ring->obj == NULL) | ||
| 365 | return; | ||
| 366 | |||
| 367 | seqno = ring->get_seqno(ring); | ||
| 353 | trace_i915_gem_request_complete(dev, seqno); | 368 | trace_i915_gem_request_complete(dev, seqno); |
| 354 | 369 | ||
| 355 | ring->irq_seqno = seqno; | 370 | ring->irq_seqno = seqno; |
| @@ -831,6 +846,8 @@ static void i915_capture_error_state(struct drm_device *dev) | |||
| 831 | i++; | 846 | i++; |
| 832 | error->pinned_bo_count = i - error->active_bo_count; | 847 | error->pinned_bo_count = i - error->active_bo_count; |
| 833 | 848 | ||
| 849 | error->active_bo = NULL; | ||
| 850 | error->pinned_bo = NULL; | ||
| 834 | if (i) { | 851 | if (i) { |
| 835 | error->active_bo = kmalloc(sizeof(*error->active_bo)*i, | 852 | error->active_bo = kmalloc(sizeof(*error->active_bo)*i, |
| 836 | GFP_ATOMIC); | 853 | GFP_ATOMIC); |
| @@ -1179,18 +1196,18 @@ irqreturn_t i915_driver_irq_handler(DRM_IRQ_ARGS) | |||
| 1179 | intel_finish_page_flip_plane(dev, 1); | 1196 | intel_finish_page_flip_plane(dev, 1); |
| 1180 | } | 1197 | } |
| 1181 | 1198 | ||
| 1182 | if (pipea_stats & vblank_status) { | 1199 | if (pipea_stats & vblank_status && |
| 1200 | drm_handle_vblank(dev, 0)) { | ||
| 1183 | vblank++; | 1201 | vblank++; |
| 1184 | drm_handle_vblank(dev, 0); | ||
| 1185 | if (!dev_priv->flip_pending_is_done) { | 1202 | if (!dev_priv->flip_pending_is_done) { |
| 1186 | i915_pageflip_stall_check(dev, 0); | 1203 | i915_pageflip_stall_check(dev, 0); |
| 1187 | intel_finish_page_flip(dev, 0); | 1204 | intel_finish_page_flip(dev, 0); |
| 1188 | } | 1205 | } |
| 1189 | } | 1206 | } |
| 1190 | 1207 | ||
| 1191 | if (pipeb_stats & vblank_status) { | 1208 | if (pipeb_stats & vblank_status && |
| 1209 | drm_handle_vblank(dev, 1)) { | ||
| 1192 | vblank++; | 1210 | vblank++; |
| 1193 | drm_handle_vblank(dev, 1); | ||
| 1194 | if (!dev_priv->flip_pending_is_done) { | 1211 | if (!dev_priv->flip_pending_is_done) { |
| 1195 | i915_pageflip_stall_check(dev, 1); | 1212 | i915_pageflip_stall_check(dev, 1); |
| 1196 | intel_finish_page_flip(dev, 1); | 1213 | intel_finish_page_flip(dev, 1); |
| @@ -1278,12 +1295,12 @@ static int i915_wait_irq(struct drm_device * dev, int irq_nr) | |||
| 1278 | if (master_priv->sarea_priv) | 1295 | if (master_priv->sarea_priv) |
| 1279 | master_priv->sarea_priv->perf_boxes |= I915_BOX_WAIT; | 1296 | master_priv->sarea_priv->perf_boxes |= I915_BOX_WAIT; |
| 1280 | 1297 | ||
| 1281 | ret = -ENODEV; | ||
| 1282 | if (ring->irq_get(ring)) { | 1298 | if (ring->irq_get(ring)) { |
| 1283 | DRM_WAIT_ON(ret, ring->irq_queue, 3 * DRM_HZ, | 1299 | DRM_WAIT_ON(ret, ring->irq_queue, 3 * DRM_HZ, |
| 1284 | READ_BREADCRUMB(dev_priv) >= irq_nr); | 1300 | READ_BREADCRUMB(dev_priv) >= irq_nr); |
| 1285 | ring->irq_put(ring); | 1301 | ring->irq_put(ring); |
| 1286 | } | 1302 | } else if (wait_for(READ_BREADCRUMB(dev_priv) >= irq_nr, 3000)) |
| 1303 | ret = -EBUSY; | ||
| 1287 | 1304 | ||
| 1288 | if (ret == -EBUSY) { | 1305 | if (ret == -EBUSY) { |
| 1289 | DRM_ERROR("EBUSY -- rec: %d emitted: %d\n", | 1306 | DRM_ERROR("EBUSY -- rec: %d emitted: %d\n", |
diff --git a/drivers/gpu/drm/i915/i915_reg.h b/drivers/gpu/drm/i915/i915_reg.h index 40a407f41f6..5cfc68940f1 100644 --- a/drivers/gpu/drm/i915/i915_reg.h +++ b/drivers/gpu/drm/i915/i915_reg.h | |||
| @@ -513,6 +513,10 @@ | |||
| 513 | #define GEN6_BLITTER_SYNC_STATUS (1 << 24) | 513 | #define GEN6_BLITTER_SYNC_STATUS (1 << 24) |
| 514 | #define GEN6_BLITTER_USER_INTERRUPT (1 << 22) | 514 | #define GEN6_BLITTER_USER_INTERRUPT (1 << 22) |
| 515 | 515 | ||
| 516 | #define GEN6_BLITTER_ECOSKPD 0x221d0 | ||
| 517 | #define GEN6_BLITTER_LOCK_SHIFT 16 | ||
| 518 | #define GEN6_BLITTER_FBC_NOTIFY (1<<3) | ||
| 519 | |||
| 516 | #define GEN6_BSD_SLEEP_PSMI_CONTROL 0x12050 | 520 | #define GEN6_BSD_SLEEP_PSMI_CONTROL 0x12050 |
| 517 | #define GEN6_BSD_SLEEP_PSMI_CONTROL_RC_ILDL_MESSAGE_MODIFY_MASK (1 << 16) | 521 | #define GEN6_BSD_SLEEP_PSMI_CONTROL_RC_ILDL_MESSAGE_MODIFY_MASK (1 << 16) |
| 518 | #define GEN6_BSD_SLEEP_PSMI_CONTROL_RC_ILDL_MESSAGE_DISABLE (1 << 0) | 522 | #define GEN6_BSD_SLEEP_PSMI_CONTROL_RC_ILDL_MESSAGE_DISABLE (1 << 0) |
| @@ -2626,6 +2630,8 @@ | |||
| 2626 | #define DISPLAY_PORT_PLL_BIOS_2 0x46014 | 2630 | #define DISPLAY_PORT_PLL_BIOS_2 0x46014 |
| 2627 | 2631 | ||
| 2628 | #define PCH_DSPCLK_GATE_D 0x42020 | 2632 | #define PCH_DSPCLK_GATE_D 0x42020 |
| 2633 | # define DPFCUNIT_CLOCK_GATE_DISABLE (1 << 9) | ||
| 2634 | # define DPFCRUNIT_CLOCK_GATE_DISABLE (1 << 8) | ||
| 2629 | # define DPFDUNIT_CLOCK_GATE_DISABLE (1 << 7) | 2635 | # define DPFDUNIT_CLOCK_GATE_DISABLE (1 << 7) |
| 2630 | # define DPARBUNIT_CLOCK_GATE_DISABLE (1 << 5) | 2636 | # define DPARBUNIT_CLOCK_GATE_DISABLE (1 << 5) |
| 2631 | 2637 | ||
diff --git a/drivers/gpu/drm/i915/intel_crt.c b/drivers/gpu/drm/i915/intel_crt.c index 17035b87ee4..8a77ff4a723 100644 --- a/drivers/gpu/drm/i915/intel_crt.c +++ b/drivers/gpu/drm/i915/intel_crt.c | |||
| @@ -535,6 +535,15 @@ static int intel_crt_set_property(struct drm_connector *connector, | |||
| 535 | return 0; | 535 | return 0; |
| 536 | } | 536 | } |
| 537 | 537 | ||
| 538 | static void intel_crt_reset(struct drm_connector *connector) | ||
| 539 | { | ||
| 540 | struct drm_device *dev = connector->dev; | ||
| 541 | struct intel_crt *crt = intel_attached_crt(connector); | ||
| 542 | |||
| 543 | if (HAS_PCH_SPLIT(dev)) | ||
| 544 | crt->force_hotplug_required = 1; | ||
| 545 | } | ||
| 546 | |||
| 538 | /* | 547 | /* |
| 539 | * Routines for controlling stuff on the analog port | 548 | * Routines for controlling stuff on the analog port |
| 540 | */ | 549 | */ |
| @@ -548,6 +557,7 @@ static const struct drm_encoder_helper_funcs intel_crt_helper_funcs = { | |||
| 548 | }; | 557 | }; |
| 549 | 558 | ||
| 550 | static const struct drm_connector_funcs intel_crt_connector_funcs = { | 559 | static const struct drm_connector_funcs intel_crt_connector_funcs = { |
| 560 | .reset = intel_crt_reset, | ||
| 551 | .dpms = drm_helper_connector_dpms, | 561 | .dpms = drm_helper_connector_dpms, |
| 552 | .detect = intel_crt_detect, | 562 | .detect = intel_crt_detect, |
| 553 | .fill_modes = drm_helper_probe_single_connector_modes, | 563 | .fill_modes = drm_helper_probe_single_connector_modes, |
diff --git a/drivers/gpu/drm/i915/intel_display.c b/drivers/gpu/drm/i915/intel_display.c index 98967f3b772..7e42aa58650 100644 --- a/drivers/gpu/drm/i915/intel_display.c +++ b/drivers/gpu/drm/i915/intel_display.c | |||
| @@ -1213,6 +1213,26 @@ static bool g4x_fbc_enabled(struct drm_device *dev) | |||
| 1213 | return I915_READ(DPFC_CONTROL) & DPFC_CTL_EN; | 1213 | return I915_READ(DPFC_CONTROL) & DPFC_CTL_EN; |
| 1214 | } | 1214 | } |
| 1215 | 1215 | ||
| 1216 | static void sandybridge_blit_fbc_update(struct drm_device *dev) | ||
| 1217 | { | ||
| 1218 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 1219 | u32 blt_ecoskpd; | ||
| 1220 | |||
| 1221 | /* Make sure blitter notifies FBC of writes */ | ||
| 1222 | __gen6_force_wake_get(dev_priv); | ||
| 1223 | blt_ecoskpd = I915_READ(GEN6_BLITTER_ECOSKPD); | ||
| 1224 | blt_ecoskpd |= GEN6_BLITTER_FBC_NOTIFY << | ||
| 1225 | GEN6_BLITTER_LOCK_SHIFT; | ||
| 1226 | I915_WRITE(GEN6_BLITTER_ECOSKPD, blt_ecoskpd); | ||
| 1227 | blt_ecoskpd |= GEN6_BLITTER_FBC_NOTIFY; | ||
| 1228 | I915_WRITE(GEN6_BLITTER_ECOSKPD, blt_ecoskpd); | ||
| 1229 | blt_ecoskpd &= ~(GEN6_BLITTER_FBC_NOTIFY << | ||
| 1230 | GEN6_BLITTER_LOCK_SHIFT); | ||
| 1231 | I915_WRITE(GEN6_BLITTER_ECOSKPD, blt_ecoskpd); | ||
| 1232 | POSTING_READ(GEN6_BLITTER_ECOSKPD); | ||
| 1233 | __gen6_force_wake_put(dev_priv); | ||
| 1234 | } | ||
| 1235 | |||
| 1216 | static void ironlake_enable_fbc(struct drm_crtc *crtc, unsigned long interval) | 1236 | static void ironlake_enable_fbc(struct drm_crtc *crtc, unsigned long interval) |
| 1217 | { | 1237 | { |
| 1218 | struct drm_device *dev = crtc->dev; | 1238 | struct drm_device *dev = crtc->dev; |
| @@ -1266,6 +1286,7 @@ static void ironlake_enable_fbc(struct drm_crtc *crtc, unsigned long interval) | |||
| 1266 | I915_WRITE(SNB_DPFC_CTL_SA, | 1286 | I915_WRITE(SNB_DPFC_CTL_SA, |
| 1267 | SNB_CPU_FENCE_ENABLE | dev_priv->cfb_fence); | 1287 | SNB_CPU_FENCE_ENABLE | dev_priv->cfb_fence); |
| 1268 | I915_WRITE(DPFC_CPU_FENCE_OFFSET, crtc->y); | 1288 | I915_WRITE(DPFC_CPU_FENCE_OFFSET, crtc->y); |
| 1289 | sandybridge_blit_fbc_update(dev); | ||
| 1269 | } | 1290 | } |
| 1270 | 1291 | ||
| 1271 | DRM_DEBUG_KMS("enabled fbc on plane %d\n", intel_crtc->plane); | 1292 | DRM_DEBUG_KMS("enabled fbc on plane %d\n", intel_crtc->plane); |
| @@ -5530,6 +5551,18 @@ cleanup_work: | |||
| 5530 | return ret; | 5551 | return ret; |
| 5531 | } | 5552 | } |
| 5532 | 5553 | ||
| 5554 | static void intel_crtc_reset(struct drm_crtc *crtc) | ||
| 5555 | { | ||
| 5556 | struct intel_crtc *intel_crtc = to_intel_crtc(crtc); | ||
| 5557 | |||
| 5558 | /* Reset flags back to the 'unknown' status so that they | ||
| 5559 | * will be correctly set on the initial modeset. | ||
| 5560 | */ | ||
| 5561 | intel_crtc->cursor_addr = 0; | ||
| 5562 | intel_crtc->dpms_mode = -1; | ||
| 5563 | intel_crtc->active = true; /* force the pipe off on setup_init_config */ | ||
| 5564 | } | ||
| 5565 | |||
| 5533 | static struct drm_crtc_helper_funcs intel_helper_funcs = { | 5566 | static struct drm_crtc_helper_funcs intel_helper_funcs = { |
| 5534 | .dpms = intel_crtc_dpms, | 5567 | .dpms = intel_crtc_dpms, |
| 5535 | .mode_fixup = intel_crtc_mode_fixup, | 5568 | .mode_fixup = intel_crtc_mode_fixup, |
| @@ -5541,6 +5574,7 @@ static struct drm_crtc_helper_funcs intel_helper_funcs = { | |||
| 5541 | }; | 5574 | }; |
| 5542 | 5575 | ||
| 5543 | static const struct drm_crtc_funcs intel_crtc_funcs = { | 5576 | static const struct drm_crtc_funcs intel_crtc_funcs = { |
| 5577 | .reset = intel_crtc_reset, | ||
| 5544 | .cursor_set = intel_crtc_cursor_set, | 5578 | .cursor_set = intel_crtc_cursor_set, |
| 5545 | .cursor_move = intel_crtc_cursor_move, | 5579 | .cursor_move = intel_crtc_cursor_move, |
| 5546 | .gamma_set = intel_crtc_gamma_set, | 5580 | .gamma_set = intel_crtc_gamma_set, |
| @@ -5631,9 +5665,7 @@ static void intel_crtc_init(struct drm_device *dev, int pipe) | |||
| 5631 | dev_priv->plane_to_crtc_mapping[intel_crtc->plane] = &intel_crtc->base; | 5665 | dev_priv->plane_to_crtc_mapping[intel_crtc->plane] = &intel_crtc->base; |
| 5632 | dev_priv->pipe_to_crtc_mapping[intel_crtc->pipe] = &intel_crtc->base; | 5666 | dev_priv->pipe_to_crtc_mapping[intel_crtc->pipe] = &intel_crtc->base; |
| 5633 | 5667 | ||
| 5634 | intel_crtc->cursor_addr = 0; | 5668 | intel_crtc_reset(&intel_crtc->base); |
| 5635 | intel_crtc->dpms_mode = -1; | ||
| 5636 | intel_crtc->active = true; /* force the pipe off on setup_init_config */ | ||
| 5637 | 5669 | ||
| 5638 | if (HAS_PCH_SPLIT(dev)) { | 5670 | if (HAS_PCH_SPLIT(dev)) { |
| 5639 | intel_helper_funcs.prepare = ironlake_crtc_prepare; | 5671 | intel_helper_funcs.prepare = ironlake_crtc_prepare; |
| @@ -6286,7 +6318,9 @@ void intel_enable_clock_gating(struct drm_device *dev) | |||
| 6286 | 6318 | ||
| 6287 | if (IS_GEN5(dev)) { | 6319 | if (IS_GEN5(dev)) { |
| 6288 | /* Required for FBC */ | 6320 | /* Required for FBC */ |
| 6289 | dspclk_gate |= DPFDUNIT_CLOCK_GATE_DISABLE; | 6321 | dspclk_gate |= DPFCUNIT_CLOCK_GATE_DISABLE | |
| 6322 | DPFCRUNIT_CLOCK_GATE_DISABLE | | ||
| 6323 | DPFDUNIT_CLOCK_GATE_DISABLE; | ||
| 6290 | /* Required for CxSR */ | 6324 | /* Required for CxSR */ |
| 6291 | dspclk_gate |= DPARBUNIT_CLOCK_GATE_DISABLE; | 6325 | dspclk_gate |= DPARBUNIT_CLOCK_GATE_DISABLE; |
| 6292 | 6326 | ||
diff --git a/drivers/gpu/drm/i915/intel_opregion.c b/drivers/gpu/drm/i915/intel_opregion.c index f295a7aaadf..64fd64443ca 100644 --- a/drivers/gpu/drm/i915/intel_opregion.c +++ b/drivers/gpu/drm/i915/intel_opregion.c | |||
| @@ -26,6 +26,7 @@ | |||
| 26 | */ | 26 | */ |
| 27 | 27 | ||
| 28 | #include <linux/acpi.h> | 28 | #include <linux/acpi.h> |
| 29 | #include <linux/acpi_io.h> | ||
| 29 | #include <acpi/video.h> | 30 | #include <acpi/video.h> |
| 30 | 31 | ||
| 31 | #include "drmP.h" | 32 | #include "drmP.h" |
| @@ -476,7 +477,7 @@ int intel_opregion_setup(struct drm_device *dev) | |||
| 476 | return -ENOTSUPP; | 477 | return -ENOTSUPP; |
| 477 | } | 478 | } |
| 478 | 479 | ||
| 479 | base = ioremap(asls, OPREGION_SIZE); | 480 | base = acpi_os_ioremap(asls, OPREGION_SIZE); |
| 480 | if (!base) | 481 | if (!base) |
| 481 | return -ENOMEM; | 482 | return -ENOMEM; |
| 482 | 483 | ||
diff --git a/drivers/gpu/drm/i915/intel_ringbuffer.c b/drivers/gpu/drm/i915/intel_ringbuffer.c index f6b9baa6a63..6218fa97aa1 100644 --- a/drivers/gpu/drm/i915/intel_ringbuffer.c +++ b/drivers/gpu/drm/i915/intel_ringbuffer.c | |||
| @@ -34,6 +34,14 @@ | |||
| 34 | #include "i915_trace.h" | 34 | #include "i915_trace.h" |
| 35 | #include "intel_drv.h" | 35 | #include "intel_drv.h" |
| 36 | 36 | ||
| 37 | static inline int ring_space(struct intel_ring_buffer *ring) | ||
| 38 | { | ||
| 39 | int space = (ring->head & HEAD_ADDR) - (ring->tail + 8); | ||
| 40 | if (space < 0) | ||
| 41 | space += ring->size; | ||
| 42 | return space; | ||
| 43 | } | ||
| 44 | |||
| 37 | static u32 i915_gem_get_seqno(struct drm_device *dev) | 45 | static u32 i915_gem_get_seqno(struct drm_device *dev) |
| 38 | { | 46 | { |
| 39 | drm_i915_private_t *dev_priv = dev->dev_private; | 47 | drm_i915_private_t *dev_priv = dev->dev_private; |
| @@ -204,11 +212,9 @@ static int init_ring_common(struct intel_ring_buffer *ring) | |||
| 204 | if (!drm_core_check_feature(ring->dev, DRIVER_MODESET)) | 212 | if (!drm_core_check_feature(ring->dev, DRIVER_MODESET)) |
| 205 | i915_kernel_lost_context(ring->dev); | 213 | i915_kernel_lost_context(ring->dev); |
| 206 | else { | 214 | else { |
| 207 | ring->head = I915_READ_HEAD(ring) & HEAD_ADDR; | 215 | ring->head = I915_READ_HEAD(ring); |
| 208 | ring->tail = I915_READ_TAIL(ring) & TAIL_ADDR; | 216 | ring->tail = I915_READ_TAIL(ring) & TAIL_ADDR; |
| 209 | ring->space = ring->head - (ring->tail + 8); | 217 | ring->space = ring_space(ring); |
| 210 | if (ring->space < 0) | ||
| 211 | ring->space += ring->size; | ||
| 212 | } | 218 | } |
| 213 | 219 | ||
| 214 | return 0; | 220 | return 0; |
| @@ -921,32 +927,34 @@ static int intel_wrap_ring_buffer(struct intel_ring_buffer *ring) | |||
| 921 | } | 927 | } |
| 922 | 928 | ||
| 923 | ring->tail = 0; | 929 | ring->tail = 0; |
| 924 | ring->space = ring->head - 8; | 930 | ring->space = ring_space(ring); |
| 925 | 931 | ||
| 926 | return 0; | 932 | return 0; |
| 927 | } | 933 | } |
| 928 | 934 | ||
| 929 | int intel_wait_ring_buffer(struct intel_ring_buffer *ring, int n) | 935 | int intel_wait_ring_buffer(struct intel_ring_buffer *ring, int n) |
| 930 | { | 936 | { |
| 931 | int reread = 0; | ||
| 932 | struct drm_device *dev = ring->dev; | 937 | struct drm_device *dev = ring->dev; |
| 933 | struct drm_i915_private *dev_priv = dev->dev_private; | 938 | struct drm_i915_private *dev_priv = dev->dev_private; |
| 934 | unsigned long end; | 939 | unsigned long end; |
| 935 | u32 head; | 940 | u32 head; |
| 936 | 941 | ||
| 942 | /* If the reported head position has wrapped or hasn't advanced, | ||
| 943 | * fallback to the slow and accurate path. | ||
| 944 | */ | ||
| 945 | head = intel_read_status_page(ring, 4); | ||
| 946 | if (head > ring->head) { | ||
| 947 | ring->head = head; | ||
| 948 | ring->space = ring_space(ring); | ||
| 949 | if (ring->space >= n) | ||
| 950 | return 0; | ||
| 951 | } | ||
| 952 | |||
| 937 | trace_i915_ring_wait_begin (dev); | 953 | trace_i915_ring_wait_begin (dev); |
| 938 | end = jiffies + 3 * HZ; | 954 | end = jiffies + 3 * HZ; |
| 939 | do { | 955 | do { |
| 940 | /* If the reported head position has wrapped or hasn't advanced, | 956 | ring->head = I915_READ_HEAD(ring); |
| 941 | * fallback to the slow and accurate path. | 957 | ring->space = ring_space(ring); |
| 942 | */ | ||
| 943 | head = intel_read_status_page(ring, 4); | ||
| 944 | if (reread) | ||
| 945 | head = I915_READ_HEAD(ring); | ||
| 946 | ring->head = head & HEAD_ADDR; | ||
| 947 | ring->space = ring->head - (ring->tail + 8); | ||
| 948 | if (ring->space < 0) | ||
| 949 | ring->space += ring->size; | ||
| 950 | if (ring->space >= n) { | 958 | if (ring->space >= n) { |
| 951 | trace_i915_ring_wait_end(dev); | 959 | trace_i915_ring_wait_end(dev); |
| 952 | return 0; | 960 | return 0; |
| @@ -961,7 +969,6 @@ int intel_wait_ring_buffer(struct intel_ring_buffer *ring, int n) | |||
| 961 | msleep(1); | 969 | msleep(1); |
| 962 | if (atomic_read(&dev_priv->mm.wedged)) | 970 | if (atomic_read(&dev_priv->mm.wedged)) |
| 963 | return -EAGAIN; | 971 | return -EAGAIN; |
| 964 | reread = 1; | ||
| 965 | } while (!time_after(jiffies, end)); | 972 | } while (!time_after(jiffies, end)); |
| 966 | trace_i915_ring_wait_end (dev); | 973 | trace_i915_ring_wait_end (dev); |
| 967 | return -EBUSY; | 974 | return -EBUSY; |
| @@ -1292,6 +1299,48 @@ int intel_init_render_ring_buffer(struct drm_device *dev) | |||
| 1292 | return intel_init_ring_buffer(dev, ring); | 1299 | return intel_init_ring_buffer(dev, ring); |
| 1293 | } | 1300 | } |
| 1294 | 1301 | ||
| 1302 | int intel_render_ring_init_dri(struct drm_device *dev, u64 start, u32 size) | ||
| 1303 | { | ||
| 1304 | drm_i915_private_t *dev_priv = dev->dev_private; | ||
| 1305 | struct intel_ring_buffer *ring = &dev_priv->ring[RCS]; | ||
| 1306 | |||
| 1307 | *ring = render_ring; | ||
| 1308 | if (INTEL_INFO(dev)->gen >= 6) { | ||
| 1309 | ring->add_request = gen6_add_request; | ||
| 1310 | ring->irq_get = gen6_render_ring_get_irq; | ||
| 1311 | ring->irq_put = gen6_render_ring_put_irq; | ||
| 1312 | } else if (IS_GEN5(dev)) { | ||
| 1313 | ring->add_request = pc_render_add_request; | ||
| 1314 | ring->get_seqno = pc_render_get_seqno; | ||
| 1315 | } | ||
| 1316 | |||
| 1317 | ring->dev = dev; | ||
| 1318 | INIT_LIST_HEAD(&ring->active_list); | ||
| 1319 | INIT_LIST_HEAD(&ring->request_list); | ||
| 1320 | INIT_LIST_HEAD(&ring->gpu_write_list); | ||
| 1321 | |||
| 1322 | ring->size = size; | ||
| 1323 | ring->effective_size = ring->size; | ||
| 1324 | if (IS_I830(ring->dev)) | ||
| 1325 | ring->effective_size -= 128; | ||
| 1326 | |||
| 1327 | ring->map.offset = start; | ||
| 1328 | ring->map.size = size; | ||
| 1329 | ring->map.type = 0; | ||
| 1330 | ring->map.flags = 0; | ||
| 1331 | ring->map.mtrr = 0; | ||
| 1332 | |||
| 1333 | drm_core_ioremap_wc(&ring->map, dev); | ||
| 1334 | if (ring->map.handle == NULL) { | ||
| 1335 | DRM_ERROR("can not ioremap virtual address for" | ||
| 1336 | " ring buffer\n"); | ||
| 1337 | return -ENOMEM; | ||
| 1338 | } | ||
| 1339 | |||
| 1340 | ring->virtual_start = (void __force __iomem *)ring->map.handle; | ||
| 1341 | return 0; | ||
| 1342 | } | ||
| 1343 | |||
| 1295 | int intel_init_bsd_ring_buffer(struct drm_device *dev) | 1344 | int intel_init_bsd_ring_buffer(struct drm_device *dev) |
| 1296 | { | 1345 | { |
| 1297 | drm_i915_private_t *dev_priv = dev->dev_private; | 1346 | drm_i915_private_t *dev_priv = dev->dev_private; |
diff --git a/drivers/gpu/drm/i915/intel_ringbuffer.h b/drivers/gpu/drm/i915/intel_ringbuffer.h index 5b0abfa881f..6d6fde85a63 100644 --- a/drivers/gpu/drm/i915/intel_ringbuffer.h +++ b/drivers/gpu/drm/i915/intel_ringbuffer.h | |||
| @@ -166,4 +166,7 @@ int intel_init_blt_ring_buffer(struct drm_device *dev); | |||
| 166 | u32 intel_ring_get_active_head(struct intel_ring_buffer *ring); | 166 | u32 intel_ring_get_active_head(struct intel_ring_buffer *ring); |
| 167 | void intel_ring_setup_status_page(struct intel_ring_buffer *ring); | 167 | void intel_ring_setup_status_page(struct intel_ring_buffer *ring); |
| 168 | 168 | ||
| 169 | /* DRI warts */ | ||
| 170 | int intel_render_ring_init_dri(struct drm_device *dev, u64 start, u32 size); | ||
| 171 | |||
| 169 | #endif /* _INTEL_RINGBUFFER_H_ */ | 172 | #endif /* _INTEL_RINGBUFFER_H_ */ |
diff --git a/drivers/gpu/drm/i915/intel_sdvo.c b/drivers/gpu/drm/i915/intel_sdvo.c index 45cd37652a3..6a09c1413d6 100644 --- a/drivers/gpu/drm/i915/intel_sdvo.c +++ b/drivers/gpu/drm/i915/intel_sdvo.c | |||
| @@ -473,20 +473,6 @@ static bool intel_sdvo_write_cmd(struct intel_sdvo *intel_sdvo, u8 cmd, | |||
| 473 | return false; | 473 | return false; |
| 474 | } | 474 | } |
| 475 | 475 | ||
| 476 | i = 3; | ||
| 477 | while (status == SDVO_CMD_STATUS_PENDING && i--) { | ||
| 478 | if (!intel_sdvo_read_byte(intel_sdvo, | ||
| 479 | SDVO_I2C_CMD_STATUS, | ||
| 480 | &status)) | ||
| 481 | return false; | ||
| 482 | } | ||
| 483 | if (status != SDVO_CMD_STATUS_SUCCESS) { | ||
| 484 | DRM_DEBUG_KMS("command returns response %s [%d]\n", | ||
| 485 | status <= SDVO_CMD_STATUS_SCALING_NOT_SUPP ? cmd_status_names[status] : "???", | ||
| 486 | status); | ||
| 487 | return false; | ||
| 488 | } | ||
| 489 | |||
| 490 | return true; | 476 | return true; |
| 491 | } | 477 | } |
| 492 | 478 | ||
| @@ -497,6 +483,8 @@ static bool intel_sdvo_read_response(struct intel_sdvo *intel_sdvo, | |||
| 497 | u8 status; | 483 | u8 status; |
| 498 | int i; | 484 | int i; |
| 499 | 485 | ||
| 486 | DRM_DEBUG_KMS("%s: R: ", SDVO_NAME(intel_sdvo)); | ||
| 487 | |||
| 500 | /* | 488 | /* |
| 501 | * The documentation states that all commands will be | 489 | * The documentation states that all commands will be |
| 502 | * processed within 15µs, and that we need only poll | 490 | * processed within 15µs, and that we need only poll |
| @@ -505,14 +493,19 @@ static bool intel_sdvo_read_response(struct intel_sdvo *intel_sdvo, | |||
| 505 | * | 493 | * |
| 506 | * Check 5 times in case the hardware failed to read the docs. | 494 | * Check 5 times in case the hardware failed to read the docs. |
| 507 | */ | 495 | */ |
| 508 | do { | 496 | if (!intel_sdvo_read_byte(intel_sdvo, |
| 497 | SDVO_I2C_CMD_STATUS, | ||
| 498 | &status)) | ||
| 499 | goto log_fail; | ||
| 500 | |||
| 501 | while (status == SDVO_CMD_STATUS_PENDING && retry--) { | ||
| 502 | udelay(15); | ||
| 509 | if (!intel_sdvo_read_byte(intel_sdvo, | 503 | if (!intel_sdvo_read_byte(intel_sdvo, |
| 510 | SDVO_I2C_CMD_STATUS, | 504 | SDVO_I2C_CMD_STATUS, |
| 511 | &status)) | 505 | &status)) |
| 512 | return false; | 506 | goto log_fail; |
| 513 | } while (status == SDVO_CMD_STATUS_PENDING && --retry); | 507 | } |
| 514 | 508 | ||
| 515 | DRM_DEBUG_KMS("%s: R: ", SDVO_NAME(intel_sdvo)); | ||
| 516 | if (status <= SDVO_CMD_STATUS_SCALING_NOT_SUPP) | 509 | if (status <= SDVO_CMD_STATUS_SCALING_NOT_SUPP) |
| 517 | DRM_LOG_KMS("(%s)", cmd_status_names[status]); | 510 | DRM_LOG_KMS("(%s)", cmd_status_names[status]); |
| 518 | else | 511 | else |
| @@ -533,7 +526,7 @@ static bool intel_sdvo_read_response(struct intel_sdvo *intel_sdvo, | |||
| 533 | return true; | 526 | return true; |
| 534 | 527 | ||
| 535 | log_fail: | 528 | log_fail: |
| 536 | DRM_LOG_KMS("\n"); | 529 | DRM_LOG_KMS("... failed\n"); |
| 537 | return false; | 530 | return false; |
| 538 | } | 531 | } |
| 539 | 532 | ||
| @@ -550,6 +543,7 @@ static int intel_sdvo_get_pixel_multiplier(struct drm_display_mode *mode) | |||
| 550 | static bool intel_sdvo_set_control_bus_switch(struct intel_sdvo *intel_sdvo, | 543 | static bool intel_sdvo_set_control_bus_switch(struct intel_sdvo *intel_sdvo, |
| 551 | u8 ddc_bus) | 544 | u8 ddc_bus) |
| 552 | { | 545 | { |
| 546 | /* This must be the immediately preceding write before the i2c xfer */ | ||
| 553 | return intel_sdvo_write_cmd(intel_sdvo, | 547 | return intel_sdvo_write_cmd(intel_sdvo, |
| 554 | SDVO_CMD_SET_CONTROL_BUS_SWITCH, | 548 | SDVO_CMD_SET_CONTROL_BUS_SWITCH, |
| 555 | &ddc_bus, 1); | 549 | &ddc_bus, 1); |
| @@ -557,7 +551,10 @@ static bool intel_sdvo_set_control_bus_switch(struct intel_sdvo *intel_sdvo, | |||
| 557 | 551 | ||
| 558 | static bool intel_sdvo_set_value(struct intel_sdvo *intel_sdvo, u8 cmd, const void *data, int len) | 552 | static bool intel_sdvo_set_value(struct intel_sdvo *intel_sdvo, u8 cmd, const void *data, int len) |
| 559 | { | 553 | { |
| 560 | return intel_sdvo_write_cmd(intel_sdvo, cmd, data, len); | 554 | if (!intel_sdvo_write_cmd(intel_sdvo, cmd, data, len)) |
| 555 | return false; | ||
| 556 | |||
| 557 | return intel_sdvo_read_response(intel_sdvo, NULL, 0); | ||
| 561 | } | 558 | } |
| 562 | 559 | ||
| 563 | static bool | 560 | static bool |
| @@ -859,18 +856,21 @@ static bool intel_sdvo_set_avi_infoframe(struct intel_sdvo *intel_sdvo) | |||
| 859 | 856 | ||
| 860 | intel_dip_infoframe_csum(&avi_if); | 857 | intel_dip_infoframe_csum(&avi_if); |
| 861 | 858 | ||
| 862 | if (!intel_sdvo_write_cmd(intel_sdvo, SDVO_CMD_SET_HBUF_INDEX, | 859 | if (!intel_sdvo_set_value(intel_sdvo, |
| 860 | SDVO_CMD_SET_HBUF_INDEX, | ||
| 863 | set_buf_index, 2)) | 861 | set_buf_index, 2)) |
| 864 | return false; | 862 | return false; |
| 865 | 863 | ||
| 866 | for (i = 0; i < sizeof(avi_if); i += 8) { | 864 | for (i = 0; i < sizeof(avi_if); i += 8) { |
| 867 | if (!intel_sdvo_write_cmd(intel_sdvo, SDVO_CMD_SET_HBUF_DATA, | 865 | if (!intel_sdvo_set_value(intel_sdvo, |
| 866 | SDVO_CMD_SET_HBUF_DATA, | ||
| 868 | data, 8)) | 867 | data, 8)) |
| 869 | return false; | 868 | return false; |
| 870 | data++; | 869 | data++; |
| 871 | } | 870 | } |
| 872 | 871 | ||
| 873 | return intel_sdvo_write_cmd(intel_sdvo, SDVO_CMD_SET_HBUF_TXRATE, | 872 | return intel_sdvo_set_value(intel_sdvo, |
| 873 | SDVO_CMD_SET_HBUF_TXRATE, | ||
| 874 | &tx_rate, 1); | 874 | &tx_rate, 1); |
| 875 | } | 875 | } |
| 876 | 876 | ||
diff --git a/drivers/gpu/drm/nouveau/nouveau_bios.c b/drivers/gpu/drm/nouveau/nouveau_bios.c index 2aef5cd3acf..49e5e99917e 100644 --- a/drivers/gpu/drm/nouveau/nouveau_bios.c +++ b/drivers/gpu/drm/nouveau/nouveau_bios.c | |||
| @@ -6310,6 +6310,9 @@ void merge_like_dcb_entries(struct drm_device *dev, struct dcb_table *dcb) | |||
| 6310 | static bool | 6310 | static bool |
| 6311 | apply_dcb_encoder_quirks(struct drm_device *dev, int idx, u32 *conn, u32 *conf) | 6311 | apply_dcb_encoder_quirks(struct drm_device *dev, int idx, u32 *conn, u32 *conf) |
| 6312 | { | 6312 | { |
| 6313 | struct drm_nouveau_private *dev_priv = dev->dev_private; | ||
| 6314 | struct dcb_table *dcb = &dev_priv->vbios.dcb; | ||
| 6315 | |||
| 6313 | /* Dell Precision M6300 | 6316 | /* Dell Precision M6300 |
| 6314 | * DCB entry 2: 02025312 00000010 | 6317 | * DCB entry 2: 02025312 00000010 |
| 6315 | * DCB entry 3: 02026312 00000020 | 6318 | * DCB entry 3: 02026312 00000020 |
| @@ -6327,6 +6330,18 @@ apply_dcb_encoder_quirks(struct drm_device *dev, int idx, u32 *conn, u32 *conf) | |||
| 6327 | return false; | 6330 | return false; |
| 6328 | } | 6331 | } |
| 6329 | 6332 | ||
| 6333 | /* GeForce3 Ti 200 | ||
| 6334 | * | ||
| 6335 | * DCB reports an LVDS output that should be TMDS: | ||
| 6336 | * DCB entry 1: f2005014 ffffffff | ||
| 6337 | */ | ||
| 6338 | if (nv_match_device(dev, 0x0201, 0x1462, 0x8851)) { | ||
| 6339 | if (*conn == 0xf2005014 && *conf == 0xffffffff) { | ||
| 6340 | fabricate_dcb_output(dcb, OUTPUT_TMDS, 1, 1, 1); | ||
| 6341 | return false; | ||
| 6342 | } | ||
| 6343 | } | ||
| 6344 | |||
| 6330 | return true; | 6345 | return true; |
| 6331 | } | 6346 | } |
| 6332 | 6347 | ||
diff --git a/drivers/gpu/drm/nouveau/nouveau_drv.c b/drivers/gpu/drm/nouveau/nouveau_drv.c index 13bb672a16f..f658a04eecf 100644 --- a/drivers/gpu/drm/nouveau/nouveau_drv.c +++ b/drivers/gpu/drm/nouveau/nouveau_drv.c | |||
| @@ -234,9 +234,9 @@ nouveau_pci_suspend(struct pci_dev *pdev, pm_message_t pm_state) | |||
| 234 | pci_set_power_state(pdev, PCI_D3hot); | 234 | pci_set_power_state(pdev, PCI_D3hot); |
| 235 | } | 235 | } |
| 236 | 236 | ||
| 237 | acquire_console_sem(); | 237 | console_lock(); |
| 238 | nouveau_fbcon_set_suspend(dev, 1); | 238 | nouveau_fbcon_set_suspend(dev, 1); |
| 239 | release_console_sem(); | 239 | console_unlock(); |
| 240 | nouveau_fbcon_restore_accel(dev); | 240 | nouveau_fbcon_restore_accel(dev); |
| 241 | return 0; | 241 | return 0; |
| 242 | 242 | ||
| @@ -359,9 +359,9 @@ nouveau_pci_resume(struct pci_dev *pdev) | |||
| 359 | nv_crtc->lut.depth = 0; | 359 | nv_crtc->lut.depth = 0; |
| 360 | } | 360 | } |
| 361 | 361 | ||
| 362 | acquire_console_sem(); | 362 | console_lock(); |
| 363 | nouveau_fbcon_set_suspend(dev, 0); | 363 | nouveau_fbcon_set_suspend(dev, 0); |
| 364 | release_console_sem(); | 364 | console_unlock(); |
| 365 | 365 | ||
| 366 | nouveau_fbcon_zfill_all(dev); | 366 | nouveau_fbcon_zfill_all(dev); |
| 367 | 367 | ||
diff --git a/drivers/gpu/drm/nouveau/nouveau_drv.h b/drivers/gpu/drm/nouveau/nouveau_drv.h index 01bffc4412d..9821fcacc3d 100644 --- a/drivers/gpu/drm/nouveau/nouveau_drv.h +++ b/drivers/gpu/drm/nouveau/nouveau_drv.h | |||
| @@ -848,9 +848,6 @@ extern void nv10_mem_put_tile_region(struct drm_device *dev, | |||
| 848 | struct nouveau_fence *fence); | 848 | struct nouveau_fence *fence); |
| 849 | extern const struct ttm_mem_type_manager_func nouveau_vram_manager; | 849 | extern const struct ttm_mem_type_manager_func nouveau_vram_manager; |
| 850 | 850 | ||
| 851 | /* nvc0_vram.c */ | ||
| 852 | extern const struct ttm_mem_type_manager_func nvc0_vram_manager; | ||
| 853 | |||
| 854 | /* nouveau_notifier.c */ | 851 | /* nouveau_notifier.c */ |
| 855 | extern int nouveau_notifier_init_channel(struct nouveau_channel *); | 852 | extern int nouveau_notifier_init_channel(struct nouveau_channel *); |
| 856 | extern void nouveau_notifier_takedown_channel(struct nouveau_channel *); | 853 | extern void nouveau_notifier_takedown_channel(struct nouveau_channel *); |
diff --git a/drivers/gpu/drm/nouveau/nouveau_pm.c b/drivers/gpu/drm/nouveau/nouveau_pm.c index fb846a3fef1..f05c0cddfec 100644 --- a/drivers/gpu/drm/nouveau/nouveau_pm.c +++ b/drivers/gpu/drm/nouveau/nouveau_pm.c | |||
| @@ -443,7 +443,7 @@ nouveau_hwmon_fini(struct drm_device *dev) | |||
| 443 | struct nouveau_pm_engine *pm = &dev_priv->engine.pm; | 443 | struct nouveau_pm_engine *pm = &dev_priv->engine.pm; |
| 444 | 444 | ||
| 445 | if (pm->hwmon) { | 445 | if (pm->hwmon) { |
| 446 | sysfs_remove_group(&pm->hwmon->kobj, &hwmon_attrgroup); | 446 | sysfs_remove_group(&dev->pdev->dev.kobj, &hwmon_attrgroup); |
| 447 | hwmon_device_unregister(pm->hwmon); | 447 | hwmon_device_unregister(pm->hwmon); |
| 448 | } | 448 | } |
| 449 | #endif | 449 | #endif |
diff --git a/drivers/gpu/drm/nouveau/nouveau_temp.c b/drivers/gpu/drm/nouveau/nouveau_temp.c index 7ecc4adc1e4..8d9968e1cba 100644 --- a/drivers/gpu/drm/nouveau/nouveau_temp.c +++ b/drivers/gpu/drm/nouveau/nouveau_temp.c | |||
| @@ -265,8 +265,8 @@ nouveau_temp_probe_i2c(struct drm_device *dev) | |||
| 265 | struct i2c_board_info info[] = { | 265 | struct i2c_board_info info[] = { |
| 266 | { I2C_BOARD_INFO("w83l785ts", 0x2d) }, | 266 | { I2C_BOARD_INFO("w83l785ts", 0x2d) }, |
| 267 | { I2C_BOARD_INFO("w83781d", 0x2d) }, | 267 | { I2C_BOARD_INFO("w83781d", 0x2d) }, |
| 268 | { I2C_BOARD_INFO("f75375", 0x2e) }, | ||
| 269 | { I2C_BOARD_INFO("adt7473", 0x2e) }, | 268 | { I2C_BOARD_INFO("adt7473", 0x2e) }, |
| 269 | { I2C_BOARD_INFO("f75375", 0x2e) }, | ||
| 270 | { I2C_BOARD_INFO("lm99", 0x4c) }, | 270 | { I2C_BOARD_INFO("lm99", 0x4c) }, |
| 271 | { } | 271 | { } |
| 272 | }; | 272 | }; |
diff --git a/drivers/gpu/drm/nouveau/nv50_evo.c b/drivers/gpu/drm/nouveau/nv50_evo.c index 14e24e906ee..0ea090f4244 100644 --- a/drivers/gpu/drm/nouveau/nv50_evo.c +++ b/drivers/gpu/drm/nouveau/nv50_evo.c | |||
| @@ -283,8 +283,7 @@ nv50_evo_create(struct drm_device *dev) | |||
| 283 | nv50_evo_channel_del(&dev_priv->evo); | 283 | nv50_evo_channel_del(&dev_priv->evo); |
| 284 | return ret; | 284 | return ret; |
| 285 | } | 285 | } |
| 286 | } else | 286 | } else { |
| 287 | if (dev_priv->chipset != 0x50) { | ||
| 288 | ret = nv50_evo_dmaobj_new(evo, 0x3d, NvEvoFB16, 0x70, 0x19, | 287 | ret = nv50_evo_dmaobj_new(evo, 0x3d, NvEvoFB16, 0x70, 0x19, |
| 289 | 0, 0xffffffff, 0x00010000); | 288 | 0, 0xffffffff, 0x00010000); |
| 290 | if (ret) { | 289 | if (ret) { |
diff --git a/drivers/gpu/drm/nouveau/nv50_graph.c b/drivers/gpu/drm/nouveau/nv50_graph.c index 2d7ea75a09d..37e21d2be95 100644 --- a/drivers/gpu/drm/nouveau/nv50_graph.c +++ b/drivers/gpu/drm/nouveau/nv50_graph.c | |||
| @@ -256,6 +256,7 @@ nv50_graph_destroy_context(struct nouveau_channel *chan) | |||
| 256 | struct drm_device *dev = chan->dev; | 256 | struct drm_device *dev = chan->dev; |
| 257 | struct drm_nouveau_private *dev_priv = dev->dev_private; | 257 | struct drm_nouveau_private *dev_priv = dev->dev_private; |
| 258 | struct nouveau_pgraph_engine *pgraph = &dev_priv->engine.graph; | 258 | struct nouveau_pgraph_engine *pgraph = &dev_priv->engine.graph; |
| 259 | struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo; | ||
| 259 | int i, hdr = (dev_priv->chipset == 0x50) ? 0x200 : 0x20; | 260 | int i, hdr = (dev_priv->chipset == 0x50) ? 0x200 : 0x20; |
| 260 | unsigned long flags; | 261 | unsigned long flags; |
| 261 | 262 | ||
| @@ -265,6 +266,7 @@ nv50_graph_destroy_context(struct nouveau_channel *chan) | |||
| 265 | return; | 266 | return; |
| 266 | 267 | ||
| 267 | spin_lock_irqsave(&dev_priv->context_switch_lock, flags); | 268 | spin_lock_irqsave(&dev_priv->context_switch_lock, flags); |
| 269 | pfifo->reassign(dev, false); | ||
| 268 | pgraph->fifo_access(dev, false); | 270 | pgraph->fifo_access(dev, false); |
| 269 | 271 | ||
| 270 | if (pgraph->channel(dev) == chan) | 272 | if (pgraph->channel(dev) == chan) |
| @@ -275,6 +277,7 @@ nv50_graph_destroy_context(struct nouveau_channel *chan) | |||
| 275 | dev_priv->engine.instmem.flush(dev); | 277 | dev_priv->engine.instmem.flush(dev); |
| 276 | 278 | ||
| 277 | pgraph->fifo_access(dev, true); | 279 | pgraph->fifo_access(dev, true); |
| 280 | pfifo->reassign(dev, true); | ||
| 278 | spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags); | 281 | spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags); |
| 279 | 282 | ||
| 280 | nouveau_gpuobj_ref(NULL, &chan->ramin_grctx); | 283 | nouveau_gpuobj_ref(NULL, &chan->ramin_grctx); |
diff --git a/drivers/gpu/drm/nouveau/nv50_vm.c b/drivers/gpu/drm/nouveau/nv50_vm.c index 38e523e1099..459ff08241e 100644 --- a/drivers/gpu/drm/nouveau/nv50_vm.c +++ b/drivers/gpu/drm/nouveau/nv50_vm.c | |||
| @@ -45,11 +45,6 @@ nv50_vm_map_pgt(struct nouveau_gpuobj *pgd, u32 pde, | |||
| 45 | } | 45 | } |
| 46 | 46 | ||
| 47 | if (phys & 1) { | 47 | if (phys & 1) { |
| 48 | if (dev_priv->vram_sys_base) { | ||
| 49 | phys += dev_priv->vram_sys_base; | ||
| 50 | phys |= 0x30; | ||
| 51 | } | ||
| 52 | |||
| 53 | if (coverage <= 32 * 1024 * 1024) | 48 | if (coverage <= 32 * 1024 * 1024) |
| 54 | phys |= 0x60; | 49 | phys |= 0x60; |
| 55 | else if (coverage <= 64 * 1024 * 1024) | 50 | else if (coverage <= 64 * 1024 * 1024) |
diff --git a/drivers/gpu/drm/nouveau/nvc0_graph.c b/drivers/gpu/drm/nouveau/nvc0_graph.c index e6ea7d83187..eb18a7e89f5 100644 --- a/drivers/gpu/drm/nouveau/nvc0_graph.c +++ b/drivers/gpu/drm/nouveau/nvc0_graph.c | |||
| @@ -31,6 +31,7 @@ | |||
| 31 | #include "nvc0_graph.h" | 31 | #include "nvc0_graph.h" |
| 32 | 32 | ||
| 33 | static void nvc0_graph_isr(struct drm_device *); | 33 | static void nvc0_graph_isr(struct drm_device *); |
| 34 | static void nvc0_runk140_isr(struct drm_device *); | ||
| 34 | static int nvc0_graph_unload_context_to(struct drm_device *dev, u64 chan); | 35 | static int nvc0_graph_unload_context_to(struct drm_device *dev, u64 chan); |
| 35 | 36 | ||
| 36 | void | 37 | void |
| @@ -281,6 +282,7 @@ nvc0_graph_destroy(struct drm_device *dev) | |||
| 281 | return; | 282 | return; |
| 282 | 283 | ||
| 283 | nouveau_irq_unregister(dev, 12); | 284 | nouveau_irq_unregister(dev, 12); |
| 285 | nouveau_irq_unregister(dev, 25); | ||
| 284 | 286 | ||
| 285 | nouveau_gpuobj_ref(NULL, &priv->unk4188b8); | 287 | nouveau_gpuobj_ref(NULL, &priv->unk4188b8); |
| 286 | nouveau_gpuobj_ref(NULL, &priv->unk4188b4); | 288 | nouveau_gpuobj_ref(NULL, &priv->unk4188b4); |
| @@ -390,6 +392,7 @@ nvc0_graph_create(struct drm_device *dev) | |||
| 390 | } | 392 | } |
| 391 | 393 | ||
| 392 | nouveau_irq_register(dev, 12, nvc0_graph_isr); | 394 | nouveau_irq_register(dev, 12, nvc0_graph_isr); |
| 395 | nouveau_irq_register(dev, 25, nvc0_runk140_isr); | ||
| 393 | NVOBJ_CLASS(dev, 0x902d, GR); /* 2D */ | 396 | NVOBJ_CLASS(dev, 0x902d, GR); /* 2D */ |
| 394 | NVOBJ_CLASS(dev, 0x9039, GR); /* M2MF */ | 397 | NVOBJ_CLASS(dev, 0x9039, GR); /* M2MF */ |
| 395 | NVOBJ_CLASS(dev, 0x9097, GR); /* 3D */ | 398 | NVOBJ_CLASS(dev, 0x9097, GR); /* 3D */ |
| @@ -512,8 +515,8 @@ nvc0_graph_init_gpc_1(struct drm_device *dev) | |||
| 512 | nv_wr32(dev, TP_UNIT(gpc, tp, 0x224), 0xc0000000); | 515 | nv_wr32(dev, TP_UNIT(gpc, tp, 0x224), 0xc0000000); |
| 513 | nv_wr32(dev, TP_UNIT(gpc, tp, 0x48c), 0xc0000000); | 516 | nv_wr32(dev, TP_UNIT(gpc, tp, 0x48c), 0xc0000000); |
| 514 | nv_wr32(dev, TP_UNIT(gpc, tp, 0x084), 0xc0000000); | 517 | nv_wr32(dev, TP_UNIT(gpc, tp, 0x084), 0xc0000000); |
| 515 | nv_wr32(dev, TP_UNIT(gpc, tp, 0xe44), 0x001ffffe); | 518 | nv_wr32(dev, TP_UNIT(gpc, tp, 0x644), 0x001ffffe); |
| 516 | nv_wr32(dev, TP_UNIT(gpc, tp, 0xe4c), 0x0000000f); | 519 | nv_wr32(dev, TP_UNIT(gpc, tp, 0x64c), 0x0000000f); |
| 517 | } | 520 | } |
| 518 | nv_wr32(dev, GPC_UNIT(gpc, 0x2c90), 0xffffffff); | 521 | nv_wr32(dev, GPC_UNIT(gpc, 0x2c90), 0xffffffff); |
| 519 | nv_wr32(dev, GPC_UNIT(gpc, 0x2c94), 0xffffffff); | 522 | nv_wr32(dev, GPC_UNIT(gpc, 0x2c94), 0xffffffff); |
| @@ -777,3 +780,19 @@ nvc0_graph_isr(struct drm_device *dev) | |||
| 777 | 780 | ||
| 778 | nv_wr32(dev, 0x400500, 0x00010001); | 781 | nv_wr32(dev, 0x400500, 0x00010001); |
| 779 | } | 782 | } |
| 783 | |||
| 784 | static void | ||
| 785 | nvc0_runk140_isr(struct drm_device *dev) | ||
| 786 | { | ||
| 787 | u32 units = nv_rd32(dev, 0x00017c) & 0x1f; | ||
| 788 | |||
| 789 | while (units) { | ||
| 790 | u32 unit = ffs(units) - 1; | ||
| 791 | u32 reg = 0x140000 + unit * 0x2000; | ||
| 792 | u32 st0 = nv_mask(dev, reg + 0x1020, 0, 0); | ||
| 793 | u32 st1 = nv_mask(dev, reg + 0x1420, 0, 0); | ||
| 794 | |||
| 795 | NV_INFO(dev, "PRUNK140: %d 0x%08x 0x%08x\n", unit, st0, st1); | ||
| 796 | units &= ~(1 << unit); | ||
| 797 | } | ||
| 798 | } | ||
diff --git a/drivers/gpu/drm/nouveau/nvc0_grctx.c b/drivers/gpu/drm/nouveau/nvc0_grctx.c index b9e68b2d30a..f880ff776db 100644 --- a/drivers/gpu/drm/nouveau/nvc0_grctx.c +++ b/drivers/gpu/drm/nouveau/nvc0_grctx.c | |||
| @@ -1830,7 +1830,7 @@ nvc0_grctx_generate(struct nouveau_channel *chan) | |||
| 1830 | 1830 | ||
| 1831 | for (tp = 0, id = 0; tp < 4; tp++) { | 1831 | for (tp = 0, id = 0; tp < 4; tp++) { |
| 1832 | for (gpc = 0; gpc < priv->gpc_nr; gpc++) { | 1832 | for (gpc = 0; gpc < priv->gpc_nr; gpc++) { |
| 1833 | if (tp <= priv->tp_nr[gpc]) { | 1833 | if (tp < priv->tp_nr[gpc]) { |
| 1834 | nv_wr32(dev, TP_UNIT(gpc, tp, 0x698), id); | 1834 | nv_wr32(dev, TP_UNIT(gpc, tp, 0x698), id); |
| 1835 | nv_wr32(dev, TP_UNIT(gpc, tp, 0x4e8), id); | 1835 | nv_wr32(dev, TP_UNIT(gpc, tp, 0x4e8), id); |
| 1836 | nv_wr32(dev, GPC_UNIT(gpc, 0x0c10 + tp * 4), id); | 1836 | nv_wr32(dev, GPC_UNIT(gpc, 0x0c10 + tp * 4), id); |
diff --git a/drivers/gpu/drm/radeon/atombios_crtc.c b/drivers/gpu/drm/radeon/atombios_crtc.c index b0ab185b86f..b1537000a10 100644 --- a/drivers/gpu/drm/radeon/atombios_crtc.c +++ b/drivers/gpu/drm/radeon/atombios_crtc.c | |||
| @@ -555,6 +555,7 @@ static u32 atombios_adjust_pll(struct drm_crtc *crtc, | |||
| 555 | dp_clock = dig_connector->dp_clock; | 555 | dp_clock = dig_connector->dp_clock; |
| 556 | } | 556 | } |
| 557 | } | 557 | } |
| 558 | /* this might work properly with the new pll algo */ | ||
| 558 | #if 0 /* doesn't work properly on some laptops */ | 559 | #if 0 /* doesn't work properly on some laptops */ |
| 559 | /* use recommended ref_div for ss */ | 560 | /* use recommended ref_div for ss */ |
| 560 | if (radeon_encoder->devices & (ATOM_DEVICE_LCD_SUPPORT)) { | 561 | if (radeon_encoder->devices & (ATOM_DEVICE_LCD_SUPPORT)) { |
| @@ -572,6 +573,11 @@ static u32 atombios_adjust_pll(struct drm_crtc *crtc, | |||
| 572 | adjusted_clock = mode->clock * 2; | 573 | adjusted_clock = mode->clock * 2; |
| 573 | if (radeon_encoder->active_device & (ATOM_DEVICE_TV_SUPPORT)) | 574 | if (radeon_encoder->active_device & (ATOM_DEVICE_TV_SUPPORT)) |
| 574 | pll->flags |= RADEON_PLL_PREFER_CLOSEST_LOWER; | 575 | pll->flags |= RADEON_PLL_PREFER_CLOSEST_LOWER; |
| 576 | /* rv515 needs more testing with this option */ | ||
| 577 | if (rdev->family != CHIP_RV515) { | ||
| 578 | if (radeon_encoder->devices & (ATOM_DEVICE_LCD_SUPPORT)) | ||
| 579 | pll->flags |= RADEON_PLL_IS_LCD; | ||
| 580 | } | ||
| 575 | } else { | 581 | } else { |
| 576 | if (encoder->encoder_type != DRM_MODE_ENCODER_DAC) | 582 | if (encoder->encoder_type != DRM_MODE_ENCODER_DAC) |
| 577 | pll->flags |= RADEON_PLL_NO_ODD_POST_DIV; | 583 | pll->flags |= RADEON_PLL_NO_ODD_POST_DIV; |
| @@ -606,14 +612,9 @@ static u32 atombios_adjust_pll(struct drm_crtc *crtc, | |||
| 606 | args.v1.usPixelClock = cpu_to_le16(mode->clock / 10); | 612 | args.v1.usPixelClock = cpu_to_le16(mode->clock / 10); |
| 607 | args.v1.ucTransmitterID = radeon_encoder->encoder_id; | 613 | args.v1.ucTransmitterID = radeon_encoder->encoder_id; |
| 608 | args.v1.ucEncodeMode = encoder_mode; | 614 | args.v1.ucEncodeMode = encoder_mode; |
| 609 | if (encoder_mode == ATOM_ENCODER_MODE_DP) { | 615 | if (ss_enabled) |
| 610 | if (ss_enabled) | ||
| 611 | args.v1.ucConfig |= | ||
| 612 | ADJUST_DISPLAY_CONFIG_SS_ENABLE; | ||
| 613 | } else if (encoder_mode == ATOM_ENCODER_MODE_LVDS) { | ||
| 614 | args.v1.ucConfig |= | 616 | args.v1.ucConfig |= |
| 615 | ADJUST_DISPLAY_CONFIG_SS_ENABLE; | 617 | ADJUST_DISPLAY_CONFIG_SS_ENABLE; |
| 616 | } | ||
| 617 | 618 | ||
| 618 | atom_execute_table(rdev->mode_info.atom_context, | 619 | atom_execute_table(rdev->mode_info.atom_context, |
| 619 | index, (uint32_t *)&args); | 620 | index, (uint32_t *)&args); |
| @@ -624,12 +625,12 @@ static u32 atombios_adjust_pll(struct drm_crtc *crtc, | |||
| 624 | args.v3.sInput.ucTransmitterID = radeon_encoder->encoder_id; | 625 | args.v3.sInput.ucTransmitterID = radeon_encoder->encoder_id; |
| 625 | args.v3.sInput.ucEncodeMode = encoder_mode; | 626 | args.v3.sInput.ucEncodeMode = encoder_mode; |
| 626 | args.v3.sInput.ucDispPllConfig = 0; | 627 | args.v3.sInput.ucDispPllConfig = 0; |
| 628 | if (ss_enabled) | ||
| 629 | args.v3.sInput.ucDispPllConfig |= | ||
| 630 | DISPPLL_CONFIG_SS_ENABLE; | ||
| 627 | if (radeon_encoder->devices & (ATOM_DEVICE_DFP_SUPPORT)) { | 631 | if (radeon_encoder->devices & (ATOM_DEVICE_DFP_SUPPORT)) { |
| 628 | struct radeon_encoder_atom_dig *dig = radeon_encoder->enc_priv; | 632 | struct radeon_encoder_atom_dig *dig = radeon_encoder->enc_priv; |
| 629 | if (encoder_mode == ATOM_ENCODER_MODE_DP) { | 633 | if (encoder_mode == ATOM_ENCODER_MODE_DP) { |
| 630 | if (ss_enabled) | ||
| 631 | args.v3.sInput.ucDispPllConfig |= | ||
| 632 | DISPPLL_CONFIG_SS_ENABLE; | ||
| 633 | args.v3.sInput.ucDispPllConfig |= | 634 | args.v3.sInput.ucDispPllConfig |= |
| 634 | DISPPLL_CONFIG_COHERENT_MODE; | 635 | DISPPLL_CONFIG_COHERENT_MODE; |
| 635 | /* 16200 or 27000 */ | 636 | /* 16200 or 27000 */ |
| @@ -649,18 +650,11 @@ static u32 atombios_adjust_pll(struct drm_crtc *crtc, | |||
| 649 | } | 650 | } |
| 650 | } else if (radeon_encoder->devices & (ATOM_DEVICE_LCD_SUPPORT)) { | 651 | } else if (radeon_encoder->devices & (ATOM_DEVICE_LCD_SUPPORT)) { |
| 651 | if (encoder_mode == ATOM_ENCODER_MODE_DP) { | 652 | if (encoder_mode == ATOM_ENCODER_MODE_DP) { |
| 652 | if (ss_enabled) | ||
| 653 | args.v3.sInput.ucDispPllConfig |= | ||
| 654 | DISPPLL_CONFIG_SS_ENABLE; | ||
| 655 | args.v3.sInput.ucDispPllConfig |= | 653 | args.v3.sInput.ucDispPllConfig |= |
| 656 | DISPPLL_CONFIG_COHERENT_MODE; | 654 | DISPPLL_CONFIG_COHERENT_MODE; |
| 657 | /* 16200 or 27000 */ | 655 | /* 16200 or 27000 */ |
| 658 | args.v3.sInput.usPixelClock = cpu_to_le16(dp_clock / 10); | 656 | args.v3.sInput.usPixelClock = cpu_to_le16(dp_clock / 10); |
| 659 | } else if (encoder_mode == ATOM_ENCODER_MODE_LVDS) { | 657 | } else if (encoder_mode != ATOM_ENCODER_MODE_LVDS) { |
| 660 | if (ss_enabled) | ||
| 661 | args.v3.sInput.ucDispPllConfig |= | ||
| 662 | DISPPLL_CONFIG_SS_ENABLE; | ||
| 663 | } else { | ||
| 664 | if (mode->clock > 165000) | 658 | if (mode->clock > 165000) |
| 665 | args.v3.sInput.ucDispPllConfig |= | 659 | args.v3.sInput.ucDispPllConfig |= |
| 666 | DISPPLL_CONFIG_DUAL_LINK; | 660 | DISPPLL_CONFIG_DUAL_LINK; |
| @@ -963,8 +957,16 @@ static void atombios_crtc_set_pll(struct drm_crtc *crtc, struct drm_display_mode | |||
| 963 | /* adjust pixel clock as needed */ | 957 | /* adjust pixel clock as needed */ |
| 964 | adjusted_clock = atombios_adjust_pll(crtc, mode, pll, ss_enabled, &ss); | 958 | adjusted_clock = atombios_adjust_pll(crtc, mode, pll, ss_enabled, &ss); |
| 965 | 959 | ||
| 966 | radeon_compute_pll(pll, adjusted_clock, &pll_clock, &fb_div, &frac_fb_div, | 960 | /* rv515 seems happier with the old algo */ |
| 967 | &ref_div, &post_div); | 961 | if (rdev->family == CHIP_RV515) |
| 962 | radeon_compute_pll_legacy(pll, adjusted_clock, &pll_clock, &fb_div, &frac_fb_div, | ||
| 963 | &ref_div, &post_div); | ||
| 964 | else if (ASIC_IS_AVIVO(rdev)) | ||
| 965 | radeon_compute_pll_avivo(pll, adjusted_clock, &pll_clock, &fb_div, &frac_fb_div, | ||
| 966 | &ref_div, &post_div); | ||
| 967 | else | ||
| 968 | radeon_compute_pll_legacy(pll, adjusted_clock, &pll_clock, &fb_div, &frac_fb_div, | ||
| 969 | &ref_div, &post_div); | ||
| 968 | 970 | ||
| 969 | atombios_crtc_program_ss(crtc, ATOM_DISABLE, radeon_crtc->pll_id, &ss); | 971 | atombios_crtc_program_ss(crtc, ATOM_DISABLE, radeon_crtc->pll_id, &ss); |
| 970 | 972 | ||
| @@ -1006,6 +1008,7 @@ static int evergreen_crtc_do_set_base(struct drm_crtc *crtc, | |||
| 1006 | struct radeon_bo *rbo; | 1008 | struct radeon_bo *rbo; |
| 1007 | uint64_t fb_location; | 1009 | uint64_t fb_location; |
| 1008 | uint32_t fb_format, fb_pitch_pixels, tiling_flags; | 1010 | uint32_t fb_format, fb_pitch_pixels, tiling_flags; |
| 1011 | u32 fb_swap = EVERGREEN_GRPH_ENDIAN_SWAP(EVERGREEN_GRPH_ENDIAN_NONE); | ||
| 1009 | int r; | 1012 | int r; |
| 1010 | 1013 | ||
| 1011 | /* no fb bound */ | 1014 | /* no fb bound */ |
| @@ -1057,11 +1060,17 @@ static int evergreen_crtc_do_set_base(struct drm_crtc *crtc, | |||
| 1057 | case 16: | 1060 | case 16: |
| 1058 | fb_format = (EVERGREEN_GRPH_DEPTH(EVERGREEN_GRPH_DEPTH_16BPP) | | 1061 | fb_format = (EVERGREEN_GRPH_DEPTH(EVERGREEN_GRPH_DEPTH_16BPP) | |
| 1059 | EVERGREEN_GRPH_FORMAT(EVERGREEN_GRPH_FORMAT_ARGB565)); | 1062 | EVERGREEN_GRPH_FORMAT(EVERGREEN_GRPH_FORMAT_ARGB565)); |
| 1063 | #ifdef __BIG_ENDIAN | ||
| 1064 | fb_swap = EVERGREEN_GRPH_ENDIAN_SWAP(EVERGREEN_GRPH_ENDIAN_8IN16); | ||
| 1065 | #endif | ||
| 1060 | break; | 1066 | break; |
| 1061 | case 24: | 1067 | case 24: |
| 1062 | case 32: | 1068 | case 32: |
| 1063 | fb_format = (EVERGREEN_GRPH_DEPTH(EVERGREEN_GRPH_DEPTH_32BPP) | | 1069 | fb_format = (EVERGREEN_GRPH_DEPTH(EVERGREEN_GRPH_DEPTH_32BPP) | |
| 1064 | EVERGREEN_GRPH_FORMAT(EVERGREEN_GRPH_FORMAT_ARGB8888)); | 1070 | EVERGREEN_GRPH_FORMAT(EVERGREEN_GRPH_FORMAT_ARGB8888)); |
| 1071 | #ifdef __BIG_ENDIAN | ||
| 1072 | fb_swap = EVERGREEN_GRPH_ENDIAN_SWAP(EVERGREEN_GRPH_ENDIAN_8IN32); | ||
| 1073 | #endif | ||
| 1065 | break; | 1074 | break; |
| 1066 | default: | 1075 | default: |
| 1067 | DRM_ERROR("Unsupported screen depth %d\n", | 1076 | DRM_ERROR("Unsupported screen depth %d\n", |
| @@ -1106,6 +1115,7 @@ static int evergreen_crtc_do_set_base(struct drm_crtc *crtc, | |||
| 1106 | WREG32(EVERGREEN_GRPH_SECONDARY_SURFACE_ADDRESS + radeon_crtc->crtc_offset, | 1115 | WREG32(EVERGREEN_GRPH_SECONDARY_SURFACE_ADDRESS + radeon_crtc->crtc_offset, |
| 1107 | (u32) fb_location & EVERGREEN_GRPH_SURFACE_ADDRESS_MASK); | 1116 | (u32) fb_location & EVERGREEN_GRPH_SURFACE_ADDRESS_MASK); |
| 1108 | WREG32(EVERGREEN_GRPH_CONTROL + radeon_crtc->crtc_offset, fb_format); | 1117 | WREG32(EVERGREEN_GRPH_CONTROL + radeon_crtc->crtc_offset, fb_format); |
| 1118 | WREG32(EVERGREEN_GRPH_SWAP_CONTROL + radeon_crtc->crtc_offset, fb_swap); | ||
| 1109 | 1119 | ||
| 1110 | WREG32(EVERGREEN_GRPH_SURFACE_OFFSET_X + radeon_crtc->crtc_offset, 0); | 1120 | WREG32(EVERGREEN_GRPH_SURFACE_OFFSET_X + radeon_crtc->crtc_offset, 0); |
| 1111 | WREG32(EVERGREEN_GRPH_SURFACE_OFFSET_Y + radeon_crtc->crtc_offset, 0); | 1121 | WREG32(EVERGREEN_GRPH_SURFACE_OFFSET_Y + radeon_crtc->crtc_offset, 0); |
| @@ -1162,6 +1172,7 @@ static int avivo_crtc_do_set_base(struct drm_crtc *crtc, | |||
| 1162 | struct drm_framebuffer *target_fb; | 1172 | struct drm_framebuffer *target_fb; |
| 1163 | uint64_t fb_location; | 1173 | uint64_t fb_location; |
| 1164 | uint32_t fb_format, fb_pitch_pixels, tiling_flags; | 1174 | uint32_t fb_format, fb_pitch_pixels, tiling_flags; |
| 1175 | u32 fb_swap = R600_D1GRPH_SWAP_ENDIAN_NONE; | ||
| 1165 | int r; | 1176 | int r; |
| 1166 | 1177 | ||
| 1167 | /* no fb bound */ | 1178 | /* no fb bound */ |
| @@ -1215,12 +1226,18 @@ static int avivo_crtc_do_set_base(struct drm_crtc *crtc, | |||
| 1215 | fb_format = | 1226 | fb_format = |
| 1216 | AVIVO_D1GRPH_CONTROL_DEPTH_16BPP | | 1227 | AVIVO_D1GRPH_CONTROL_DEPTH_16BPP | |
| 1217 | AVIVO_D1GRPH_CONTROL_16BPP_RGB565; | 1228 | AVIVO_D1GRPH_CONTROL_16BPP_RGB565; |
| 1229 | #ifdef __BIG_ENDIAN | ||
| 1230 | fb_swap = R600_D1GRPH_SWAP_ENDIAN_16BIT; | ||
| 1231 | #endif | ||
| 1218 | break; | 1232 | break; |
| 1219 | case 24: | 1233 | case 24: |
| 1220 | case 32: | 1234 | case 32: |
| 1221 | fb_format = | 1235 | fb_format = |
| 1222 | AVIVO_D1GRPH_CONTROL_DEPTH_32BPP | | 1236 | AVIVO_D1GRPH_CONTROL_DEPTH_32BPP | |
| 1223 | AVIVO_D1GRPH_CONTROL_32BPP_ARGB8888; | 1237 | AVIVO_D1GRPH_CONTROL_32BPP_ARGB8888; |
| 1238 | #ifdef __BIG_ENDIAN | ||
| 1239 | fb_swap = R600_D1GRPH_SWAP_ENDIAN_32BIT; | ||
| 1240 | #endif | ||
| 1224 | break; | 1241 | break; |
| 1225 | default: | 1242 | default: |
| 1226 | DRM_ERROR("Unsupported screen depth %d\n", | 1243 | DRM_ERROR("Unsupported screen depth %d\n", |
| @@ -1260,6 +1277,8 @@ static int avivo_crtc_do_set_base(struct drm_crtc *crtc, | |||
| 1260 | WREG32(AVIVO_D1GRPH_SECONDARY_SURFACE_ADDRESS + | 1277 | WREG32(AVIVO_D1GRPH_SECONDARY_SURFACE_ADDRESS + |
| 1261 | radeon_crtc->crtc_offset, (u32) fb_location); | 1278 | radeon_crtc->crtc_offset, (u32) fb_location); |
| 1262 | WREG32(AVIVO_D1GRPH_CONTROL + radeon_crtc->crtc_offset, fb_format); | 1279 | WREG32(AVIVO_D1GRPH_CONTROL + radeon_crtc->crtc_offset, fb_format); |
| 1280 | if (rdev->family >= CHIP_R600) | ||
| 1281 | WREG32(R600_D1GRPH_SWAP_CONTROL + radeon_crtc->crtc_offset, fb_swap); | ||
| 1263 | 1282 | ||
| 1264 | WREG32(AVIVO_D1GRPH_SURFACE_OFFSET_X + radeon_crtc->crtc_offset, 0); | 1283 | WREG32(AVIVO_D1GRPH_SURFACE_OFFSET_X + radeon_crtc->crtc_offset, 0); |
| 1265 | WREG32(AVIVO_D1GRPH_SURFACE_OFFSET_Y + radeon_crtc->crtc_offset, 0); | 1284 | WREG32(AVIVO_D1GRPH_SURFACE_OFFSET_Y + radeon_crtc->crtc_offset, 0); |
diff --git a/drivers/gpu/drm/radeon/atombios_dp.c b/drivers/gpu/drm/radeon/atombios_dp.c index 4e7778d44b8..695de9a3850 100644 --- a/drivers/gpu/drm/radeon/atombios_dp.c +++ b/drivers/gpu/drm/radeon/atombios_dp.c | |||
| @@ -187,9 +187,9 @@ static int dp_link_clock_for_mode_clock(u8 dpcd[DP_DPCD_SIZE], int mode_clock) | |||
| 187 | int dp_mode_valid(u8 dpcd[DP_DPCD_SIZE], int mode_clock) | 187 | int dp_mode_valid(u8 dpcd[DP_DPCD_SIZE], int mode_clock) |
| 188 | { | 188 | { |
| 189 | int lanes = dp_lanes_for_mode_clock(dpcd, mode_clock); | 189 | int lanes = dp_lanes_for_mode_clock(dpcd, mode_clock); |
| 190 | int bw = dp_lanes_for_mode_clock(dpcd, mode_clock); | 190 | int dp_clock = dp_link_clock_for_mode_clock(dpcd, mode_clock); |
| 191 | 191 | ||
| 192 | if ((lanes == 0) || (bw == 0)) | 192 | if ((lanes == 0) || (dp_clock == 0)) |
| 193 | return MODE_CLOCK_HIGH; | 193 | return MODE_CLOCK_HIGH; |
| 194 | 194 | ||
| 195 | return MODE_OK; | 195 | return MODE_OK; |
diff --git a/drivers/gpu/drm/radeon/evergreen.c b/drivers/gpu/drm/radeon/evergreen.c index a8973acb398..ffdc8332b76 100644 --- a/drivers/gpu/drm/radeon/evergreen.c +++ b/drivers/gpu/drm/radeon/evergreen.c | |||
| @@ -97,26 +97,29 @@ u32 evergreen_page_flip(struct radeon_device *rdev, int crtc_id, u64 crtc_base) | |||
| 97 | } | 97 | } |
| 98 | 98 | ||
| 99 | /* get temperature in millidegrees */ | 99 | /* get temperature in millidegrees */ |
| 100 | u32 evergreen_get_temp(struct radeon_device *rdev) | 100 | int evergreen_get_temp(struct radeon_device *rdev) |
| 101 | { | 101 | { |
| 102 | u32 temp = (RREG32(CG_MULT_THERMAL_STATUS) & ASIC_T_MASK) >> | 102 | u32 temp = (RREG32(CG_MULT_THERMAL_STATUS) & ASIC_T_MASK) >> |
| 103 | ASIC_T_SHIFT; | 103 | ASIC_T_SHIFT; |
| 104 | u32 actual_temp = 0; | 104 | u32 actual_temp = 0; |
| 105 | 105 | ||
| 106 | if ((temp >> 10) & 1) | 106 | if (temp & 0x400) |
| 107 | actual_temp = 0; | 107 | actual_temp = -256; |
| 108 | else if ((temp >> 9) & 1) | 108 | else if (temp & 0x200) |
| 109 | actual_temp = 255; | 109 | actual_temp = 255; |
| 110 | else | 110 | else if (temp & 0x100) { |
| 111 | actual_temp = (temp >> 1) & 0xff; | 111 | actual_temp = temp & 0x1ff; |
| 112 | actual_temp |= ~0x1ff; | ||
| 113 | } else | ||
| 114 | actual_temp = temp & 0xff; | ||
| 112 | 115 | ||
| 113 | return actual_temp * 1000; | 116 | return (actual_temp * 1000) / 2; |
| 114 | } | 117 | } |
| 115 | 118 | ||
| 116 | u32 sumo_get_temp(struct radeon_device *rdev) | 119 | int sumo_get_temp(struct radeon_device *rdev) |
| 117 | { | 120 | { |
| 118 | u32 temp = RREG32(CG_THERMAL_STATUS) & 0xff; | 121 | u32 temp = RREG32(CG_THERMAL_STATUS) & 0xff; |
| 119 | u32 actual_temp = (temp >> 1) & 0xff; | 122 | int actual_temp = temp - 49; |
| 120 | 123 | ||
| 121 | return actual_temp * 1000; | 124 | return actual_temp * 1000; |
| 122 | } | 125 | } |
| @@ -1182,6 +1185,18 @@ static void evergreen_mc_program(struct radeon_device *rdev) | |||
| 1182 | /* | 1185 | /* |
| 1183 | * CP. | 1186 | * CP. |
| 1184 | */ | 1187 | */ |
| 1188 | void evergreen_ring_ib_execute(struct radeon_device *rdev, struct radeon_ib *ib) | ||
| 1189 | { | ||
| 1190 | /* set to DX10/11 mode */ | ||
| 1191 | radeon_ring_write(rdev, PACKET3(PACKET3_MODE_CONTROL, 0)); | ||
| 1192 | radeon_ring_write(rdev, 1); | ||
| 1193 | /* FIXME: implement */ | ||
| 1194 | radeon_ring_write(rdev, PACKET3(PACKET3_INDIRECT_BUFFER, 2)); | ||
| 1195 | radeon_ring_write(rdev, ib->gpu_addr & 0xFFFFFFFC); | ||
| 1196 | radeon_ring_write(rdev, upper_32_bits(ib->gpu_addr) & 0xFF); | ||
| 1197 | radeon_ring_write(rdev, ib->length_dw); | ||
| 1198 | } | ||
| 1199 | |||
| 1185 | 1200 | ||
| 1186 | static int evergreen_cp_load_microcode(struct radeon_device *rdev) | 1201 | static int evergreen_cp_load_microcode(struct radeon_device *rdev) |
| 1187 | { | 1202 | { |
| @@ -1233,7 +1248,7 @@ static int evergreen_cp_start(struct radeon_device *rdev) | |||
| 1233 | cp_me = 0xff; | 1248 | cp_me = 0xff; |
| 1234 | WREG32(CP_ME_CNTL, cp_me); | 1249 | WREG32(CP_ME_CNTL, cp_me); |
| 1235 | 1250 | ||
| 1236 | r = radeon_ring_lock(rdev, evergreen_default_size + 15); | 1251 | r = radeon_ring_lock(rdev, evergreen_default_size + 19); |
| 1237 | if (r) { | 1252 | if (r) { |
| 1238 | DRM_ERROR("radeon: cp failed to lock ring (%d).\n", r); | 1253 | DRM_ERROR("radeon: cp failed to lock ring (%d).\n", r); |
| 1239 | return r; | 1254 | return r; |
| @@ -1266,6 +1281,11 @@ static int evergreen_cp_start(struct radeon_device *rdev) | |||
| 1266 | radeon_ring_write(rdev, 0xffffffff); | 1281 | radeon_ring_write(rdev, 0xffffffff); |
| 1267 | radeon_ring_write(rdev, 0xffffffff); | 1282 | radeon_ring_write(rdev, 0xffffffff); |
| 1268 | 1283 | ||
| 1284 | radeon_ring_write(rdev, 0xc0026900); | ||
| 1285 | radeon_ring_write(rdev, 0x00000316); | ||
| 1286 | radeon_ring_write(rdev, 0x0000000e); /* VGT_VERTEX_REUSE_BLOCK_CNTL */ | ||
| 1287 | radeon_ring_write(rdev, 0x00000010); /* */ | ||
| 1288 | |||
| 1269 | radeon_ring_unlock_commit(rdev); | 1289 | radeon_ring_unlock_commit(rdev); |
| 1270 | 1290 | ||
| 1271 | return 0; | 1291 | return 0; |
| @@ -2072,6 +2092,7 @@ static void evergreen_gpu_init(struct radeon_device *rdev) | |||
| 2072 | WREG32(VGT_CACHE_INVALIDATION, vgt_cache_invalidation); | 2092 | WREG32(VGT_CACHE_INVALIDATION, vgt_cache_invalidation); |
| 2073 | 2093 | ||
| 2074 | WREG32(VGT_GS_VERTEX_REUSE, 16); | 2094 | WREG32(VGT_GS_VERTEX_REUSE, 16); |
| 2095 | WREG32(PA_SU_LINE_STIPPLE_VALUE, 0); | ||
| 2075 | WREG32(PA_SC_LINE_STIPPLE_STATE, 0); | 2096 | WREG32(PA_SC_LINE_STIPPLE_STATE, 0); |
| 2076 | 2097 | ||
| 2077 | WREG32(VGT_VERTEX_REUSE_BLOCK_CNTL, 14); | 2098 | WREG32(VGT_VERTEX_REUSE_BLOCK_CNTL, 14); |
| @@ -2201,6 +2222,9 @@ static int evergreen_gpu_soft_reset(struct radeon_device *rdev) | |||
| 2201 | struct evergreen_mc_save save; | 2222 | struct evergreen_mc_save save; |
| 2202 | u32 grbm_reset = 0; | 2223 | u32 grbm_reset = 0; |
| 2203 | 2224 | ||
| 2225 | if (!(RREG32(GRBM_STATUS) & GUI_ACTIVE)) | ||
| 2226 | return 0; | ||
| 2227 | |||
| 2204 | dev_info(rdev->dev, "GPU softreset \n"); | 2228 | dev_info(rdev->dev, "GPU softreset \n"); |
| 2205 | dev_info(rdev->dev, " GRBM_STATUS=0x%08X\n", | 2229 | dev_info(rdev->dev, " GRBM_STATUS=0x%08X\n", |
| 2206 | RREG32(GRBM_STATUS)); | 2230 | RREG32(GRBM_STATUS)); |
diff --git a/drivers/gpu/drm/radeon/evergreen_blit_kms.c b/drivers/gpu/drm/radeon/evergreen_blit_kms.c index b758dc7f2f2..a1ba4b3053d 100644 --- a/drivers/gpu/drm/radeon/evergreen_blit_kms.c +++ b/drivers/gpu/drm/radeon/evergreen_blit_kms.c | |||
| @@ -232,7 +232,7 @@ draw_auto(struct radeon_device *rdev) | |||
| 232 | 232 | ||
| 233 | } | 233 | } |
| 234 | 234 | ||
| 235 | /* emits 30 */ | 235 | /* emits 36 */ |
| 236 | static void | 236 | static void |
| 237 | set_default_state(struct radeon_device *rdev) | 237 | set_default_state(struct radeon_device *rdev) |
| 238 | { | 238 | { |
| @@ -245,6 +245,8 @@ set_default_state(struct radeon_device *rdev) | |||
| 245 | int num_hs_threads, num_ls_threads; | 245 | int num_hs_threads, num_ls_threads; |
| 246 | int num_ps_stack_entries, num_vs_stack_entries, num_gs_stack_entries, num_es_stack_entries; | 246 | int num_ps_stack_entries, num_vs_stack_entries, num_gs_stack_entries, num_es_stack_entries; |
| 247 | int num_hs_stack_entries, num_ls_stack_entries; | 247 | int num_hs_stack_entries, num_ls_stack_entries; |
| 248 | u64 gpu_addr; | ||
| 249 | int dwords; | ||
| 248 | 250 | ||
| 249 | switch (rdev->family) { | 251 | switch (rdev->family) { |
| 250 | case CHIP_CEDAR: | 252 | case CHIP_CEDAR: |
| @@ -497,6 +499,18 @@ set_default_state(struct radeon_device *rdev) | |||
| 497 | radeon_ring_write(rdev, 0x00000000); | 499 | radeon_ring_write(rdev, 0x00000000); |
| 498 | radeon_ring_write(rdev, 0x00000000); | 500 | radeon_ring_write(rdev, 0x00000000); |
| 499 | 501 | ||
| 502 | /* set to DX10/11 mode */ | ||
| 503 | radeon_ring_write(rdev, PACKET3(PACKET3_MODE_CONTROL, 0)); | ||
| 504 | radeon_ring_write(rdev, 1); | ||
| 505 | |||
| 506 | /* emit an IB pointing at default state */ | ||
| 507 | dwords = ALIGN(rdev->r600_blit.state_len, 0x10); | ||
| 508 | gpu_addr = rdev->r600_blit.shader_gpu_addr + rdev->r600_blit.state_offset; | ||
| 509 | radeon_ring_write(rdev, PACKET3(PACKET3_INDIRECT_BUFFER, 2)); | ||
| 510 | radeon_ring_write(rdev, gpu_addr & 0xFFFFFFFC); | ||
| 511 | radeon_ring_write(rdev, upper_32_bits(gpu_addr) & 0xFF); | ||
| 512 | radeon_ring_write(rdev, dwords); | ||
| 513 | |||
| 500 | } | 514 | } |
| 501 | 515 | ||
| 502 | static inline uint32_t i2f(uint32_t input) | 516 | static inline uint32_t i2f(uint32_t input) |
| @@ -527,8 +541,10 @@ static inline uint32_t i2f(uint32_t input) | |||
| 527 | int evergreen_blit_init(struct radeon_device *rdev) | 541 | int evergreen_blit_init(struct radeon_device *rdev) |
| 528 | { | 542 | { |
| 529 | u32 obj_size; | 543 | u32 obj_size; |
| 530 | int r; | 544 | int r, dwords; |
| 531 | void *ptr; | 545 | void *ptr; |
| 546 | u32 packet2s[16]; | ||
| 547 | int num_packet2s = 0; | ||
| 532 | 548 | ||
| 533 | /* pin copy shader into vram if already initialized */ | 549 | /* pin copy shader into vram if already initialized */ |
| 534 | if (rdev->r600_blit.shader_obj) | 550 | if (rdev->r600_blit.shader_obj) |
| @@ -536,8 +552,17 @@ int evergreen_blit_init(struct radeon_device *rdev) | |||
| 536 | 552 | ||
| 537 | mutex_init(&rdev->r600_blit.mutex); | 553 | mutex_init(&rdev->r600_blit.mutex); |
| 538 | rdev->r600_blit.state_offset = 0; | 554 | rdev->r600_blit.state_offset = 0; |
| 539 | rdev->r600_blit.state_len = 0; | 555 | |
| 540 | obj_size = 0; | 556 | rdev->r600_blit.state_len = evergreen_default_size; |
| 557 | |||
| 558 | dwords = rdev->r600_blit.state_len; | ||
| 559 | while (dwords & 0xf) { | ||
| 560 | packet2s[num_packet2s++] = PACKET2(0); | ||
| 561 | dwords++; | ||
| 562 | } | ||
| 563 | |||
| 564 | obj_size = dwords * 4; | ||
| 565 | obj_size = ALIGN(obj_size, 256); | ||
| 541 | 566 | ||
| 542 | rdev->r600_blit.vs_offset = obj_size; | 567 | rdev->r600_blit.vs_offset = obj_size; |
| 543 | obj_size += evergreen_vs_size * 4; | 568 | obj_size += evergreen_vs_size * 4; |
| @@ -567,6 +592,12 @@ int evergreen_blit_init(struct radeon_device *rdev) | |||
| 567 | return r; | 592 | return r; |
| 568 | } | 593 | } |
| 569 | 594 | ||
| 595 | memcpy_toio(ptr + rdev->r600_blit.state_offset, | ||
| 596 | evergreen_default_state, rdev->r600_blit.state_len * 4); | ||
| 597 | |||
| 598 | if (num_packet2s) | ||
| 599 | memcpy_toio(ptr + rdev->r600_blit.state_offset + (rdev->r600_blit.state_len * 4), | ||
| 600 | packet2s, num_packet2s * 4); | ||
| 570 | memcpy(ptr + rdev->r600_blit.vs_offset, evergreen_vs, evergreen_vs_size * 4); | 601 | memcpy(ptr + rdev->r600_blit.vs_offset, evergreen_vs, evergreen_vs_size * 4); |
| 571 | memcpy(ptr + rdev->r600_blit.ps_offset, evergreen_ps, evergreen_ps_size * 4); | 602 | memcpy(ptr + rdev->r600_blit.ps_offset, evergreen_ps, evergreen_ps_size * 4); |
| 572 | radeon_bo_kunmap(rdev->r600_blit.shader_obj); | 603 | radeon_bo_kunmap(rdev->r600_blit.shader_obj); |
| @@ -652,7 +683,7 @@ int evergreen_blit_prepare_copy(struct radeon_device *rdev, int size_bytes) | |||
| 652 | /* calculate number of loops correctly */ | 683 | /* calculate number of loops correctly */ |
| 653 | ring_size = num_loops * dwords_per_loop; | 684 | ring_size = num_loops * dwords_per_loop; |
| 654 | /* set default + shaders */ | 685 | /* set default + shaders */ |
| 655 | ring_size += 46; /* shaders + def state */ | 686 | ring_size += 52; /* shaders + def state */ |
| 656 | ring_size += 10; /* fence emit for VB IB */ | 687 | ring_size += 10; /* fence emit for VB IB */ |
| 657 | ring_size += 5; /* done copy */ | 688 | ring_size += 5; /* done copy */ |
| 658 | ring_size += 10; /* fence emit for done copy */ | 689 | ring_size += 10; /* fence emit for done copy */ |
| @@ -660,7 +691,7 @@ int evergreen_blit_prepare_copy(struct radeon_device *rdev, int size_bytes) | |||
| 660 | if (r) | 691 | if (r) |
| 661 | return r; | 692 | return r; |
| 662 | 693 | ||
| 663 | set_default_state(rdev); /* 30 */ | 694 | set_default_state(rdev); /* 36 */ |
| 664 | set_shaders(rdev); /* 16 */ | 695 | set_shaders(rdev); /* 16 */ |
| 665 | return 0; | 696 | return 0; |
| 666 | } | 697 | } |
diff --git a/drivers/gpu/drm/radeon/evergreend.h b/drivers/gpu/drm/radeon/evergreend.h index 36d32d83d86..afec1aca2a7 100644 --- a/drivers/gpu/drm/radeon/evergreend.h +++ b/drivers/gpu/drm/radeon/evergreend.h | |||
| @@ -240,6 +240,7 @@ | |||
| 240 | #define FORCE_EOV_MAX_CLK_CNT(x) ((x) << 0) | 240 | #define FORCE_EOV_MAX_CLK_CNT(x) ((x) << 0) |
| 241 | #define FORCE_EOV_MAX_REZ_CNT(x) ((x) << 16) | 241 | #define FORCE_EOV_MAX_REZ_CNT(x) ((x) << 16) |
| 242 | #define PA_SC_LINE_STIPPLE 0x28A0C | 242 | #define PA_SC_LINE_STIPPLE 0x28A0C |
| 243 | #define PA_SU_LINE_STIPPLE_VALUE 0x8A60 | ||
| 243 | #define PA_SC_LINE_STIPPLE_STATE 0x8B10 | 244 | #define PA_SC_LINE_STIPPLE_STATE 0x8B10 |
| 244 | 245 | ||
| 245 | #define SCRATCH_REG0 0x8500 | 246 | #define SCRATCH_REG0 0x8500 |
| @@ -652,6 +653,7 @@ | |||
| 652 | #define PACKET3_DISPATCH_DIRECT 0x15 | 653 | #define PACKET3_DISPATCH_DIRECT 0x15 |
| 653 | #define PACKET3_DISPATCH_INDIRECT 0x16 | 654 | #define PACKET3_DISPATCH_INDIRECT 0x16 |
| 654 | #define PACKET3_INDIRECT_BUFFER_END 0x17 | 655 | #define PACKET3_INDIRECT_BUFFER_END 0x17 |
| 656 | #define PACKET3_MODE_CONTROL 0x18 | ||
| 655 | #define PACKET3_SET_PREDICATION 0x20 | 657 | #define PACKET3_SET_PREDICATION 0x20 |
| 656 | #define PACKET3_REG_RMW 0x21 | 658 | #define PACKET3_REG_RMW 0x21 |
| 657 | #define PACKET3_COND_EXEC 0x22 | 659 | #define PACKET3_COND_EXEC 0x22 |
diff --git a/drivers/gpu/drm/radeon/r100.c b/drivers/gpu/drm/radeon/r100.c index 46da5142b13..5f15820efe1 100644 --- a/drivers/gpu/drm/radeon/r100.c +++ b/drivers/gpu/drm/radeon/r100.c | |||
| @@ -1031,8 +1031,8 @@ int r100_cp_init(struct radeon_device *rdev, unsigned ring_size) | |||
| 1031 | WREG32(RADEON_CP_CSQ_MODE, | 1031 | WREG32(RADEON_CP_CSQ_MODE, |
| 1032 | REG_SET(RADEON_INDIRECT2_START, indirect2_start) | | 1032 | REG_SET(RADEON_INDIRECT2_START, indirect2_start) | |
| 1033 | REG_SET(RADEON_INDIRECT1_START, indirect1_start)); | 1033 | REG_SET(RADEON_INDIRECT1_START, indirect1_start)); |
| 1034 | WREG32(0x718, 0); | 1034 | WREG32(RADEON_CP_RB_WPTR_DELAY, 0); |
| 1035 | WREG32(0x744, 0x00004D4D); | 1035 | WREG32(RADEON_CP_CSQ_MODE, 0x00004D4D); |
| 1036 | WREG32(RADEON_CP_CSQ_CNTL, RADEON_CSQ_PRIBM_INDBM); | 1036 | WREG32(RADEON_CP_CSQ_CNTL, RADEON_CSQ_PRIBM_INDBM); |
| 1037 | radeon_ring_start(rdev); | 1037 | radeon_ring_start(rdev); |
| 1038 | r = radeon_ring_test(rdev); | 1038 | r = radeon_ring_test(rdev); |
| @@ -2347,10 +2347,10 @@ void r100_vga_set_state(struct radeon_device *rdev, bool state) | |||
| 2347 | 2347 | ||
| 2348 | temp = RREG32(RADEON_CONFIG_CNTL); | 2348 | temp = RREG32(RADEON_CONFIG_CNTL); |
| 2349 | if (state == false) { | 2349 | if (state == false) { |
| 2350 | temp &= ~(1<<8); | 2350 | temp &= ~RADEON_CFG_VGA_RAM_EN; |
| 2351 | temp |= (1<<9); | 2351 | temp |= RADEON_CFG_VGA_IO_DIS; |
| 2352 | } else { | 2352 | } else { |
| 2353 | temp &= ~(1<<9); | 2353 | temp &= ~RADEON_CFG_VGA_IO_DIS; |
| 2354 | } | 2354 | } |
| 2355 | WREG32(RADEON_CONFIG_CNTL, temp); | 2355 | WREG32(RADEON_CONFIG_CNTL, temp); |
| 2356 | } | 2356 | } |
| @@ -3522,7 +3522,7 @@ int r100_ring_test(struct radeon_device *rdev) | |||
| 3522 | if (i < rdev->usec_timeout) { | 3522 | if (i < rdev->usec_timeout) { |
| 3523 | DRM_INFO("ring test succeeded in %d usecs\n", i); | 3523 | DRM_INFO("ring test succeeded in %d usecs\n", i); |
| 3524 | } else { | 3524 | } else { |
| 3525 | DRM_ERROR("radeon: ring test failed (sracth(0x%04X)=0x%08X)\n", | 3525 | DRM_ERROR("radeon: ring test failed (scratch(0x%04X)=0x%08X)\n", |
| 3526 | scratch, tmp); | 3526 | scratch, tmp); |
| 3527 | r = -EINVAL; | 3527 | r = -EINVAL; |
| 3528 | } | 3528 | } |
diff --git a/drivers/gpu/drm/radeon/r300.c b/drivers/gpu/drm/radeon/r300.c index cf862ca580b..55fe5ba7def 100644 --- a/drivers/gpu/drm/radeon/r300.c +++ b/drivers/gpu/drm/radeon/r300.c | |||
| @@ -69,6 +69,9 @@ void rv370_pcie_gart_tlb_flush(struct radeon_device *rdev) | |||
| 69 | mb(); | 69 | mb(); |
| 70 | } | 70 | } |
| 71 | 71 | ||
| 72 | #define R300_PTE_WRITEABLE (1 << 2) | ||
| 73 | #define R300_PTE_READABLE (1 << 3) | ||
| 74 | |||
| 72 | int rv370_pcie_gart_set_page(struct radeon_device *rdev, int i, uint64_t addr) | 75 | int rv370_pcie_gart_set_page(struct radeon_device *rdev, int i, uint64_t addr) |
| 73 | { | 76 | { |
| 74 | void __iomem *ptr = (void *)rdev->gart.table.vram.ptr; | 77 | void __iomem *ptr = (void *)rdev->gart.table.vram.ptr; |
| @@ -78,7 +81,7 @@ int rv370_pcie_gart_set_page(struct radeon_device *rdev, int i, uint64_t addr) | |||
| 78 | } | 81 | } |
| 79 | addr = (lower_32_bits(addr) >> 8) | | 82 | addr = (lower_32_bits(addr) >> 8) | |
| 80 | ((upper_32_bits(addr) & 0xff) << 24) | | 83 | ((upper_32_bits(addr) & 0xff) << 24) | |
| 81 | 0xc; | 84 | R300_PTE_WRITEABLE | R300_PTE_READABLE; |
| 82 | /* on x86 we want this to be CPU endian, on powerpc | 85 | /* on x86 we want this to be CPU endian, on powerpc |
| 83 | * on powerpc without HW swappers, it'll get swapped on way | 86 | * on powerpc without HW swappers, it'll get swapped on way |
| 84 | * into VRAM - so no need for cpu_to_le32 on VRAM tables */ | 87 | * into VRAM - so no need for cpu_to_le32 on VRAM tables */ |
| @@ -135,7 +138,7 @@ int rv370_pcie_gart_enable(struct radeon_device *rdev) | |||
| 135 | WREG32_PCIE(RADEON_PCIE_TX_DISCARD_RD_ADDR_LO, rdev->mc.vram_start); | 138 | WREG32_PCIE(RADEON_PCIE_TX_DISCARD_RD_ADDR_LO, rdev->mc.vram_start); |
| 136 | WREG32_PCIE(RADEON_PCIE_TX_DISCARD_RD_ADDR_HI, 0); | 139 | WREG32_PCIE(RADEON_PCIE_TX_DISCARD_RD_ADDR_HI, 0); |
| 137 | /* Clear error */ | 140 | /* Clear error */ |
| 138 | WREG32_PCIE(0x18, 0); | 141 | WREG32_PCIE(RADEON_PCIE_TX_GART_ERROR, 0); |
| 139 | tmp = RREG32_PCIE(RADEON_PCIE_TX_GART_CNTL); | 142 | tmp = RREG32_PCIE(RADEON_PCIE_TX_GART_CNTL); |
| 140 | tmp |= RADEON_PCIE_TX_GART_EN; | 143 | tmp |= RADEON_PCIE_TX_GART_EN; |
| 141 | tmp |= RADEON_PCIE_TX_GART_UNMAPPED_ACCESS_DISCARD; | 144 | tmp |= RADEON_PCIE_TX_GART_UNMAPPED_ACCESS_DISCARD; |
diff --git a/drivers/gpu/drm/radeon/r420.c b/drivers/gpu/drm/radeon/r420.c index c387346f93a..0b59ed7c7d2 100644 --- a/drivers/gpu/drm/radeon/r420.c +++ b/drivers/gpu/drm/radeon/r420.c | |||
| @@ -96,7 +96,7 @@ void r420_pipes_init(struct radeon_device *rdev) | |||
| 96 | "programming pipes. Bad things might happen.\n"); | 96 | "programming pipes. Bad things might happen.\n"); |
| 97 | } | 97 | } |
| 98 | /* get max number of pipes */ | 98 | /* get max number of pipes */ |
| 99 | gb_pipe_select = RREG32(0x402C); | 99 | gb_pipe_select = RREG32(R400_GB_PIPE_SELECT); |
| 100 | num_pipes = ((gb_pipe_select >> 12) & 3) + 1; | 100 | num_pipes = ((gb_pipe_select >> 12) & 3) + 1; |
| 101 | 101 | ||
| 102 | /* SE chips have 1 pipe */ | 102 | /* SE chips have 1 pipe */ |
diff --git a/drivers/gpu/drm/radeon/r520.c b/drivers/gpu/drm/radeon/r520.c index 3c8677f9e38..2ce80d97656 100644 --- a/drivers/gpu/drm/radeon/r520.c +++ b/drivers/gpu/drm/radeon/r520.c | |||
| @@ -79,8 +79,8 @@ static void r520_gpu_init(struct radeon_device *rdev) | |||
| 79 | WREG32(0x4128, 0xFF); | 79 | WREG32(0x4128, 0xFF); |
| 80 | } | 80 | } |
| 81 | r420_pipes_init(rdev); | 81 | r420_pipes_init(rdev); |
| 82 | gb_pipe_select = RREG32(0x402C); | 82 | gb_pipe_select = RREG32(R400_GB_PIPE_SELECT); |
| 83 | tmp = RREG32(0x170C); | 83 | tmp = RREG32(R300_DST_PIPE_CONFIG); |
| 84 | pipe_select_current = (tmp >> 2) & 3; | 84 | pipe_select_current = (tmp >> 2) & 3; |
| 85 | tmp = (1 << pipe_select_current) | | 85 | tmp = (1 << pipe_select_current) | |
| 86 | (((gb_pipe_select >> 8) & 0xF) << 4); | 86 | (((gb_pipe_select >> 8) & 0xF) << 4); |
diff --git a/drivers/gpu/drm/radeon/r600.c b/drivers/gpu/drm/radeon/r600.c index aca2236268f..650672a0f5a 100644 --- a/drivers/gpu/drm/radeon/r600.c +++ b/drivers/gpu/drm/radeon/r600.c | |||
| @@ -97,12 +97,16 @@ void r600_irq_disable(struct radeon_device *rdev); | |||
| 97 | static void r600_pcie_gen2_enable(struct radeon_device *rdev); | 97 | static void r600_pcie_gen2_enable(struct radeon_device *rdev); |
| 98 | 98 | ||
| 99 | /* get temperature in millidegrees */ | 99 | /* get temperature in millidegrees */ |
| 100 | u32 rv6xx_get_temp(struct radeon_device *rdev) | 100 | int rv6xx_get_temp(struct radeon_device *rdev) |
| 101 | { | 101 | { |
| 102 | u32 temp = (RREG32(CG_THERMAL_STATUS) & ASIC_T_MASK) >> | 102 | u32 temp = (RREG32(CG_THERMAL_STATUS) & ASIC_T_MASK) >> |
| 103 | ASIC_T_SHIFT; | 103 | ASIC_T_SHIFT; |
| 104 | int actual_temp = temp & 0xff; | ||
| 104 | 105 | ||
| 105 | return temp * 1000; | 106 | if (temp & 0x100) |
| 107 | actual_temp -= 256; | ||
| 108 | |||
| 109 | return actual_temp * 1000; | ||
| 106 | } | 110 | } |
| 107 | 111 | ||
| 108 | void r600_pm_get_dynpm_state(struct radeon_device *rdev) | 112 | void r600_pm_get_dynpm_state(struct radeon_device *rdev) |
| @@ -1287,6 +1291,9 @@ int r600_gpu_soft_reset(struct radeon_device *rdev) | |||
| 1287 | S_008014_CB2_BUSY(1) | S_008014_CB3_BUSY(1); | 1291 | S_008014_CB2_BUSY(1) | S_008014_CB3_BUSY(1); |
| 1288 | u32 tmp; | 1292 | u32 tmp; |
| 1289 | 1293 | ||
| 1294 | if (!(RREG32(GRBM_STATUS) & GUI_ACTIVE)) | ||
| 1295 | return 0; | ||
| 1296 | |||
| 1290 | dev_info(rdev->dev, "GPU softreset \n"); | 1297 | dev_info(rdev->dev, "GPU softreset \n"); |
| 1291 | dev_info(rdev->dev, " R_008010_GRBM_STATUS=0x%08X\n", | 1298 | dev_info(rdev->dev, " R_008010_GRBM_STATUS=0x%08X\n", |
| 1292 | RREG32(R_008010_GRBM_STATUS)); | 1299 | RREG32(R_008010_GRBM_STATUS)); |
diff --git a/drivers/gpu/drm/radeon/r600_reg.h b/drivers/gpu/drm/radeon/r600_reg.h index 33cda016b08..f869897c745 100644 --- a/drivers/gpu/drm/radeon/r600_reg.h +++ b/drivers/gpu/drm/radeon/r600_reg.h | |||
| @@ -81,7 +81,11 @@ | |||
| 81 | #define R600_MEDIUM_VID_LOWER_GPIO_CNTL 0x720 | 81 | #define R600_MEDIUM_VID_LOWER_GPIO_CNTL 0x720 |
| 82 | #define R600_LOW_VID_LOWER_GPIO_CNTL 0x724 | 82 | #define R600_LOW_VID_LOWER_GPIO_CNTL 0x724 |
| 83 | 83 | ||
| 84 | 84 | #define R600_D1GRPH_SWAP_CONTROL 0x610C | |
| 85 | # define R600_D1GRPH_SWAP_ENDIAN_NONE (0 << 0) | ||
| 86 | # define R600_D1GRPH_SWAP_ENDIAN_16BIT (1 << 0) | ||
| 87 | # define R600_D1GRPH_SWAP_ENDIAN_32BIT (2 << 0) | ||
| 88 | # define R600_D1GRPH_SWAP_ENDIAN_64BIT (3 << 0) | ||
| 85 | 89 | ||
| 86 | #define R600_HDP_NONSURFACE_BASE 0x2c04 | 90 | #define R600_HDP_NONSURFACE_BASE 0x2c04 |
| 87 | 91 | ||
diff --git a/drivers/gpu/drm/radeon/radeon.h b/drivers/gpu/drm/radeon/radeon.h index 71d2a554bbe..56c48b67ef3 100644 --- a/drivers/gpu/drm/radeon/radeon.h +++ b/drivers/gpu/drm/radeon/radeon.h | |||
| @@ -179,10 +179,10 @@ void radeon_combios_get_power_modes(struct radeon_device *rdev); | |||
| 179 | void radeon_atombios_get_power_modes(struct radeon_device *rdev); | 179 | void radeon_atombios_get_power_modes(struct radeon_device *rdev); |
| 180 | void radeon_atom_set_voltage(struct radeon_device *rdev, u16 level); | 180 | void radeon_atom_set_voltage(struct radeon_device *rdev, u16 level); |
| 181 | void rs690_pm_info(struct radeon_device *rdev); | 181 | void rs690_pm_info(struct radeon_device *rdev); |
| 182 | extern u32 rv6xx_get_temp(struct radeon_device *rdev); | 182 | extern int rv6xx_get_temp(struct radeon_device *rdev); |
| 183 | extern u32 rv770_get_temp(struct radeon_device *rdev); | 183 | extern int rv770_get_temp(struct radeon_device *rdev); |
| 184 | extern u32 evergreen_get_temp(struct radeon_device *rdev); | 184 | extern int evergreen_get_temp(struct radeon_device *rdev); |
| 185 | extern u32 sumo_get_temp(struct radeon_device *rdev); | 185 | extern int sumo_get_temp(struct radeon_device *rdev); |
| 186 | 186 | ||
| 187 | /* | 187 | /* |
| 188 | * Fences. | 188 | * Fences. |
| @@ -812,8 +812,7 @@ struct radeon_pm { | |||
| 812 | fixed20_12 sclk; | 812 | fixed20_12 sclk; |
| 813 | fixed20_12 mclk; | 813 | fixed20_12 mclk; |
| 814 | fixed20_12 needed_bandwidth; | 814 | fixed20_12 needed_bandwidth; |
| 815 | /* XXX: use a define for num power modes */ | 815 | struct radeon_power_state *power_state; |
| 816 | struct radeon_power_state power_state[8]; | ||
| 817 | /* number of valid power states */ | 816 | /* number of valid power states */ |
| 818 | int num_power_states; | 817 | int num_power_states; |
| 819 | int current_power_state_index; | 818 | int current_power_state_index; |
diff --git a/drivers/gpu/drm/radeon/radeon_asic.c b/drivers/gpu/drm/radeon/radeon_asic.c index 3a1b1618622..e75d63b8e21 100644 --- a/drivers/gpu/drm/radeon/radeon_asic.c +++ b/drivers/gpu/drm/radeon/radeon_asic.c | |||
| @@ -759,7 +759,7 @@ static struct radeon_asic evergreen_asic = { | |||
| 759 | .gart_tlb_flush = &evergreen_pcie_gart_tlb_flush, | 759 | .gart_tlb_flush = &evergreen_pcie_gart_tlb_flush, |
| 760 | .gart_set_page = &rs600_gart_set_page, | 760 | .gart_set_page = &rs600_gart_set_page, |
| 761 | .ring_test = &r600_ring_test, | 761 | .ring_test = &r600_ring_test, |
| 762 | .ring_ib_execute = &r600_ring_ib_execute, | 762 | .ring_ib_execute = &evergreen_ring_ib_execute, |
| 763 | .irq_set = &evergreen_irq_set, | 763 | .irq_set = &evergreen_irq_set, |
| 764 | .irq_process = &evergreen_irq_process, | 764 | .irq_process = &evergreen_irq_process, |
| 765 | .get_vblank_counter = &evergreen_get_vblank_counter, | 765 | .get_vblank_counter = &evergreen_get_vblank_counter, |
| @@ -805,7 +805,7 @@ static struct radeon_asic sumo_asic = { | |||
| 805 | .gart_tlb_flush = &evergreen_pcie_gart_tlb_flush, | 805 | .gart_tlb_flush = &evergreen_pcie_gart_tlb_flush, |
| 806 | .gart_set_page = &rs600_gart_set_page, | 806 | .gart_set_page = &rs600_gart_set_page, |
| 807 | .ring_test = &r600_ring_test, | 807 | .ring_test = &r600_ring_test, |
| 808 | .ring_ib_execute = &r600_ring_ib_execute, | 808 | .ring_ib_execute = &evergreen_ring_ib_execute, |
| 809 | .irq_set = &evergreen_irq_set, | 809 | .irq_set = &evergreen_irq_set, |
| 810 | .irq_process = &evergreen_irq_process, | 810 | .irq_process = &evergreen_irq_process, |
| 811 | .get_vblank_counter = &evergreen_get_vblank_counter, | 811 | .get_vblank_counter = &evergreen_get_vblank_counter, |
| @@ -848,7 +848,7 @@ static struct radeon_asic btc_asic = { | |||
| 848 | .gart_tlb_flush = &evergreen_pcie_gart_tlb_flush, | 848 | .gart_tlb_flush = &evergreen_pcie_gart_tlb_flush, |
| 849 | .gart_set_page = &rs600_gart_set_page, | 849 | .gart_set_page = &rs600_gart_set_page, |
| 850 | .ring_test = &r600_ring_test, | 850 | .ring_test = &r600_ring_test, |
| 851 | .ring_ib_execute = &r600_ring_ib_execute, | 851 | .ring_ib_execute = &evergreen_ring_ib_execute, |
| 852 | .irq_set = &evergreen_irq_set, | 852 | .irq_set = &evergreen_irq_set, |
| 853 | .irq_process = &evergreen_irq_process, | 853 | .irq_process = &evergreen_irq_process, |
| 854 | .get_vblank_counter = &evergreen_get_vblank_counter, | 854 | .get_vblank_counter = &evergreen_get_vblank_counter, |
diff --git a/drivers/gpu/drm/radeon/radeon_asic.h b/drivers/gpu/drm/radeon/radeon_asic.h index e01f0771853..c59bd98a202 100644 --- a/drivers/gpu/drm/radeon/radeon_asic.h +++ b/drivers/gpu/drm/radeon/radeon_asic.h | |||
| @@ -355,6 +355,7 @@ int evergreen_resume(struct radeon_device *rdev); | |||
| 355 | bool evergreen_gpu_is_lockup(struct radeon_device *rdev); | 355 | bool evergreen_gpu_is_lockup(struct radeon_device *rdev); |
| 356 | int evergreen_asic_reset(struct radeon_device *rdev); | 356 | int evergreen_asic_reset(struct radeon_device *rdev); |
| 357 | void evergreen_bandwidth_update(struct radeon_device *rdev); | 357 | void evergreen_bandwidth_update(struct radeon_device *rdev); |
| 358 | void evergreen_ring_ib_execute(struct radeon_device *rdev, struct radeon_ib *ib); | ||
| 358 | int evergreen_copy_blit(struct radeon_device *rdev, | 359 | int evergreen_copy_blit(struct radeon_device *rdev, |
| 359 | uint64_t src_offset, uint64_t dst_offset, | 360 | uint64_t src_offset, uint64_t dst_offset, |
| 360 | unsigned num_pages, struct radeon_fence *fence); | 361 | unsigned num_pages, struct radeon_fence *fence); |
diff --git a/drivers/gpu/drm/radeon/radeon_atombios.c b/drivers/gpu/drm/radeon/radeon_atombios.c index 1573202a641..5c1cc7ad9a1 100644 --- a/drivers/gpu/drm/radeon/radeon_atombios.c +++ b/drivers/gpu/drm/radeon/radeon_atombios.c | |||
| @@ -387,15 +387,11 @@ static bool radeon_atom_apply_quirks(struct drm_device *dev, | |||
| 387 | *line_mux = 0x90; | 387 | *line_mux = 0x90; |
| 388 | } | 388 | } |
| 389 | 389 | ||
| 390 | /* mac rv630 */ | 390 | /* mac rv630, rv730, others */ |
| 391 | if ((dev->pdev->device == 0x9588) && | 391 | if ((supported_device == ATOM_DEVICE_TV1_SUPPORT) && |
| 392 | (dev->pdev->subsystem_vendor == 0x106b) && | 392 | (*connector_type == DRM_MODE_CONNECTOR_DVII)) { |
| 393 | (dev->pdev->subsystem_device == 0x00a6)) { | 393 | *connector_type = DRM_MODE_CONNECTOR_9PinDIN; |
| 394 | if ((supported_device == ATOM_DEVICE_TV1_SUPPORT) && | 394 | *line_mux = CONNECTOR_7PIN_DIN_ENUM_ID1; |
| 395 | (*connector_type == DRM_MODE_CONNECTOR_DVII)) { | ||
| 396 | *connector_type = DRM_MODE_CONNECTOR_9PinDIN; | ||
| 397 | *line_mux = CONNECTOR_7PIN_DIN_ENUM_ID1; | ||
| 398 | } | ||
| 399 | } | 395 | } |
| 400 | 396 | ||
| 401 | /* ASUS HD 3600 XT board lists the DVI port as HDMI */ | 397 | /* ASUS HD 3600 XT board lists the DVI port as HDMI */ |
| @@ -1167,16 +1163,6 @@ bool radeon_atom_get_clock_info(struct drm_device *dev) | |||
| 1167 | p1pll->pll_out_min = 64800; | 1163 | p1pll->pll_out_min = 64800; |
| 1168 | else | 1164 | else |
| 1169 | p1pll->pll_out_min = 20000; | 1165 | p1pll->pll_out_min = 20000; |
| 1170 | } else if (p1pll->pll_out_min > 64800) { | ||
| 1171 | /* Limiting the pll output range is a good thing generally as | ||
| 1172 | * it limits the number of possible pll combinations for a given | ||
| 1173 | * frequency presumably to the ones that work best on each card. | ||
| 1174 | * However, certain duallink DVI monitors seem to like | ||
| 1175 | * pll combinations that would be limited by this at least on | ||
| 1176 | * pre-DCE 3.0 r6xx hardware. This might need to be adjusted per | ||
| 1177 | * family. | ||
| 1178 | */ | ||
| 1179 | p1pll->pll_out_min = 64800; | ||
| 1180 | } | 1166 | } |
| 1181 | 1167 | ||
| 1182 | p1pll->pll_in_min = | 1168 | p1pll->pll_in_min = |
| @@ -1991,6 +1977,9 @@ static int radeon_atombios_parse_power_table_1_3(struct radeon_device *rdev) | |||
| 1991 | num_modes = power_info->info.ucNumOfPowerModeEntries; | 1977 | num_modes = power_info->info.ucNumOfPowerModeEntries; |
| 1992 | if (num_modes > ATOM_MAX_NUMBEROF_POWER_BLOCK) | 1978 | if (num_modes > ATOM_MAX_NUMBEROF_POWER_BLOCK) |
| 1993 | num_modes = ATOM_MAX_NUMBEROF_POWER_BLOCK; | 1979 | num_modes = ATOM_MAX_NUMBEROF_POWER_BLOCK; |
| 1980 | rdev->pm.power_state = kzalloc(sizeof(struct radeon_power_state) * num_modes, GFP_KERNEL); | ||
| 1981 | if (!rdev->pm.power_state) | ||
| 1982 | return state_index; | ||
| 1994 | /* last mode is usually default, array is low to high */ | 1983 | /* last mode is usually default, array is low to high */ |
| 1995 | for (i = 0; i < num_modes; i++) { | 1984 | for (i = 0; i < num_modes; i++) { |
| 1996 | rdev->pm.power_state[state_index].clock_info[0].voltage.type = VOLTAGE_NONE; | 1985 | rdev->pm.power_state[state_index].clock_info[0].voltage.type = VOLTAGE_NONE; |
| @@ -2342,6 +2331,10 @@ static int radeon_atombios_parse_power_table_4_5(struct radeon_device *rdev) | |||
| 2342 | power_info = (union power_info *)(mode_info->atom_context->bios + data_offset); | 2331 | power_info = (union power_info *)(mode_info->atom_context->bios + data_offset); |
| 2343 | 2332 | ||
| 2344 | radeon_atombios_add_pplib_thermal_controller(rdev, &power_info->pplib.sThermalController); | 2333 | radeon_atombios_add_pplib_thermal_controller(rdev, &power_info->pplib.sThermalController); |
| 2334 | rdev->pm.power_state = kzalloc(sizeof(struct radeon_power_state) * | ||
| 2335 | power_info->pplib.ucNumStates, GFP_KERNEL); | ||
| 2336 | if (!rdev->pm.power_state) | ||
| 2337 | return state_index; | ||
| 2345 | /* first mode is usually default, followed by low to high */ | 2338 | /* first mode is usually default, followed by low to high */ |
| 2346 | for (i = 0; i < power_info->pplib.ucNumStates; i++) { | 2339 | for (i = 0; i < power_info->pplib.ucNumStates; i++) { |
| 2347 | mode_index = 0; | 2340 | mode_index = 0; |
| @@ -2422,6 +2415,10 @@ static int radeon_atombios_parse_power_table_6(struct radeon_device *rdev) | |||
| 2422 | non_clock_info_array = (struct NonClockInfoArray *) | 2415 | non_clock_info_array = (struct NonClockInfoArray *) |
| 2423 | (mode_info->atom_context->bios + data_offset + | 2416 | (mode_info->atom_context->bios + data_offset + |
| 2424 | power_info->pplib.usNonClockInfoArrayOffset); | 2417 | power_info->pplib.usNonClockInfoArrayOffset); |
| 2418 | rdev->pm.power_state = kzalloc(sizeof(struct radeon_power_state) * | ||
| 2419 | state_array->ucNumEntries, GFP_KERNEL); | ||
| 2420 | if (!rdev->pm.power_state) | ||
| 2421 | return state_index; | ||
| 2425 | for (i = 0; i < state_array->ucNumEntries; i++) { | 2422 | for (i = 0; i < state_array->ucNumEntries; i++) { |
| 2426 | mode_index = 0; | 2423 | mode_index = 0; |
| 2427 | power_state = (union pplib_power_state *)&state_array->states[i]; | 2424 | power_state = (union pplib_power_state *)&state_array->states[i]; |
| @@ -2495,19 +2492,22 @@ void radeon_atombios_get_power_modes(struct radeon_device *rdev) | |||
| 2495 | break; | 2492 | break; |
| 2496 | } | 2493 | } |
| 2497 | } else { | 2494 | } else { |
| 2498 | /* add the default mode */ | 2495 | rdev->pm.power_state = kzalloc(sizeof(struct radeon_power_state), GFP_KERNEL); |
| 2499 | rdev->pm.power_state[state_index].type = | 2496 | if (rdev->pm.power_state) { |
| 2500 | POWER_STATE_TYPE_DEFAULT; | 2497 | /* add the default mode */ |
| 2501 | rdev->pm.power_state[state_index].num_clock_modes = 1; | 2498 | rdev->pm.power_state[state_index].type = |
| 2502 | rdev->pm.power_state[state_index].clock_info[0].mclk = rdev->clock.default_mclk; | 2499 | POWER_STATE_TYPE_DEFAULT; |
| 2503 | rdev->pm.power_state[state_index].clock_info[0].sclk = rdev->clock.default_sclk; | 2500 | rdev->pm.power_state[state_index].num_clock_modes = 1; |
| 2504 | rdev->pm.power_state[state_index].default_clock_mode = | 2501 | rdev->pm.power_state[state_index].clock_info[0].mclk = rdev->clock.default_mclk; |
| 2505 | &rdev->pm.power_state[state_index].clock_info[0]; | 2502 | rdev->pm.power_state[state_index].clock_info[0].sclk = rdev->clock.default_sclk; |
| 2506 | rdev->pm.power_state[state_index].clock_info[0].voltage.type = VOLTAGE_NONE; | 2503 | rdev->pm.power_state[state_index].default_clock_mode = |
| 2507 | rdev->pm.power_state[state_index].pcie_lanes = 16; | 2504 | &rdev->pm.power_state[state_index].clock_info[0]; |
| 2508 | rdev->pm.default_power_state_index = state_index; | 2505 | rdev->pm.power_state[state_index].clock_info[0].voltage.type = VOLTAGE_NONE; |
| 2509 | rdev->pm.power_state[state_index].flags = 0; | 2506 | rdev->pm.power_state[state_index].pcie_lanes = 16; |
| 2510 | state_index++; | 2507 | rdev->pm.default_power_state_index = state_index; |
| 2508 | rdev->pm.power_state[state_index].flags = 0; | ||
| 2509 | state_index++; | ||
| 2510 | } | ||
| 2511 | } | 2511 | } |
| 2512 | 2512 | ||
| 2513 | rdev->pm.num_power_states = state_index; | 2513 | rdev->pm.num_power_states = state_index; |
| @@ -2623,7 +2623,7 @@ void radeon_atom_initialize_bios_scratch_regs(struct drm_device *dev) | |||
| 2623 | bios_2_scratch &= ~ATOM_S2_VRI_BRIGHT_ENABLE; | 2623 | bios_2_scratch &= ~ATOM_S2_VRI_BRIGHT_ENABLE; |
| 2624 | 2624 | ||
| 2625 | /* tell the bios not to handle mode switching */ | 2625 | /* tell the bios not to handle mode switching */ |
| 2626 | bios_6_scratch |= (ATOM_S6_ACC_BLOCK_DISPLAY_SWITCH | ATOM_S6_ACC_MODE); | 2626 | bios_6_scratch |= ATOM_S6_ACC_BLOCK_DISPLAY_SWITCH; |
| 2627 | 2627 | ||
| 2628 | if (rdev->family >= CHIP_R600) { | 2628 | if (rdev->family >= CHIP_R600) { |
| 2629 | WREG32(R600_BIOS_2_SCRATCH, bios_2_scratch); | 2629 | WREG32(R600_BIOS_2_SCRATCH, bios_2_scratch); |
| @@ -2674,10 +2674,13 @@ void radeon_atom_output_lock(struct drm_encoder *encoder, bool lock) | |||
| 2674 | else | 2674 | else |
| 2675 | bios_6_scratch = RREG32(RADEON_BIOS_6_SCRATCH); | 2675 | bios_6_scratch = RREG32(RADEON_BIOS_6_SCRATCH); |
| 2676 | 2676 | ||
| 2677 | if (lock) | 2677 | if (lock) { |
| 2678 | bios_6_scratch |= ATOM_S6_CRITICAL_STATE; | 2678 | bios_6_scratch |= ATOM_S6_CRITICAL_STATE; |
| 2679 | else | 2679 | bios_6_scratch &= ~ATOM_S6_ACC_MODE; |
| 2680 | } else { | ||
| 2680 | bios_6_scratch &= ~ATOM_S6_CRITICAL_STATE; | 2681 | bios_6_scratch &= ~ATOM_S6_CRITICAL_STATE; |
| 2682 | bios_6_scratch |= ATOM_S6_ACC_MODE; | ||
| 2683 | } | ||
| 2681 | 2684 | ||
| 2682 | if (rdev->family >= CHIP_R600) | 2685 | if (rdev->family >= CHIP_R600) |
| 2683 | WREG32(R600_BIOS_6_SCRATCH, bios_6_scratch); | 2686 | WREG32(R600_BIOS_6_SCRATCH, bios_6_scratch); |
diff --git a/drivers/gpu/drm/radeon/radeon_combios.c b/drivers/gpu/drm/radeon/radeon_combios.c index 591fcae8f22..d27ef74590c 100644 --- a/drivers/gpu/drm/radeon/radeon_combios.c +++ b/drivers/gpu/drm/radeon/radeon_combios.c | |||
| @@ -2442,6 +2442,17 @@ void radeon_combios_get_power_modes(struct radeon_device *rdev) | |||
| 2442 | 2442 | ||
| 2443 | rdev->pm.default_power_state_index = -1; | 2443 | rdev->pm.default_power_state_index = -1; |
| 2444 | 2444 | ||
| 2445 | /* allocate 2 power states */ | ||
| 2446 | rdev->pm.power_state = kzalloc(sizeof(struct radeon_power_state) * 2, GFP_KERNEL); | ||
| 2447 | if (!rdev->pm.power_state) { | ||
| 2448 | rdev->pm.default_power_state_index = state_index; | ||
| 2449 | rdev->pm.num_power_states = 0; | ||
| 2450 | |||
| 2451 | rdev->pm.current_power_state_index = rdev->pm.default_power_state_index; | ||
| 2452 | rdev->pm.current_clock_mode_index = 0; | ||
| 2453 | return; | ||
| 2454 | } | ||
| 2455 | |||
| 2445 | if (rdev->flags & RADEON_IS_MOBILITY) { | 2456 | if (rdev->flags & RADEON_IS_MOBILITY) { |
| 2446 | offset = combios_get_table_offset(dev, COMBIOS_POWERPLAY_INFO_TABLE); | 2457 | offset = combios_get_table_offset(dev, COMBIOS_POWERPLAY_INFO_TABLE); |
| 2447 | if (offset) { | 2458 | if (offset) { |
diff --git a/drivers/gpu/drm/radeon/radeon_device.c b/drivers/gpu/drm/radeon/radeon_device.c index 26091d602b8..0d478932b1a 100644 --- a/drivers/gpu/drm/radeon/radeon_device.c +++ b/drivers/gpu/drm/radeon/radeon_device.c | |||
| @@ -891,9 +891,9 @@ int radeon_suspend_kms(struct drm_device *dev, pm_message_t state) | |||
| 891 | pci_disable_device(dev->pdev); | 891 | pci_disable_device(dev->pdev); |
| 892 | pci_set_power_state(dev->pdev, PCI_D3hot); | 892 | pci_set_power_state(dev->pdev, PCI_D3hot); |
| 893 | } | 893 | } |
| 894 | acquire_console_sem(); | 894 | console_lock(); |
| 895 | radeon_fbdev_set_suspend(rdev, 1); | 895 | radeon_fbdev_set_suspend(rdev, 1); |
| 896 | release_console_sem(); | 896 | console_unlock(); |
| 897 | return 0; | 897 | return 0; |
| 898 | } | 898 | } |
| 899 | 899 | ||
| @@ -905,11 +905,11 @@ int radeon_resume_kms(struct drm_device *dev) | |||
| 905 | if (dev->switch_power_state == DRM_SWITCH_POWER_OFF) | 905 | if (dev->switch_power_state == DRM_SWITCH_POWER_OFF) |
| 906 | return 0; | 906 | return 0; |
| 907 | 907 | ||
| 908 | acquire_console_sem(); | 908 | console_lock(); |
| 909 | pci_set_power_state(dev->pdev, PCI_D0); | 909 | pci_set_power_state(dev->pdev, PCI_D0); |
| 910 | pci_restore_state(dev->pdev); | 910 | pci_restore_state(dev->pdev); |
| 911 | if (pci_enable_device(dev->pdev)) { | 911 | if (pci_enable_device(dev->pdev)) { |
| 912 | release_console_sem(); | 912 | console_unlock(); |
| 913 | return -1; | 913 | return -1; |
| 914 | } | 914 | } |
| 915 | pci_set_master(dev->pdev); | 915 | pci_set_master(dev->pdev); |
| @@ -920,7 +920,7 @@ int radeon_resume_kms(struct drm_device *dev) | |||
| 920 | radeon_restore_bios_scratch_regs(rdev); | 920 | radeon_restore_bios_scratch_regs(rdev); |
| 921 | 921 | ||
| 922 | radeon_fbdev_set_suspend(rdev, 0); | 922 | radeon_fbdev_set_suspend(rdev, 0); |
| 923 | release_console_sem(); | 923 | console_unlock(); |
| 924 | 924 | ||
| 925 | /* reset hpd state */ | 925 | /* reset hpd state */ |
| 926 | radeon_hpd_init(rdev); | 926 | radeon_hpd_init(rdev); |
diff --git a/drivers/gpu/drm/radeon/radeon_display.c b/drivers/gpu/drm/radeon/radeon_display.c index d26dabf878d..2eff98cfd72 100644 --- a/drivers/gpu/drm/radeon/radeon_display.c +++ b/drivers/gpu/drm/radeon/radeon_display.c | |||
| @@ -780,6 +780,115 @@ static int radeon_ddc_dump(struct drm_connector *connector) | |||
| 780 | return ret; | 780 | return ret; |
| 781 | } | 781 | } |
| 782 | 782 | ||
| 783 | /* avivo */ | ||
| 784 | static void avivo_get_fb_div(struct radeon_pll *pll, | ||
| 785 | u32 target_clock, | ||
| 786 | u32 post_div, | ||
| 787 | u32 ref_div, | ||
| 788 | u32 *fb_div, | ||
| 789 | u32 *frac_fb_div) | ||
| 790 | { | ||
| 791 | u32 tmp = post_div * ref_div; | ||
| 792 | |||
| 793 | tmp *= target_clock; | ||
| 794 | *fb_div = tmp / pll->reference_freq; | ||
| 795 | *frac_fb_div = tmp % pll->reference_freq; | ||
| 796 | } | ||
| 797 | |||
| 798 | static u32 avivo_get_post_div(struct radeon_pll *pll, | ||
| 799 | u32 target_clock) | ||
| 800 | { | ||
| 801 | u32 vco, post_div, tmp; | ||
| 802 | |||
| 803 | if (pll->flags & RADEON_PLL_USE_POST_DIV) | ||
| 804 | return pll->post_div; | ||
| 805 | |||
| 806 | if (pll->flags & RADEON_PLL_PREFER_MINM_OVER_MAXP) { | ||
| 807 | if (pll->flags & RADEON_PLL_IS_LCD) | ||
| 808 | vco = pll->lcd_pll_out_min; | ||
| 809 | else | ||
| 810 | vco = pll->pll_out_min; | ||
| 811 | } else { | ||
| 812 | if (pll->flags & RADEON_PLL_IS_LCD) | ||
| 813 | vco = pll->lcd_pll_out_max; | ||
| 814 | else | ||
| 815 | vco = pll->pll_out_max; | ||
| 816 | } | ||
| 817 | |||
| 818 | post_div = vco / target_clock; | ||
| 819 | tmp = vco % target_clock; | ||
| 820 | |||
| 821 | if (pll->flags & RADEON_PLL_PREFER_MINM_OVER_MAXP) { | ||
| 822 | if (tmp) | ||
| 823 | post_div++; | ||
| 824 | } else { | ||
| 825 | if (!tmp) | ||
| 826 | post_div--; | ||
| 827 | } | ||
| 828 | |||
| 829 | return post_div; | ||
| 830 | } | ||
| 831 | |||
| 832 | #define MAX_TOLERANCE 10 | ||
| 833 | |||
| 834 | void radeon_compute_pll_avivo(struct radeon_pll *pll, | ||
| 835 | u32 freq, | ||
| 836 | u32 *dot_clock_p, | ||
| 837 | u32 *fb_div_p, | ||
| 838 | u32 *frac_fb_div_p, | ||
| 839 | u32 *ref_div_p, | ||
| 840 | u32 *post_div_p) | ||
| 841 | { | ||
| 842 | u32 target_clock = freq / 10; | ||
| 843 | u32 post_div = avivo_get_post_div(pll, target_clock); | ||
| 844 | u32 ref_div = pll->min_ref_div; | ||
| 845 | u32 fb_div = 0, frac_fb_div = 0, tmp; | ||
| 846 | |||
| 847 | if (pll->flags & RADEON_PLL_USE_REF_DIV) | ||
| 848 | ref_div = pll->reference_div; | ||
| 849 | |||
| 850 | if (pll->flags & RADEON_PLL_USE_FRAC_FB_DIV) { | ||
| 851 | avivo_get_fb_div(pll, target_clock, post_div, ref_div, &fb_div, &frac_fb_div); | ||
| 852 | frac_fb_div = (100 * frac_fb_div) / pll->reference_freq; | ||
| 853 | if (frac_fb_div >= 5) { | ||
| 854 | frac_fb_div -= 5; | ||
| 855 | frac_fb_div = frac_fb_div / 10; | ||
| 856 | frac_fb_div++; | ||
| 857 | } | ||
| 858 | if (frac_fb_div >= 10) { | ||
| 859 | fb_div++; | ||
| 860 | frac_fb_div = 0; | ||
| 861 | } | ||
| 862 | } else { | ||
| 863 | while (ref_div <= pll->max_ref_div) { | ||
| 864 | avivo_get_fb_div(pll, target_clock, post_div, ref_div, | ||
| 865 | &fb_div, &frac_fb_div); | ||
| 866 | if (frac_fb_div >= (pll->reference_freq / 2)) | ||
| 867 | fb_div++; | ||
| 868 | frac_fb_div = 0; | ||
| 869 | tmp = (pll->reference_freq * fb_div) / (post_div * ref_div); | ||
| 870 | tmp = (tmp * 10000) / target_clock; | ||
| 871 | |||
| 872 | if (tmp > (10000 + MAX_TOLERANCE)) | ||
| 873 | ref_div++; | ||
| 874 | else if (tmp >= (10000 - MAX_TOLERANCE)) | ||
| 875 | break; | ||
| 876 | else | ||
| 877 | ref_div++; | ||
| 878 | } | ||
| 879 | } | ||
| 880 | |||
| 881 | *dot_clock_p = ((pll->reference_freq * fb_div * 10) + (pll->reference_freq * frac_fb_div)) / | ||
| 882 | (ref_div * post_div * 10); | ||
| 883 | *fb_div_p = fb_div; | ||
| 884 | *frac_fb_div_p = frac_fb_div; | ||
| 885 | *ref_div_p = ref_div; | ||
| 886 | *post_div_p = post_div; | ||
| 887 | DRM_DEBUG_KMS("%d, pll dividers - fb: %d.%d ref: %d, post %d\n", | ||
| 888 | *dot_clock_p, fb_div, frac_fb_div, ref_div, post_div); | ||
| 889 | } | ||
| 890 | |||
| 891 | /* pre-avivo */ | ||
| 783 | static inline uint32_t radeon_div(uint64_t n, uint32_t d) | 892 | static inline uint32_t radeon_div(uint64_t n, uint32_t d) |
| 784 | { | 893 | { |
| 785 | uint64_t mod; | 894 | uint64_t mod; |
| @@ -790,13 +899,13 @@ static inline uint32_t radeon_div(uint64_t n, uint32_t d) | |||
| 790 | return n; | 899 | return n; |
| 791 | } | 900 | } |
| 792 | 901 | ||
| 793 | void radeon_compute_pll(struct radeon_pll *pll, | 902 | void radeon_compute_pll_legacy(struct radeon_pll *pll, |
| 794 | uint64_t freq, | 903 | uint64_t freq, |
| 795 | uint32_t *dot_clock_p, | 904 | uint32_t *dot_clock_p, |
| 796 | uint32_t *fb_div_p, | 905 | uint32_t *fb_div_p, |
| 797 | uint32_t *frac_fb_div_p, | 906 | uint32_t *frac_fb_div_p, |
| 798 | uint32_t *ref_div_p, | 907 | uint32_t *ref_div_p, |
| 799 | uint32_t *post_div_p) | 908 | uint32_t *post_div_p) |
| 800 | { | 909 | { |
| 801 | uint32_t min_ref_div = pll->min_ref_div; | 910 | uint32_t min_ref_div = pll->min_ref_div; |
| 802 | uint32_t max_ref_div = pll->max_ref_div; | 911 | uint32_t max_ref_div = pll->max_ref_div; |
| @@ -826,6 +935,9 @@ void radeon_compute_pll(struct radeon_pll *pll, | |||
| 826 | pll_out_max = pll->pll_out_max; | 935 | pll_out_max = pll->pll_out_max; |
| 827 | } | 936 | } |
| 828 | 937 | ||
| 938 | if (pll_out_min > 64800) | ||
| 939 | pll_out_min = 64800; | ||
| 940 | |||
| 829 | if (pll->flags & RADEON_PLL_USE_REF_DIV) | 941 | if (pll->flags & RADEON_PLL_USE_REF_DIV) |
| 830 | min_ref_div = max_ref_div = pll->reference_div; | 942 | min_ref_div = max_ref_div = pll->reference_div; |
| 831 | else { | 943 | else { |
| @@ -849,7 +961,7 @@ void radeon_compute_pll(struct radeon_pll *pll, | |||
| 849 | max_fractional_feed_div = pll->max_frac_feedback_div; | 961 | max_fractional_feed_div = pll->max_frac_feedback_div; |
| 850 | } | 962 | } |
| 851 | 963 | ||
| 852 | for (post_div = max_post_div; post_div >= min_post_div; --post_div) { | 964 | for (post_div = min_post_div; post_div <= max_post_div; ++post_div) { |
| 853 | uint32_t ref_div; | 965 | uint32_t ref_div; |
| 854 | 966 | ||
| 855 | if ((pll->flags & RADEON_PLL_NO_ODD_POST_DIV) && (post_div & 1)) | 967 | if ((pll->flags & RADEON_PLL_NO_ODD_POST_DIV) && (post_div & 1)) |
| @@ -965,6 +1077,10 @@ void radeon_compute_pll(struct radeon_pll *pll, | |||
| 965 | *frac_fb_div_p = best_frac_feedback_div; | 1077 | *frac_fb_div_p = best_frac_feedback_div; |
| 966 | *ref_div_p = best_ref_div; | 1078 | *ref_div_p = best_ref_div; |
| 967 | *post_div_p = best_post_div; | 1079 | *post_div_p = best_post_div; |
| 1080 | DRM_DEBUG_KMS("%d %d, pll dividers - fb: %d.%d ref: %d, post %d\n", | ||
| 1081 | freq, best_freq / 1000, best_feedback_div, best_frac_feedback_div, | ||
| 1082 | best_ref_div, best_post_div); | ||
| 1083 | |||
| 968 | } | 1084 | } |
| 969 | 1085 | ||
| 970 | static void radeon_user_framebuffer_destroy(struct drm_framebuffer *fb) | 1086 | static void radeon_user_framebuffer_destroy(struct drm_framebuffer *fb) |
diff --git a/drivers/gpu/drm/radeon/radeon_drv.c b/drivers/gpu/drm/radeon/radeon_drv.c index d5680a0c87a..275b26a708d 100644 --- a/drivers/gpu/drm/radeon/radeon_drv.c +++ b/drivers/gpu/drm/radeon/radeon_drv.c | |||
| @@ -48,7 +48,7 @@ | |||
| 48 | * - 2.5.0 - add get accel 2 to work around ddx breakage for evergreen | 48 | * - 2.5.0 - add get accel 2 to work around ddx breakage for evergreen |
| 49 | * - 2.6.0 - add tiling config query (r6xx+), add initial HiZ support (r300->r500) | 49 | * - 2.6.0 - add tiling config query (r6xx+), add initial HiZ support (r300->r500) |
| 50 | * 2.7.0 - fixups for r600 2D tiling support. (no external ABI change), add eg dyn gpr regs | 50 | * 2.7.0 - fixups for r600 2D tiling support. (no external ABI change), add eg dyn gpr regs |
| 51 | * 2.8.0 - pageflip support, r500 US_FORMAT regs. r500 ARGB2101010 colorbuf, r300->r500 CMASK | 51 | * 2.8.0 - pageflip support, r500 US_FORMAT regs. r500 ARGB2101010 colorbuf, r300->r500 CMASK, clock crystal query |
| 52 | */ | 52 | */ |
| 53 | #define KMS_DRIVER_MAJOR 2 | 53 | #define KMS_DRIVER_MAJOR 2 |
| 54 | #define KMS_DRIVER_MINOR 8 | 54 | #define KMS_DRIVER_MINOR 8 |
diff --git a/drivers/gpu/drm/radeon/radeon_encoders.c b/drivers/gpu/drm/radeon/radeon_encoders.c index 8fd184286c0..d4a54224761 100644 --- a/drivers/gpu/drm/radeon/radeon_encoders.c +++ b/drivers/gpu/drm/radeon/radeon_encoders.c | |||
| @@ -641,7 +641,7 @@ atombios_get_encoder_mode(struct drm_encoder *encoder) | |||
| 641 | switch (connector->connector_type) { | 641 | switch (connector->connector_type) { |
| 642 | case DRM_MODE_CONNECTOR_DVII: | 642 | case DRM_MODE_CONNECTOR_DVII: |
| 643 | case DRM_MODE_CONNECTOR_HDMIB: /* HDMI-B is basically DL-DVI; analog works fine */ | 643 | case DRM_MODE_CONNECTOR_HDMIB: /* HDMI-B is basically DL-DVI; analog works fine */ |
| 644 | if (drm_detect_monitor_audio(radeon_connector->edid)) { | 644 | if (drm_detect_monitor_audio(radeon_connector->edid) && radeon_audio) { |
| 645 | /* fix me */ | 645 | /* fix me */ |
| 646 | if (ASIC_IS_DCE4(rdev)) | 646 | if (ASIC_IS_DCE4(rdev)) |
| 647 | return ATOM_ENCODER_MODE_DVI; | 647 | return ATOM_ENCODER_MODE_DVI; |
| @@ -655,7 +655,7 @@ atombios_get_encoder_mode(struct drm_encoder *encoder) | |||
| 655 | case DRM_MODE_CONNECTOR_DVID: | 655 | case DRM_MODE_CONNECTOR_DVID: |
| 656 | case DRM_MODE_CONNECTOR_HDMIA: | 656 | case DRM_MODE_CONNECTOR_HDMIA: |
| 657 | default: | 657 | default: |
| 658 | if (drm_detect_monitor_audio(radeon_connector->edid)) { | 658 | if (drm_detect_monitor_audio(radeon_connector->edid) && radeon_audio) { |
| 659 | /* fix me */ | 659 | /* fix me */ |
| 660 | if (ASIC_IS_DCE4(rdev)) | 660 | if (ASIC_IS_DCE4(rdev)) |
| 661 | return ATOM_ENCODER_MODE_DVI; | 661 | return ATOM_ENCODER_MODE_DVI; |
| @@ -673,7 +673,7 @@ atombios_get_encoder_mode(struct drm_encoder *encoder) | |||
| 673 | if ((dig_connector->dp_sink_type == CONNECTOR_OBJECT_ID_DISPLAYPORT) || | 673 | if ((dig_connector->dp_sink_type == CONNECTOR_OBJECT_ID_DISPLAYPORT) || |
| 674 | (dig_connector->dp_sink_type == CONNECTOR_OBJECT_ID_eDP)) | 674 | (dig_connector->dp_sink_type == CONNECTOR_OBJECT_ID_eDP)) |
| 675 | return ATOM_ENCODER_MODE_DP; | 675 | return ATOM_ENCODER_MODE_DP; |
| 676 | else if (drm_detect_monitor_audio(radeon_connector->edid)) { | 676 | else if (drm_detect_monitor_audio(radeon_connector->edid) && radeon_audio) { |
| 677 | /* fix me */ | 677 | /* fix me */ |
| 678 | if (ASIC_IS_DCE4(rdev)) | 678 | if (ASIC_IS_DCE4(rdev)) |
| 679 | return ATOM_ENCODER_MODE_DVI; | 679 | return ATOM_ENCODER_MODE_DVI; |
| @@ -1063,7 +1063,7 @@ atombios_set_edp_panel_power(struct drm_connector *connector, int action) | |||
| 1063 | if (!ASIC_IS_DCE4(rdev)) | 1063 | if (!ASIC_IS_DCE4(rdev)) |
| 1064 | return; | 1064 | return; |
| 1065 | 1065 | ||
| 1066 | if ((action != ATOM_TRANSMITTER_ACTION_POWER_ON) || | 1066 | if ((action != ATOM_TRANSMITTER_ACTION_POWER_ON) && |
| 1067 | (action != ATOM_TRANSMITTER_ACTION_POWER_OFF)) | 1067 | (action != ATOM_TRANSMITTER_ACTION_POWER_OFF)) |
| 1068 | return; | 1068 | return; |
| 1069 | 1069 | ||
diff --git a/drivers/gpu/drm/radeon/radeon_irq_kms.c b/drivers/gpu/drm/radeon/radeon_irq_kms.c index a289646e8aa..9ec830c77af 100644 --- a/drivers/gpu/drm/radeon/radeon_irq_kms.c +++ b/drivers/gpu/drm/radeon/radeon_irq_kms.c | |||
| @@ -110,11 +110,14 @@ void radeon_driver_irq_uninstall_kms(struct drm_device *dev) | |||
| 110 | 110 | ||
| 111 | int radeon_irq_kms_init(struct radeon_device *rdev) | 111 | int radeon_irq_kms_init(struct radeon_device *rdev) |
| 112 | { | 112 | { |
| 113 | int i; | ||
| 113 | int r = 0; | 114 | int r = 0; |
| 114 | 115 | ||
| 115 | INIT_WORK(&rdev->hotplug_work, radeon_hotplug_work_func); | 116 | INIT_WORK(&rdev->hotplug_work, radeon_hotplug_work_func); |
| 116 | 117 | ||
| 117 | spin_lock_init(&rdev->irq.sw_lock); | 118 | spin_lock_init(&rdev->irq.sw_lock); |
| 119 | for (i = 0; i < rdev->num_crtc; i++) | ||
| 120 | spin_lock_init(&rdev->irq.pflip_lock[i]); | ||
| 118 | r = drm_vblank_init(rdev->ddev, rdev->num_crtc); | 121 | r = drm_vblank_init(rdev->ddev, rdev->num_crtc); |
| 119 | if (r) { | 122 | if (r) { |
| 120 | return r; | 123 | return r; |
diff --git a/drivers/gpu/drm/radeon/radeon_kms.c b/drivers/gpu/drm/radeon/radeon_kms.c index 28a53e4a925..8387d32caaa 100644 --- a/drivers/gpu/drm/radeon/radeon_kms.c +++ b/drivers/gpu/drm/radeon/radeon_kms.c | |||
| @@ -201,6 +201,10 @@ int radeon_info_ioctl(struct drm_device *dev, void *data, struct drm_file *filp) | |||
| 201 | } | 201 | } |
| 202 | radeon_set_filp_rights(dev, &rdev->cmask_filp, filp, &value); | 202 | radeon_set_filp_rights(dev, &rdev->cmask_filp, filp, &value); |
| 203 | break; | 203 | break; |
| 204 | case RADEON_INFO_CLOCK_CRYSTAL_FREQ: | ||
| 205 | /* return clock value in KHz */ | ||
| 206 | value = rdev->clock.spll.reference_freq * 10; | ||
| 207 | break; | ||
| 204 | default: | 208 | default: |
| 205 | DRM_DEBUG_KMS("Invalid request %d\n", info->request); | 209 | DRM_DEBUG_KMS("Invalid request %d\n", info->request); |
| 206 | return -EINVAL; | 210 | return -EINVAL; |
| @@ -243,6 +247,8 @@ void radeon_driver_preclose_kms(struct drm_device *dev, | |||
| 243 | struct radeon_device *rdev = dev->dev_private; | 247 | struct radeon_device *rdev = dev->dev_private; |
| 244 | if (rdev->hyperz_filp == file_priv) | 248 | if (rdev->hyperz_filp == file_priv) |
| 245 | rdev->hyperz_filp = NULL; | 249 | rdev->hyperz_filp = NULL; |
| 250 | if (rdev->cmask_filp == file_priv) | ||
| 251 | rdev->cmask_filp = NULL; | ||
| 246 | } | 252 | } |
| 247 | 253 | ||
| 248 | /* | 254 | /* |
diff --git a/drivers/gpu/drm/radeon/radeon_legacy_crtc.c b/drivers/gpu/drm/radeon/radeon_legacy_crtc.c index ace2e6384d4..cf0638c3b7c 100644 --- a/drivers/gpu/drm/radeon/radeon_legacy_crtc.c +++ b/drivers/gpu/drm/radeon/radeon_legacy_crtc.c | |||
| @@ -778,9 +778,9 @@ static void radeon_set_pll(struct drm_crtc *crtc, struct drm_display_mode *mode) | |||
| 778 | DRM_DEBUG_KMS("\n"); | 778 | DRM_DEBUG_KMS("\n"); |
| 779 | 779 | ||
| 780 | if (!use_bios_divs) { | 780 | if (!use_bios_divs) { |
| 781 | radeon_compute_pll(pll, mode->clock, | 781 | radeon_compute_pll_legacy(pll, mode->clock, |
| 782 | &freq, &feedback_div, &frac_fb_div, | 782 | &freq, &feedback_div, &frac_fb_div, |
| 783 | &reference_div, &post_divider); | 783 | &reference_div, &post_divider); |
| 784 | 784 | ||
| 785 | for (post_div = &post_divs[0]; post_div->divider; ++post_div) { | 785 | for (post_div = &post_divs[0]; post_div->divider; ++post_div) { |
| 786 | if (post_div->divider == post_divider) | 786 | if (post_div->divider == post_divider) |
diff --git a/drivers/gpu/drm/radeon/radeon_mode.h b/drivers/gpu/drm/radeon/radeon_mode.h index 12bdeab91c8..6794cdf91f2 100644 --- a/drivers/gpu/drm/radeon/radeon_mode.h +++ b/drivers/gpu/drm/radeon/radeon_mode.h | |||
| @@ -149,6 +149,7 @@ struct radeon_tmds_pll { | |||
| 149 | #define RADEON_PLL_PREFER_CLOSEST_LOWER (1 << 11) | 149 | #define RADEON_PLL_PREFER_CLOSEST_LOWER (1 << 11) |
| 150 | #define RADEON_PLL_USE_POST_DIV (1 << 12) | 150 | #define RADEON_PLL_USE_POST_DIV (1 << 12) |
| 151 | #define RADEON_PLL_IS_LCD (1 << 13) | 151 | #define RADEON_PLL_IS_LCD (1 << 13) |
| 152 | #define RADEON_PLL_PREFER_MINM_OVER_MAXP (1 << 14) | ||
| 152 | 153 | ||
| 153 | struct radeon_pll { | 154 | struct radeon_pll { |
| 154 | /* reference frequency */ | 155 | /* reference frequency */ |
| @@ -510,13 +511,21 @@ extern bool radeon_atombios_get_asic_ss_info(struct radeon_device *rdev, | |||
| 510 | struct radeon_atom_ss *ss, | 511 | struct radeon_atom_ss *ss, |
| 511 | int id, u32 clock); | 512 | int id, u32 clock); |
| 512 | 513 | ||
| 513 | extern void radeon_compute_pll(struct radeon_pll *pll, | 514 | extern void radeon_compute_pll_legacy(struct radeon_pll *pll, |
| 514 | uint64_t freq, | 515 | uint64_t freq, |
| 515 | uint32_t *dot_clock_p, | 516 | uint32_t *dot_clock_p, |
| 516 | uint32_t *fb_div_p, | 517 | uint32_t *fb_div_p, |
| 517 | uint32_t *frac_fb_div_p, | 518 | uint32_t *frac_fb_div_p, |
| 518 | uint32_t *ref_div_p, | 519 | uint32_t *ref_div_p, |
| 519 | uint32_t *post_div_p); | 520 | uint32_t *post_div_p); |
| 521 | |||
| 522 | extern void radeon_compute_pll_avivo(struct radeon_pll *pll, | ||
| 523 | u32 freq, | ||
| 524 | u32 *dot_clock_p, | ||
| 525 | u32 *fb_div_p, | ||
| 526 | u32 *frac_fb_div_p, | ||
| 527 | u32 *ref_div_p, | ||
| 528 | u32 *post_div_p); | ||
| 520 | 529 | ||
| 521 | extern void radeon_setup_encoder_clones(struct drm_device *dev); | 530 | extern void radeon_setup_encoder_clones(struct drm_device *dev); |
| 522 | 531 | ||
diff --git a/drivers/gpu/drm/radeon/radeon_pm.c b/drivers/gpu/drm/radeon/radeon_pm.c index 3b1b2bf9cdd..2aed03bde4b 100644 --- a/drivers/gpu/drm/radeon/radeon_pm.c +++ b/drivers/gpu/drm/radeon/radeon_pm.c | |||
| @@ -430,7 +430,7 @@ static ssize_t radeon_hwmon_show_temp(struct device *dev, | |||
| 430 | { | 430 | { |
| 431 | struct drm_device *ddev = pci_get_drvdata(to_pci_dev(dev)); | 431 | struct drm_device *ddev = pci_get_drvdata(to_pci_dev(dev)); |
| 432 | struct radeon_device *rdev = ddev->dev_private; | 432 | struct radeon_device *rdev = ddev->dev_private; |
| 433 | u32 temp; | 433 | int temp; |
| 434 | 434 | ||
| 435 | switch (rdev->pm.int_thermal_type) { | 435 | switch (rdev->pm.int_thermal_type) { |
| 436 | case THERMAL_TYPE_RV6XX: | 436 | case THERMAL_TYPE_RV6XX: |
| @@ -646,6 +646,9 @@ void radeon_pm_fini(struct radeon_device *rdev) | |||
| 646 | #endif | 646 | #endif |
| 647 | } | 647 | } |
| 648 | 648 | ||
| 649 | if (rdev->pm.power_state) | ||
| 650 | kfree(rdev->pm.power_state); | ||
| 651 | |||
| 649 | radeon_hwmon_fini(rdev); | 652 | radeon_hwmon_fini(rdev); |
| 650 | } | 653 | } |
| 651 | 654 | ||
diff --git a/drivers/gpu/drm/radeon/radeon_reg.h b/drivers/gpu/drm/radeon/radeon_reg.h index 3cd4dace57c..ec93a75369e 100644 --- a/drivers/gpu/drm/radeon/radeon_reg.h +++ b/drivers/gpu/drm/radeon/radeon_reg.h | |||
| @@ -375,6 +375,8 @@ | |||
| 375 | #define RADEON_CONFIG_APER_SIZE 0x0108 | 375 | #define RADEON_CONFIG_APER_SIZE 0x0108 |
| 376 | #define RADEON_CONFIG_BONDS 0x00e8 | 376 | #define RADEON_CONFIG_BONDS 0x00e8 |
| 377 | #define RADEON_CONFIG_CNTL 0x00e0 | 377 | #define RADEON_CONFIG_CNTL 0x00e0 |
| 378 | # define RADEON_CFG_VGA_RAM_EN (1 << 8) | ||
| 379 | # define RADEON_CFG_VGA_IO_DIS (1 << 9) | ||
| 378 | # define RADEON_CFG_ATI_REV_A11 (0 << 16) | 380 | # define RADEON_CFG_ATI_REV_A11 (0 << 16) |
| 379 | # define RADEON_CFG_ATI_REV_A12 (1 << 16) | 381 | # define RADEON_CFG_ATI_REV_A12 (1 << 16) |
| 380 | # define RADEON_CFG_ATI_REV_A13 (2 << 16) | 382 | # define RADEON_CFG_ATI_REV_A13 (2 << 16) |
diff --git a/drivers/gpu/drm/radeon/rs400.c b/drivers/gpu/drm/radeon/rs400.c index 5512e4e5e63..c76283d9eb3 100644 --- a/drivers/gpu/drm/radeon/rs400.c +++ b/drivers/gpu/drm/radeon/rs400.c | |||
| @@ -203,6 +203,9 @@ void rs400_gart_fini(struct radeon_device *rdev) | |||
| 203 | radeon_gart_table_ram_free(rdev); | 203 | radeon_gart_table_ram_free(rdev); |
| 204 | } | 204 | } |
| 205 | 205 | ||
| 206 | #define RS400_PTE_WRITEABLE (1 << 2) | ||
| 207 | #define RS400_PTE_READABLE (1 << 3) | ||
| 208 | |||
| 206 | int rs400_gart_set_page(struct radeon_device *rdev, int i, uint64_t addr) | 209 | int rs400_gart_set_page(struct radeon_device *rdev, int i, uint64_t addr) |
| 207 | { | 210 | { |
| 208 | uint32_t entry; | 211 | uint32_t entry; |
| @@ -213,7 +216,7 @@ int rs400_gart_set_page(struct radeon_device *rdev, int i, uint64_t addr) | |||
| 213 | 216 | ||
| 214 | entry = (lower_32_bits(addr) & PAGE_MASK) | | 217 | entry = (lower_32_bits(addr) & PAGE_MASK) | |
| 215 | ((upper_32_bits(addr) & 0xff) << 4) | | 218 | ((upper_32_bits(addr) & 0xff) << 4) | |
| 216 | 0xc; | 219 | RS400_PTE_WRITEABLE | RS400_PTE_READABLE; |
| 217 | entry = cpu_to_le32(entry); | 220 | entry = cpu_to_le32(entry); |
| 218 | rdev->gart.table.ram.ptr[i] = entry; | 221 | rdev->gart.table.ram.ptr[i] = entry; |
| 219 | return 0; | 222 | return 0; |
| @@ -226,8 +229,8 @@ int rs400_mc_wait_for_idle(struct radeon_device *rdev) | |||
| 226 | 229 | ||
| 227 | for (i = 0; i < rdev->usec_timeout; i++) { | 230 | for (i = 0; i < rdev->usec_timeout; i++) { |
| 228 | /* read MC_STATUS */ | 231 | /* read MC_STATUS */ |
| 229 | tmp = RREG32(0x0150); | 232 | tmp = RREG32(RADEON_MC_STATUS); |
| 230 | if (tmp & (1 << 2)) { | 233 | if (tmp & RADEON_MC_IDLE) { |
| 231 | return 0; | 234 | return 0; |
| 232 | } | 235 | } |
| 233 | DRM_UDELAY(1); | 236 | DRM_UDELAY(1); |
| @@ -241,7 +244,7 @@ void rs400_gpu_init(struct radeon_device *rdev) | |||
| 241 | r420_pipes_init(rdev); | 244 | r420_pipes_init(rdev); |
| 242 | if (rs400_mc_wait_for_idle(rdev)) { | 245 | if (rs400_mc_wait_for_idle(rdev)) { |
| 243 | printk(KERN_WARNING "rs400: Failed to wait MC idle while " | 246 | printk(KERN_WARNING "rs400: Failed to wait MC idle while " |
| 244 | "programming pipes. Bad things might happen. %08x\n", RREG32(0x150)); | 247 | "programming pipes. Bad things might happen. %08x\n", RREG32(RADEON_MC_STATUS)); |
| 245 | } | 248 | } |
| 246 | } | 249 | } |
| 247 | 250 | ||
| @@ -300,9 +303,9 @@ static int rs400_debugfs_gart_info(struct seq_file *m, void *data) | |||
| 300 | seq_printf(m, "MCCFG_AGP_BASE_2 0x%08x\n", tmp); | 303 | seq_printf(m, "MCCFG_AGP_BASE_2 0x%08x\n", tmp); |
| 301 | tmp = RREG32_MC(RS690_MCCFG_AGP_LOCATION); | 304 | tmp = RREG32_MC(RS690_MCCFG_AGP_LOCATION); |
| 302 | seq_printf(m, "MCCFG_AGP_LOCATION 0x%08x\n", tmp); | 305 | seq_printf(m, "MCCFG_AGP_LOCATION 0x%08x\n", tmp); |
| 303 | tmp = RREG32_MC(0x100); | 306 | tmp = RREG32_MC(RS690_MCCFG_FB_LOCATION); |
| 304 | seq_printf(m, "MCCFG_FB_LOCATION 0x%08x\n", tmp); | 307 | seq_printf(m, "MCCFG_FB_LOCATION 0x%08x\n", tmp); |
| 305 | tmp = RREG32(0x134); | 308 | tmp = RREG32(RS690_HDP_FB_LOCATION); |
| 306 | seq_printf(m, "HDP_FB_LOCATION 0x%08x\n", tmp); | 309 | seq_printf(m, "HDP_FB_LOCATION 0x%08x\n", tmp); |
| 307 | } else { | 310 | } else { |
| 308 | tmp = RREG32(RADEON_AGP_BASE); | 311 | tmp = RREG32(RADEON_AGP_BASE); |
diff --git a/drivers/gpu/drm/radeon/rv515.c b/drivers/gpu/drm/radeon/rv515.c index 5d569f41f4a..64b57af9371 100644 --- a/drivers/gpu/drm/radeon/rv515.c +++ b/drivers/gpu/drm/radeon/rv515.c | |||
| @@ -69,13 +69,13 @@ void rv515_ring_start(struct radeon_device *rdev) | |||
| 69 | ISYNC_CPSCRATCH_IDLEGUI); | 69 | ISYNC_CPSCRATCH_IDLEGUI); |
| 70 | radeon_ring_write(rdev, PACKET0(WAIT_UNTIL, 0)); | 70 | radeon_ring_write(rdev, PACKET0(WAIT_UNTIL, 0)); |
| 71 | radeon_ring_write(rdev, WAIT_2D_IDLECLEAN | WAIT_3D_IDLECLEAN); | 71 | radeon_ring_write(rdev, WAIT_2D_IDLECLEAN | WAIT_3D_IDLECLEAN); |
| 72 | radeon_ring_write(rdev, PACKET0(0x170C, 0)); | 72 | radeon_ring_write(rdev, PACKET0(R300_DST_PIPE_CONFIG, 0)); |
| 73 | radeon_ring_write(rdev, 1 << 31); | 73 | radeon_ring_write(rdev, R300_PIPE_AUTO_CONFIG); |
| 74 | radeon_ring_write(rdev, PACKET0(GB_SELECT, 0)); | 74 | radeon_ring_write(rdev, PACKET0(GB_SELECT, 0)); |
| 75 | radeon_ring_write(rdev, 0); | 75 | radeon_ring_write(rdev, 0); |
| 76 | radeon_ring_write(rdev, PACKET0(GB_ENABLE, 0)); | 76 | radeon_ring_write(rdev, PACKET0(GB_ENABLE, 0)); |
| 77 | radeon_ring_write(rdev, 0); | 77 | radeon_ring_write(rdev, 0); |
| 78 | radeon_ring_write(rdev, PACKET0(0x42C8, 0)); | 78 | radeon_ring_write(rdev, PACKET0(R500_SU_REG_DEST, 0)); |
| 79 | radeon_ring_write(rdev, (1 << rdev->num_gb_pipes) - 1); | 79 | radeon_ring_write(rdev, (1 << rdev->num_gb_pipes) - 1); |
| 80 | radeon_ring_write(rdev, PACKET0(VAP_INDEX_OFFSET, 0)); | 80 | radeon_ring_write(rdev, PACKET0(VAP_INDEX_OFFSET, 0)); |
| 81 | radeon_ring_write(rdev, 0); | 81 | radeon_ring_write(rdev, 0); |
| @@ -153,8 +153,8 @@ void rv515_gpu_init(struct radeon_device *rdev) | |||
| 153 | } | 153 | } |
| 154 | rv515_vga_render_disable(rdev); | 154 | rv515_vga_render_disable(rdev); |
| 155 | r420_pipes_init(rdev); | 155 | r420_pipes_init(rdev); |
| 156 | gb_pipe_select = RREG32(0x402C); | 156 | gb_pipe_select = RREG32(R400_GB_PIPE_SELECT); |
| 157 | tmp = RREG32(0x170C); | 157 | tmp = RREG32(R300_DST_PIPE_CONFIG); |
| 158 | pipe_select_current = (tmp >> 2) & 3; | 158 | pipe_select_current = (tmp >> 2) & 3; |
| 159 | tmp = (1 << pipe_select_current) | | 159 | tmp = (1 << pipe_select_current) | |
| 160 | (((gb_pipe_select >> 8) & 0xF) << 4); | 160 | (((gb_pipe_select >> 8) & 0xF) << 4); |
diff --git a/drivers/gpu/drm/radeon/rv770.c b/drivers/gpu/drm/radeon/rv770.c index 491dc900065..2211a323db4 100644 --- a/drivers/gpu/drm/radeon/rv770.c +++ b/drivers/gpu/drm/radeon/rv770.c | |||
| @@ -78,18 +78,23 @@ u32 rv770_page_flip(struct radeon_device *rdev, int crtc_id, u64 crtc_base) | |||
| 78 | } | 78 | } |
| 79 | 79 | ||
| 80 | /* get temperature in millidegrees */ | 80 | /* get temperature in millidegrees */ |
| 81 | u32 rv770_get_temp(struct radeon_device *rdev) | 81 | int rv770_get_temp(struct radeon_device *rdev) |
| 82 | { | 82 | { |
| 83 | u32 temp = (RREG32(CG_MULT_THERMAL_STATUS) & ASIC_T_MASK) >> | 83 | u32 temp = (RREG32(CG_MULT_THERMAL_STATUS) & ASIC_T_MASK) >> |
| 84 | ASIC_T_SHIFT; | 84 | ASIC_T_SHIFT; |
| 85 | u32 actual_temp = 0; | 85 | int actual_temp; |
| 86 | 86 | ||
| 87 | if ((temp >> 9) & 1) | 87 | if (temp & 0x400) |
| 88 | actual_temp = 0; | 88 | actual_temp = -256; |
| 89 | else | 89 | else if (temp & 0x200) |
| 90 | actual_temp = (temp >> 1) & 0xff; | 90 | actual_temp = 255; |
| 91 | 91 | else if (temp & 0x100) { | |
| 92 | return actual_temp * 1000; | 92 | actual_temp = temp & 0x1ff; |
| 93 | actual_temp |= ~0x1ff; | ||
| 94 | } else | ||
| 95 | actual_temp = temp & 0xff; | ||
| 96 | |||
| 97 | return (actual_temp * 1000) / 2; | ||
| 93 | } | 98 | } |
| 94 | 99 | ||
| 95 | void rv770_pm_misc(struct radeon_device *rdev) | 100 | void rv770_pm_misc(struct radeon_device *rdev) |
diff --git a/drivers/gpu/stub/Kconfig b/drivers/gpu/stub/Kconfig index 09aea5f1556..70e60a4bb67 100644 --- a/drivers/gpu/stub/Kconfig +++ b/drivers/gpu/stub/Kconfig | |||
| @@ -1,11 +1,13 @@ | |||
| 1 | config STUB_POULSBO | 1 | config STUB_POULSBO |
| 2 | tristate "Intel GMA500 Stub Driver" | 2 | tristate "Intel GMA500 Stub Driver" |
| 3 | depends on PCI | 3 | depends on PCI |
| 4 | depends on NET # for THERMAL | ||
| 4 | # Poulsbo stub depends on ACPI_VIDEO when ACPI is enabled | 5 | # Poulsbo stub depends on ACPI_VIDEO when ACPI is enabled |
| 5 | # but for select to work, need to select ACPI_VIDEO's dependencies, ick | 6 | # but for select to work, need to select ACPI_VIDEO's dependencies, ick |
| 6 | select BACKLIGHT_CLASS_DEVICE if ACPI | 7 | select BACKLIGHT_CLASS_DEVICE if ACPI |
| 7 | select INPUT if ACPI | 8 | select INPUT if ACPI |
| 8 | select ACPI_VIDEO if ACPI | 9 | select ACPI_VIDEO if ACPI |
| 10 | select THERMAL if ACPI | ||
| 9 | help | 11 | help |
| 10 | Choose this option if you have a system that has Intel GMA500 | 12 | Choose this option if you have a system that has Intel GMA500 |
| 11 | (Poulsbo) integrated graphics. If M is selected, the module will | 13 | (Poulsbo) integrated graphics. If M is selected, the module will |
diff --git a/drivers/gpu/vga/vgaarb.c b/drivers/gpu/vga/vgaarb.c index c380c65da41..ace2b1623b2 100644 --- a/drivers/gpu/vga/vgaarb.c +++ b/drivers/gpu/vga/vgaarb.c | |||
| @@ -636,7 +636,7 @@ int vga_client_register(struct pci_dev *pdev, void *cookie, | |||
| 636 | void (*irq_set_state)(void *cookie, bool state), | 636 | void (*irq_set_state)(void *cookie, bool state), |
| 637 | unsigned int (*set_vga_decode)(void *cookie, bool decode)) | 637 | unsigned int (*set_vga_decode)(void *cookie, bool decode)) |
| 638 | { | 638 | { |
| 639 | int ret = -1; | 639 | int ret = -ENODEV; |
| 640 | struct vga_device *vgadev; | 640 | struct vga_device *vgadev; |
| 641 | unsigned long flags; | 641 | unsigned long flags; |
| 642 | 642 | ||
diff --git a/drivers/hwmon/applesmc.c b/drivers/hwmon/applesmc.c index ce0372f0615..4c0743660e9 100644 --- a/drivers/hwmon/applesmc.c +++ b/drivers/hwmon/applesmc.c | |||
| @@ -1072,6 +1072,7 @@ static int applesmc_create_nodes(struct applesmc_node_group *groups, int num) | |||
| 1072 | node->sda.dev_attr.show = grp->show; | 1072 | node->sda.dev_attr.show = grp->show; |
| 1073 | node->sda.dev_attr.store = grp->store; | 1073 | node->sda.dev_attr.store = grp->store; |
| 1074 | attr = &node->sda.dev_attr.attr; | 1074 | attr = &node->sda.dev_attr.attr; |
| 1075 | sysfs_attr_init(attr); | ||
| 1075 | attr->name = node->name; | 1076 | attr->name = node->name; |
| 1076 | attr->mode = S_IRUGO | (grp->store ? S_IWUSR : 0); | 1077 | attr->mode = S_IRUGO | (grp->store ? S_IWUSR : 0); |
| 1077 | ret = sysfs_create_file(&pdev->dev.kobj, attr); | 1078 | ret = sysfs_create_file(&pdev->dev.kobj, attr); |
diff --git a/drivers/hwmon/asus_atk0110.c b/drivers/hwmon/asus_atk0110.c index 2d68cf3c223..b5e892017e0 100644 --- a/drivers/hwmon/asus_atk0110.c +++ b/drivers/hwmon/asus_atk0110.c | |||
| @@ -13,6 +13,7 @@ | |||
| 13 | #include <linux/list.h> | 13 | #include <linux/list.h> |
| 14 | #include <linux/module.h> | 14 | #include <linux/module.h> |
| 15 | #include <linux/slab.h> | 15 | #include <linux/slab.h> |
| 16 | #include <linux/dmi.h> | ||
| 16 | 17 | ||
| 17 | #include <acpi/acpi.h> | 18 | #include <acpi/acpi.h> |
| 18 | #include <acpi/acpixf.h> | 19 | #include <acpi/acpixf.h> |
| @@ -22,6 +23,21 @@ | |||
| 22 | 23 | ||
| 23 | #define ATK_HID "ATK0110" | 24 | #define ATK_HID "ATK0110" |
| 24 | 25 | ||
| 26 | static bool new_if; | ||
| 27 | module_param(new_if, bool, 0); | ||
| 28 | MODULE_PARM_DESC(new_if, "Override detection heuristic and force the use of the new ATK0110 interface"); | ||
| 29 | |||
| 30 | static const struct dmi_system_id __initconst atk_force_new_if[] = { | ||
| 31 | { | ||
| 32 | /* Old interface has broken MCH temp monitoring */ | ||
| 33 | .ident = "Asus Sabertooth X58", | ||
| 34 | .matches = { | ||
| 35 | DMI_MATCH(DMI_BOARD_NAME, "SABERTOOTH X58") | ||
| 36 | } | ||
| 37 | }, | ||
| 38 | { } | ||
| 39 | }; | ||
| 40 | |||
| 25 | /* Minimum time between readings, enforced in order to avoid | 41 | /* Minimum time between readings, enforced in order to avoid |
| 26 | * hogging the CPU. | 42 | * hogging the CPU. |
| 27 | */ | 43 | */ |
| @@ -1302,7 +1318,9 @@ static int atk_probe_if(struct atk_data *data) | |||
| 1302 | * analysis of multiple DSDTs indicates that when both interfaces | 1318 | * analysis of multiple DSDTs indicates that when both interfaces |
| 1303 | * are present the new one (GGRP/GITM) is not functional. | 1319 | * are present the new one (GGRP/GITM) is not functional. |
| 1304 | */ | 1320 | */ |
| 1305 | if (data->rtmp_handle && data->rvlt_handle && data->rfan_handle) | 1321 | if (new_if) |
| 1322 | dev_info(dev, "Overriding interface detection\n"); | ||
| 1323 | if (data->rtmp_handle && data->rvlt_handle && data->rfan_handle && !new_if) | ||
| 1306 | data->old_interface = true; | 1324 | data->old_interface = true; |
| 1307 | else if (data->enumerate_handle && data->read_handle && | 1325 | else if (data->enumerate_handle && data->read_handle && |
| 1308 | data->write_handle) | 1326 | data->write_handle) |
| @@ -1420,6 +1438,9 @@ static int __init atk0110_init(void) | |||
| 1420 | return -EBUSY; | 1438 | return -EBUSY; |
| 1421 | } | 1439 | } |
| 1422 | 1440 | ||
| 1441 | if (dmi_check_system(atk_force_new_if)) | ||
| 1442 | new_if = true; | ||
| 1443 | |||
| 1423 | ret = acpi_bus_register_driver(&atk_driver); | 1444 | ret = acpi_bus_register_driver(&atk_driver); |
| 1424 | if (ret) | 1445 | if (ret) |
| 1425 | pr_info("acpi_bus_register_driver failed: %d\n", ret); | 1446 | pr_info("acpi_bus_register_driver failed: %d\n", ret); |
diff --git a/drivers/hwmon/lis3lv02d.c b/drivers/hwmon/lis3lv02d.c index 1b674b7d458..d805e8e5796 100644 --- a/drivers/hwmon/lis3lv02d.c +++ b/drivers/hwmon/lis3lv02d.c | |||
| @@ -957,7 +957,7 @@ int lis3lv02d_init_device(struct lis3lv02d *dev) | |||
| 957 | 957 | ||
| 958 | /* bail if we did not get an IRQ from the bus layer */ | 958 | /* bail if we did not get an IRQ from the bus layer */ |
| 959 | if (!dev->irq) { | 959 | if (!dev->irq) { |
| 960 | pr_err("No IRQ. Disabling /dev/freefall\n"); | 960 | pr_debug("No IRQ. Disabling /dev/freefall\n"); |
| 961 | goto out; | 961 | goto out; |
| 962 | } | 962 | } |
| 963 | 963 | ||
diff --git a/drivers/idle/intel_idle.c b/drivers/idle/intel_idle.c index 7acb32e7f81..1fa091e0569 100644 --- a/drivers/idle/intel_idle.c +++ b/drivers/idle/intel_idle.c | |||
| @@ -263,7 +263,7 @@ static void __setup_broadcast_timer(void *arg) | |||
| 263 | clockevents_notify(reason, &cpu); | 263 | clockevents_notify(reason, &cpu); |
| 264 | } | 264 | } |
| 265 | 265 | ||
| 266 | static int __cpuinit setup_broadcast_cpuhp_notify(struct notifier_block *n, | 266 | static int setup_broadcast_cpuhp_notify(struct notifier_block *n, |
| 267 | unsigned long action, void *hcpu) | 267 | unsigned long action, void *hcpu) |
| 268 | { | 268 | { |
| 269 | int hotcpu = (unsigned long)hcpu; | 269 | int hotcpu = (unsigned long)hcpu; |
| @@ -273,15 +273,11 @@ static int __cpuinit setup_broadcast_cpuhp_notify(struct notifier_block *n, | |||
| 273 | smp_call_function_single(hotcpu, __setup_broadcast_timer, | 273 | smp_call_function_single(hotcpu, __setup_broadcast_timer, |
| 274 | (void *)true, 1); | 274 | (void *)true, 1); |
| 275 | break; | 275 | break; |
| 276 | case CPU_DOWN_PREPARE: | ||
| 277 | smp_call_function_single(hotcpu, __setup_broadcast_timer, | ||
| 278 | (void *)false, 1); | ||
| 279 | break; | ||
| 280 | } | 276 | } |
| 281 | return NOTIFY_OK; | 277 | return NOTIFY_OK; |
| 282 | } | 278 | } |
| 283 | 279 | ||
| 284 | static struct notifier_block __cpuinitdata setup_broadcast_notifier = { | 280 | static struct notifier_block setup_broadcast_notifier = { |
| 285 | .notifier_call = setup_broadcast_cpuhp_notify, | 281 | .notifier_call = setup_broadcast_cpuhp_notify, |
| 286 | }; | 282 | }; |
| 287 | 283 | ||
diff --git a/drivers/infiniband/core/sa_query.c b/drivers/infiniband/core/sa_query.c index e38be1bcc01..fbbfa24cf57 100644 --- a/drivers/infiniband/core/sa_query.c +++ b/drivers/infiniband/core/sa_query.c | |||
| @@ -1079,7 +1079,7 @@ static void ib_sa_remove_one(struct ib_device *device) | |||
| 1079 | 1079 | ||
| 1080 | ib_unregister_event_handler(&sa_dev->event_handler); | 1080 | ib_unregister_event_handler(&sa_dev->event_handler); |
| 1081 | 1081 | ||
| 1082 | flush_scheduled_work(); | 1082 | flush_workqueue(ib_wq); |
| 1083 | 1083 | ||
| 1084 | for (i = 0; i <= sa_dev->end_port - sa_dev->start_port; ++i) { | 1084 | for (i = 0; i <= sa_dev->end_port - sa_dev->start_port; ++i) { |
| 1085 | if (rdma_port_get_link_layer(device, i + 1) == IB_LINK_LAYER_INFINIBAND) { | 1085 | if (rdma_port_get_link_layer(device, i + 1) == IB_LINK_LAYER_INFINIBAND) { |
diff --git a/drivers/infiniband/core/ucma.c b/drivers/infiniband/core/ucma.c index ca12acf3837..ec1e9da1488 100644 --- a/drivers/infiniband/core/ucma.c +++ b/drivers/infiniband/core/ucma.c | |||
| @@ -636,6 +636,16 @@ static void ucma_copy_iboe_route(struct rdma_ucm_query_route_resp *resp, | |||
| 636 | } | 636 | } |
| 637 | } | 637 | } |
| 638 | 638 | ||
| 639 | static void ucma_copy_iw_route(struct rdma_ucm_query_route_resp *resp, | ||
| 640 | struct rdma_route *route) | ||
| 641 | { | ||
| 642 | struct rdma_dev_addr *dev_addr; | ||
| 643 | |||
| 644 | dev_addr = &route->addr.dev_addr; | ||
| 645 | rdma_addr_get_dgid(dev_addr, (union ib_gid *) &resp->ib_route[0].dgid); | ||
| 646 | rdma_addr_get_sgid(dev_addr, (union ib_gid *) &resp->ib_route[0].sgid); | ||
| 647 | } | ||
| 648 | |||
| 639 | static ssize_t ucma_query_route(struct ucma_file *file, | 649 | static ssize_t ucma_query_route(struct ucma_file *file, |
| 640 | const char __user *inbuf, | 650 | const char __user *inbuf, |
| 641 | int in_len, int out_len) | 651 | int in_len, int out_len) |
| @@ -670,8 +680,10 @@ static ssize_t ucma_query_route(struct ucma_file *file, | |||
| 670 | 680 | ||
| 671 | resp.node_guid = (__force __u64) ctx->cm_id->device->node_guid; | 681 | resp.node_guid = (__force __u64) ctx->cm_id->device->node_guid; |
| 672 | resp.port_num = ctx->cm_id->port_num; | 682 | resp.port_num = ctx->cm_id->port_num; |
| 673 | if (rdma_node_get_transport(ctx->cm_id->device->node_type) == RDMA_TRANSPORT_IB) { | 683 | switch (rdma_node_get_transport(ctx->cm_id->device->node_type)) { |
| 674 | switch (rdma_port_get_link_layer(ctx->cm_id->device, ctx->cm_id->port_num)) { | 684 | case RDMA_TRANSPORT_IB: |
| 685 | switch (rdma_port_get_link_layer(ctx->cm_id->device, | ||
| 686 | ctx->cm_id->port_num)) { | ||
| 675 | case IB_LINK_LAYER_INFINIBAND: | 687 | case IB_LINK_LAYER_INFINIBAND: |
| 676 | ucma_copy_ib_route(&resp, &ctx->cm_id->route); | 688 | ucma_copy_ib_route(&resp, &ctx->cm_id->route); |
| 677 | break; | 689 | break; |
| @@ -681,6 +693,12 @@ static ssize_t ucma_query_route(struct ucma_file *file, | |||
| 681 | default: | 693 | default: |
| 682 | break; | 694 | break; |
| 683 | } | 695 | } |
| 696 | break; | ||
| 697 | case RDMA_TRANSPORT_IWARP: | ||
| 698 | ucma_copy_iw_route(&resp, &ctx->cm_id->route); | ||
| 699 | break; | ||
| 700 | default: | ||
| 701 | break; | ||
| 684 | } | 702 | } |
| 685 | 703 | ||
| 686 | out: | 704 | out: |
diff --git a/drivers/infiniband/hw/amso1100/c2_vq.c b/drivers/infiniband/hw/amso1100/c2_vq.c index 9ce7819b7b2..2ec716fb2ed 100644 --- a/drivers/infiniband/hw/amso1100/c2_vq.c +++ b/drivers/infiniband/hw/amso1100/c2_vq.c | |||
| @@ -107,7 +107,7 @@ struct c2_vq_req *vq_req_alloc(struct c2_dev *c2dev) | |||
| 107 | r = kmalloc(sizeof(struct c2_vq_req), GFP_KERNEL); | 107 | r = kmalloc(sizeof(struct c2_vq_req), GFP_KERNEL); |
| 108 | if (r) { | 108 | if (r) { |
| 109 | init_waitqueue_head(&r->wait_object); | 109 | init_waitqueue_head(&r->wait_object); |
| 110 | r->reply_msg = (u64) NULL; | 110 | r->reply_msg = 0; |
| 111 | r->event = 0; | 111 | r->event = 0; |
| 112 | r->cm_id = NULL; | 112 | r->cm_id = NULL; |
| 113 | r->qp = NULL; | 113 | r->qp = NULL; |
| @@ -123,7 +123,7 @@ struct c2_vq_req *vq_req_alloc(struct c2_dev *c2dev) | |||
| 123 | */ | 123 | */ |
| 124 | void vq_req_free(struct c2_dev *c2dev, struct c2_vq_req *r) | 124 | void vq_req_free(struct c2_dev *c2dev, struct c2_vq_req *r) |
| 125 | { | 125 | { |
| 126 | r->reply_msg = (u64) NULL; | 126 | r->reply_msg = 0; |
| 127 | if (atomic_dec_and_test(&r->refcnt)) { | 127 | if (atomic_dec_and_test(&r->refcnt)) { |
| 128 | kfree(r); | 128 | kfree(r); |
| 129 | } | 129 | } |
| @@ -151,7 +151,7 @@ void vq_req_get(struct c2_dev *c2dev, struct c2_vq_req *r) | |||
| 151 | void vq_req_put(struct c2_dev *c2dev, struct c2_vq_req *r) | 151 | void vq_req_put(struct c2_dev *c2dev, struct c2_vq_req *r) |
| 152 | { | 152 | { |
| 153 | if (atomic_dec_and_test(&r->refcnt)) { | 153 | if (atomic_dec_and_test(&r->refcnt)) { |
| 154 | if (r->reply_msg != (u64) NULL) | 154 | if (r->reply_msg != 0) |
| 155 | vq_repbuf_free(c2dev, | 155 | vq_repbuf_free(c2dev, |
| 156 | (void *) (unsigned long) r->reply_msg); | 156 | (void *) (unsigned long) r->reply_msg); |
| 157 | kfree(r); | 157 | kfree(r); |
diff --git a/drivers/infiniband/hw/cxgb4/cm.c b/drivers/infiniband/hw/cxgb4/cm.c index 0dc62b1438b..8b00e6c46f0 100644 --- a/drivers/infiniband/hw/cxgb4/cm.c +++ b/drivers/infiniband/hw/cxgb4/cm.c | |||
| @@ -380,7 +380,7 @@ static void send_flowc(struct c4iw_ep *ep, struct sk_buff *skb) | |||
| 380 | 16)) | FW_WR_FLOWID(ep->hwtid)); | 380 | 16)) | FW_WR_FLOWID(ep->hwtid)); |
| 381 | 381 | ||
| 382 | flowc->mnemval[0].mnemonic = FW_FLOWC_MNEM_PFNVFN; | 382 | flowc->mnemval[0].mnemonic = FW_FLOWC_MNEM_PFNVFN; |
| 383 | flowc->mnemval[0].val = cpu_to_be32(0); | 383 | flowc->mnemval[0].val = cpu_to_be32(PCI_FUNC(ep->com.dev->rdev.lldi.pdev->devfn) << 8); |
| 384 | flowc->mnemval[1].mnemonic = FW_FLOWC_MNEM_CH; | 384 | flowc->mnemval[1].mnemonic = FW_FLOWC_MNEM_CH; |
| 385 | flowc->mnemval[1].val = cpu_to_be32(ep->tx_chan); | 385 | flowc->mnemval[1].val = cpu_to_be32(ep->tx_chan); |
| 386 | flowc->mnemval[2].mnemonic = FW_FLOWC_MNEM_PORT; | 386 | flowc->mnemval[2].mnemonic = FW_FLOWC_MNEM_PORT; |
diff --git a/drivers/infiniband/hw/cxgb4/qp.c b/drivers/infiniband/hw/cxgb4/qp.c index 20800900ef3..4f0be25cab1 100644 --- a/drivers/infiniband/hw/cxgb4/qp.c +++ b/drivers/infiniband/hw/cxgb4/qp.c | |||
| @@ -220,7 +220,7 @@ static int create_qp(struct c4iw_rdev *rdev, struct t4_wq *wq, | |||
| 220 | V_FW_RI_RES_WR_DCAEN(0) | | 220 | V_FW_RI_RES_WR_DCAEN(0) | |
| 221 | V_FW_RI_RES_WR_DCACPU(0) | | 221 | V_FW_RI_RES_WR_DCACPU(0) | |
| 222 | V_FW_RI_RES_WR_FBMIN(2) | | 222 | V_FW_RI_RES_WR_FBMIN(2) | |
| 223 | V_FW_RI_RES_WR_FBMAX(3) | | 223 | V_FW_RI_RES_WR_FBMAX(2) | |
| 224 | V_FW_RI_RES_WR_CIDXFTHRESHO(0) | | 224 | V_FW_RI_RES_WR_CIDXFTHRESHO(0) | |
| 225 | V_FW_RI_RES_WR_CIDXFTHRESH(0) | | 225 | V_FW_RI_RES_WR_CIDXFTHRESH(0) | |
| 226 | V_FW_RI_RES_WR_EQSIZE(eqsize)); | 226 | V_FW_RI_RES_WR_EQSIZE(eqsize)); |
| @@ -243,7 +243,7 @@ static int create_qp(struct c4iw_rdev *rdev, struct t4_wq *wq, | |||
| 243 | V_FW_RI_RES_WR_DCAEN(0) | | 243 | V_FW_RI_RES_WR_DCAEN(0) | |
| 244 | V_FW_RI_RES_WR_DCACPU(0) | | 244 | V_FW_RI_RES_WR_DCACPU(0) | |
| 245 | V_FW_RI_RES_WR_FBMIN(2) | | 245 | V_FW_RI_RES_WR_FBMIN(2) | |
| 246 | V_FW_RI_RES_WR_FBMAX(3) | | 246 | V_FW_RI_RES_WR_FBMAX(2) | |
| 247 | V_FW_RI_RES_WR_CIDXFTHRESHO(0) | | 247 | V_FW_RI_RES_WR_CIDXFTHRESHO(0) | |
| 248 | V_FW_RI_RES_WR_CIDXFTHRESH(0) | | 248 | V_FW_RI_RES_WR_CIDXFTHRESH(0) | |
| 249 | V_FW_RI_RES_WR_EQSIZE(eqsize)); | 249 | V_FW_RI_RES_WR_EQSIZE(eqsize)); |
diff --git a/drivers/infiniband/hw/qib/qib_iba7322.c b/drivers/infiniband/hw/qib/qib_iba7322.c index 50cceb3ab88..b01809a82cb 100644 --- a/drivers/infiniband/hw/qib/qib_iba7322.c +++ b/drivers/infiniband/hw/qib/qib_iba7322.c | |||
| @@ -623,7 +623,6 @@ struct qib_chippport_specific { | |||
| 623 | u8 ibmalfusesnap; | 623 | u8 ibmalfusesnap; |
| 624 | struct qib_qsfp_data qsfp_data; | 624 | struct qib_qsfp_data qsfp_data; |
| 625 | char epmsgbuf[192]; /* for port error interrupt msg buffer */ | 625 | char epmsgbuf[192]; /* for port error interrupt msg buffer */ |
| 626 | u8 bounced; | ||
| 627 | }; | 626 | }; |
| 628 | 627 | ||
| 629 | static struct { | 628 | static struct { |
| @@ -1881,23 +1880,7 @@ static noinline void handle_7322_p_errors(struct qib_pportdata *ppd) | |||
| 1881 | IB_PHYSPORTSTATE_DISABLED) | 1880 | IB_PHYSPORTSTATE_DISABLED) |
| 1882 | qib_set_ib_7322_lstate(ppd, 0, | 1881 | qib_set_ib_7322_lstate(ppd, 0, |
| 1883 | QLOGIC_IB_IBCC_LINKINITCMD_DISABLE); | 1882 | QLOGIC_IB_IBCC_LINKINITCMD_DISABLE); |
| 1884 | else { | 1883 | else |
| 1885 | u32 lstate; | ||
| 1886 | /* | ||
| 1887 | * We need the current logical link state before | ||
| 1888 | * lflags are set in handle_e_ibstatuschanged. | ||
| 1889 | */ | ||
| 1890 | lstate = qib_7322_iblink_state(ibcs); | ||
| 1891 | |||
| 1892 | if (IS_QMH(dd) && !ppd->cpspec->bounced && | ||
| 1893 | ltstate == IB_PHYSPORTSTATE_LINKUP && | ||
| 1894 | (lstate >= IB_PORT_INIT && | ||
| 1895 | lstate <= IB_PORT_ACTIVE)) { | ||
| 1896 | ppd->cpspec->bounced = 1; | ||
| 1897 | qib_7322_set_ib_cfg(ppd, QIB_IB_CFG_LSTATE, | ||
| 1898 | IB_LINKCMD_DOWN | IB_LINKINITCMD_POLL); | ||
| 1899 | } | ||
| 1900 | |||
| 1901 | /* | 1884 | /* |
| 1902 | * Since going into a recovery state causes the link | 1885 | * Since going into a recovery state causes the link |
| 1903 | * state to go down and since recovery is transitory, | 1886 | * state to go down and since recovery is transitory, |
| @@ -1911,7 +1894,6 @@ static noinline void handle_7322_p_errors(struct qib_pportdata *ppd) | |||
| 1911 | ltstate != IB_PHYSPORTSTATE_RECOVERY_WAITRMT && | 1894 | ltstate != IB_PHYSPORTSTATE_RECOVERY_WAITRMT && |
| 1912 | ltstate != IB_PHYSPORTSTATE_RECOVERY_IDLE) | 1895 | ltstate != IB_PHYSPORTSTATE_RECOVERY_IDLE) |
| 1913 | qib_handle_e_ibstatuschanged(ppd, ibcs); | 1896 | qib_handle_e_ibstatuschanged(ppd, ibcs); |
| 1914 | } | ||
| 1915 | } | 1897 | } |
| 1916 | if (*msg && iserr) | 1898 | if (*msg && iserr) |
| 1917 | qib_dev_porterr(dd, ppd->port, "%s error\n", msg); | 1899 | qib_dev_porterr(dd, ppd->port, "%s error\n", msg); |
| @@ -2381,6 +2363,11 @@ static int qib_7322_bringup_serdes(struct qib_pportdata *ppd) | |||
| 2381 | qib_write_kreg_port(ppd, krp_rcvctrl, ppd->p_rcvctrl); | 2363 | qib_write_kreg_port(ppd, krp_rcvctrl, ppd->p_rcvctrl); |
| 2382 | spin_unlock_irqrestore(&dd->cspec->rcvmod_lock, flags); | 2364 | spin_unlock_irqrestore(&dd->cspec->rcvmod_lock, flags); |
| 2383 | 2365 | ||
| 2366 | /* Hold the link state machine for mezz boards */ | ||
| 2367 | if (IS_QMH(dd) || IS_QME(dd)) | ||
| 2368 | qib_set_ib_7322_lstate(ppd, 0, | ||
| 2369 | QLOGIC_IB_IBCC_LINKINITCMD_DISABLE); | ||
| 2370 | |||
| 2384 | /* Also enable IBSTATUSCHG interrupt. */ | 2371 | /* Also enable IBSTATUSCHG interrupt. */ |
| 2385 | val = qib_read_kreg_port(ppd, krp_errmask); | 2372 | val = qib_read_kreg_port(ppd, krp_errmask); |
| 2386 | qib_write_kreg_port(ppd, krp_errmask, | 2373 | qib_write_kreg_port(ppd, krp_errmask, |
| @@ -5702,6 +5689,11 @@ static void set_no_qsfp_atten(struct qib_devdata *dd, int change) | |||
| 5702 | ppd->cpspec->h1_val = h1; | 5689 | ppd->cpspec->h1_val = h1; |
| 5703 | /* now change the IBC and serdes, overriding generic */ | 5690 | /* now change the IBC and serdes, overriding generic */ |
| 5704 | init_txdds_table(ppd, 1); | 5691 | init_txdds_table(ppd, 1); |
| 5692 | /* Re-enable the physical state machine on mezz boards | ||
| 5693 | * now that the correct settings have been set. */ | ||
| 5694 | if (IS_QMH(dd) || IS_QME(dd)) | ||
| 5695 | qib_set_ib_7322_lstate(ppd, 0, | ||
| 5696 | QLOGIC_IB_IBCC_LINKINITCMD_SLEEP); | ||
| 5705 | any++; | 5697 | any++; |
| 5706 | } | 5698 | } |
| 5707 | if (*nxt == '\n') | 5699 | if (*nxt == '\n') |
diff --git a/drivers/input/keyboard/Kconfig b/drivers/input/keyboard/Kconfig index 417507348ba..c7a92028f45 100644 --- a/drivers/input/keyboard/Kconfig +++ b/drivers/input/keyboard/Kconfig | |||
| @@ -343,6 +343,16 @@ config KEYBOARD_NOMADIK | |||
| 343 | To compile this driver as a module, choose M here: the | 343 | To compile this driver as a module, choose M here: the |
| 344 | module will be called nmk-ske-keypad. | 344 | module will be called nmk-ske-keypad. |
| 345 | 345 | ||
| 346 | config KEYBOARD_TEGRA | ||
| 347 | tristate "NVIDIA Tegra internal matrix keyboard controller support" | ||
| 348 | depends on ARCH_TEGRA | ||
| 349 | help | ||
| 350 | Say Y here if you want to use a matrix keyboard connected directly | ||
| 351 | to the internal keyboard controller on Tegra SoCs. | ||
| 352 | |||
| 353 | To compile this driver as a module, choose M here: the | ||
| 354 | module will be called tegra-kbc. | ||
| 355 | |||
| 346 | config KEYBOARD_OPENCORES | 356 | config KEYBOARD_OPENCORES |
| 347 | tristate "OpenCores Keyboard Controller" | 357 | tristate "OpenCores Keyboard Controller" |
| 348 | help | 358 | help |
diff --git a/drivers/input/keyboard/Makefile b/drivers/input/keyboard/Makefile index 4e5571b72cd..468c627a284 100644 --- a/drivers/input/keyboard/Makefile +++ b/drivers/input/keyboard/Makefile | |||
| @@ -42,6 +42,7 @@ obj-$(CONFIG_KEYBOARD_STMPE) += stmpe-keypad.o | |||
| 42 | obj-$(CONFIG_KEYBOARD_STOWAWAY) += stowaway.o | 42 | obj-$(CONFIG_KEYBOARD_STOWAWAY) += stowaway.o |
| 43 | obj-$(CONFIG_KEYBOARD_SUNKBD) += sunkbd.o | 43 | obj-$(CONFIG_KEYBOARD_SUNKBD) += sunkbd.o |
| 44 | obj-$(CONFIG_KEYBOARD_TC3589X) += tc3589x-keypad.o | 44 | obj-$(CONFIG_KEYBOARD_TC3589X) += tc3589x-keypad.o |
| 45 | obj-$(CONFIG_KEYBOARD_TEGRA) += tegra-kbc.o | ||
| 45 | obj-$(CONFIG_KEYBOARD_TNETV107X) += tnetv107x-keypad.o | 46 | obj-$(CONFIG_KEYBOARD_TNETV107X) += tnetv107x-keypad.o |
| 46 | obj-$(CONFIG_KEYBOARD_TWL4030) += twl4030_keypad.o | 47 | obj-$(CONFIG_KEYBOARD_TWL4030) += twl4030_keypad.o |
| 47 | obj-$(CONFIG_KEYBOARD_XTKBD) += xtkbd.o | 48 | obj-$(CONFIG_KEYBOARD_XTKBD) += xtkbd.o |
diff --git a/drivers/input/keyboard/gpio_keys.c b/drivers/input/keyboard/gpio_keys.c index 6069abe31e4..eb3006361ee 100644 --- a/drivers/input/keyboard/gpio_keys.c +++ b/drivers/input/keyboard/gpio_keys.c | |||
| @@ -322,7 +322,7 @@ static void gpio_keys_report_event(struct gpio_button_data *bdata) | |||
| 322 | struct gpio_keys_button *button = bdata->button; | 322 | struct gpio_keys_button *button = bdata->button; |
| 323 | struct input_dev *input = bdata->input; | 323 | struct input_dev *input = bdata->input; |
| 324 | unsigned int type = button->type ?: EV_KEY; | 324 | unsigned int type = button->type ?: EV_KEY; |
| 325 | int state = (gpio_get_value(button->gpio) ? 1 : 0) ^ button->active_low; | 325 | int state = (gpio_get_value_cansleep(button->gpio) ? 1 : 0) ^ button->active_low; |
| 326 | 326 | ||
| 327 | input_event(input, type, button->code, !!state); | 327 | input_event(input, type, button->code, !!state); |
| 328 | input_sync(input); | 328 | input_sync(input); |
| @@ -410,8 +410,8 @@ static int __devinit gpio_keys_setup_key(struct platform_device *pdev, | |||
| 410 | if (!button->can_disable) | 410 | if (!button->can_disable) |
| 411 | irqflags |= IRQF_SHARED; | 411 | irqflags |= IRQF_SHARED; |
| 412 | 412 | ||
| 413 | error = request_irq(irq, gpio_keys_isr, irqflags, desc, bdata); | 413 | error = request_any_context_irq(irq, gpio_keys_isr, irqflags, desc, bdata); |
| 414 | if (error) { | 414 | if (error < 0) { |
| 415 | dev_err(dev, "Unable to claim irq %d; error %d\n", | 415 | dev_err(dev, "Unable to claim irq %d; error %d\n", |
| 416 | irq, error); | 416 | irq, error); |
| 417 | goto fail3; | 417 | goto fail3; |
diff --git a/drivers/input/keyboard/tegra-kbc.c b/drivers/input/keyboard/tegra-kbc.c new file mode 100644 index 00000000000..ac471b77c18 --- /dev/null +++ b/drivers/input/keyboard/tegra-kbc.c | |||
| @@ -0,0 +1,727 @@ | |||
| 1 | /* | ||
| 2 | * Keyboard class input driver for the NVIDIA Tegra SoC internal matrix | ||
| 3 | * keyboard controller | ||
| 4 | * | ||
| 5 | * Copyright (c) 2009-2011, NVIDIA Corporation. | ||
| 6 | * | ||
| 7 | * This program is free software; you can redistribute it and/or modify | ||
| 8 | * it under the terms of the GNU General Public License as published by | ||
| 9 | * the Free Software Foundation; either version 2 of the License, or | ||
| 10 | * (at your option) any later version. | ||
| 11 | * | ||
| 12 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
| 13 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
| 14 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
| 15 | * more details. | ||
| 16 | * | ||
| 17 | * You should have received a copy of the GNU General Public License along | ||
| 18 | * with this program; if not, write to the Free Software Foundation, Inc., | ||
| 19 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | ||
| 20 | */ | ||
| 21 | |||
| 22 | #include <linux/module.h> | ||
| 23 | #include <linux/input.h> | ||
| 24 | #include <linux/platform_device.h> | ||
| 25 | #include <linux/delay.h> | ||
| 26 | #include <linux/io.h> | ||
| 27 | #include <linux/interrupt.h> | ||
| 28 | #include <linux/clk.h> | ||
| 29 | #include <linux/slab.h> | ||
| 30 | #include <mach/clk.h> | ||
| 31 | #include <mach/kbc.h> | ||
| 32 | |||
| 33 | #define KBC_MAX_DEBOUNCE_CNT 0x3ffu | ||
| 34 | |||
| 35 | /* KBC row scan time and delay for beginning the row scan. */ | ||
| 36 | #define KBC_ROW_SCAN_TIME 16 | ||
| 37 | #define KBC_ROW_SCAN_DLY 5 | ||
| 38 | |||
| 39 | /* KBC uses a 32KHz clock so a cycle = 1/32Khz */ | ||
| 40 | #define KBC_CYCLE_USEC 32 | ||
| 41 | |||
| 42 | /* KBC Registers */ | ||
| 43 | |||
| 44 | /* KBC Control Register */ | ||
| 45 | #define KBC_CONTROL_0 0x0 | ||
| 46 | #define KBC_FIFO_TH_CNT_SHIFT(cnt) (cnt << 14) | ||
| 47 | #define KBC_DEBOUNCE_CNT_SHIFT(cnt) (cnt << 4) | ||
| 48 | #define KBC_CONTROL_FIFO_CNT_INT_EN (1 << 3) | ||
| 49 | #define KBC_CONTROL_KBC_EN (1 << 0) | ||
| 50 | |||
| 51 | /* KBC Interrupt Register */ | ||
| 52 | #define KBC_INT_0 0x4 | ||
| 53 | #define KBC_INT_FIFO_CNT_INT_STATUS (1 << 2) | ||
| 54 | |||
| 55 | #define KBC_ROW_CFG0_0 0x8 | ||
| 56 | #define KBC_COL_CFG0_0 0x18 | ||
| 57 | #define KBC_INIT_DLY_0 0x28 | ||
| 58 | #define KBC_RPT_DLY_0 0x2c | ||
| 59 | #define KBC_KP_ENT0_0 0x30 | ||
| 60 | #define KBC_KP_ENT1_0 0x34 | ||
| 61 | #define KBC_ROW0_MASK_0 0x38 | ||
| 62 | |||
| 63 | #define KBC_ROW_SHIFT 3 | ||
| 64 | |||
| 65 | struct tegra_kbc { | ||
| 66 | void __iomem *mmio; | ||
| 67 | struct input_dev *idev; | ||
| 68 | unsigned int irq; | ||
| 69 | unsigned int wake_enable_rows; | ||
| 70 | unsigned int wake_enable_cols; | ||
| 71 | spinlock_t lock; | ||
| 72 | unsigned int repoll_dly; | ||
| 73 | unsigned long cp_dly_jiffies; | ||
| 74 | const struct tegra_kbc_platform_data *pdata; | ||
| 75 | unsigned short keycode[KBC_MAX_KEY]; | ||
| 76 | unsigned short current_keys[KBC_MAX_KPENT]; | ||
| 77 | unsigned int num_pressed_keys; | ||
| 78 | struct timer_list timer; | ||
| 79 | struct clk *clk; | ||
| 80 | }; | ||
| 81 | |||
| 82 | static const u32 tegra_kbc_default_keymap[] = { | ||
| 83 | KEY(0, 2, KEY_W), | ||
| 84 | KEY(0, 3, KEY_S), | ||
| 85 | KEY(0, 4, KEY_A), | ||
| 86 | KEY(0, 5, KEY_Z), | ||
| 87 | KEY(0, 7, KEY_FN), | ||
| 88 | |||
| 89 | KEY(1, 7, KEY_LEFTMETA), | ||
| 90 | |||
| 91 | KEY(2, 6, KEY_RIGHTALT), | ||
| 92 | KEY(2, 7, KEY_LEFTALT), | ||
| 93 | |||
| 94 | KEY(3, 0, KEY_5), | ||
| 95 | KEY(3, 1, KEY_4), | ||
| 96 | KEY(3, 2, KEY_R), | ||
| 97 | KEY(3, 3, KEY_E), | ||
| 98 | KEY(3, 4, KEY_F), | ||
| 99 | KEY(3, 5, KEY_D), | ||
| 100 | KEY(3, 6, KEY_X), | ||
| 101 | |||
| 102 | KEY(4, 0, KEY_7), | ||
| 103 | KEY(4, 1, KEY_6), | ||
| 104 | KEY(4, 2, KEY_T), | ||
| 105 | KEY(4, 3, KEY_H), | ||
| 106 | KEY(4, 4, KEY_G), | ||
| 107 | KEY(4, 5, KEY_V), | ||
| 108 | KEY(4, 6, KEY_C), | ||
| 109 | KEY(4, 7, KEY_SPACE), | ||
| 110 | |||
| 111 | KEY(5, 0, KEY_9), | ||
| 112 | KEY(5, 1, KEY_8), | ||
| 113 | KEY(5, 2, KEY_U), | ||
| 114 | KEY(5, 3, KEY_Y), | ||
| 115 | KEY(5, 4, KEY_J), | ||
| 116 | KEY(5, 5, KEY_N), | ||
| 117 | KEY(5, 6, KEY_B), | ||
| 118 | KEY(5, 7, KEY_BACKSLASH), | ||
| 119 | |||
| 120 | KEY(6, 0, KEY_MINUS), | ||
| 121 | KEY(6, 1, KEY_0), | ||
| 122 | KEY(6, 2, KEY_O), | ||
| 123 | KEY(6, 3, KEY_I), | ||
| 124 | KEY(6, 4, KEY_L), | ||
| 125 | KEY(6, 5, KEY_K), | ||
| 126 | KEY(6, 6, KEY_COMMA), | ||
| 127 | KEY(6, 7, KEY_M), | ||
| 128 | |||
| 129 | KEY(7, 1, KEY_EQUAL), | ||
| 130 | KEY(7, 2, KEY_RIGHTBRACE), | ||
| 131 | KEY(7, 3, KEY_ENTER), | ||
| 132 | KEY(7, 7, KEY_MENU), | ||
| 133 | |||
| 134 | KEY(8, 4, KEY_RIGHTSHIFT), | ||
| 135 | KEY(8, 5, KEY_LEFTSHIFT), | ||
| 136 | |||
| 137 | KEY(9, 5, KEY_RIGHTCTRL), | ||
| 138 | KEY(9, 7, KEY_LEFTCTRL), | ||
| 139 | |||
| 140 | KEY(11, 0, KEY_LEFTBRACE), | ||
| 141 | KEY(11, 1, KEY_P), | ||
| 142 | KEY(11, 2, KEY_APOSTROPHE), | ||
| 143 | KEY(11, 3, KEY_SEMICOLON), | ||
| 144 | KEY(11, 4, KEY_SLASH), | ||
| 145 | KEY(11, 5, KEY_DOT), | ||
| 146 | |||
| 147 | KEY(12, 0, KEY_F10), | ||
| 148 | KEY(12, 1, KEY_F9), | ||
| 149 | KEY(12, 2, KEY_BACKSPACE), | ||
| 150 | KEY(12, 3, KEY_3), | ||
| 151 | KEY(12, 4, KEY_2), | ||
| 152 | KEY(12, 5, KEY_UP), | ||
| 153 | KEY(12, 6, KEY_PRINT), | ||
| 154 | KEY(12, 7, KEY_PAUSE), | ||
| 155 | |||
| 156 | KEY(13, 0, KEY_INSERT), | ||
| 157 | KEY(13, 1, KEY_DELETE), | ||
| 158 | KEY(13, 3, KEY_PAGEUP), | ||
| 159 | KEY(13, 4, KEY_PAGEDOWN), | ||
| 160 | KEY(13, 5, KEY_RIGHT), | ||
| 161 | KEY(13, 6, KEY_DOWN), | ||
| 162 | KEY(13, 7, KEY_LEFT), | ||
| 163 | |||
| 164 | KEY(14, 0, KEY_F11), | ||
| 165 | KEY(14, 1, KEY_F12), | ||
| 166 | KEY(14, 2, KEY_F8), | ||
| 167 | KEY(14, 3, KEY_Q), | ||
| 168 | KEY(14, 4, KEY_F4), | ||
| 169 | KEY(14, 5, KEY_F3), | ||
| 170 | KEY(14, 6, KEY_1), | ||
| 171 | KEY(14, 7, KEY_F7), | ||
| 172 | |||
| 173 | KEY(15, 0, KEY_ESC), | ||
| 174 | KEY(15, 1, KEY_GRAVE), | ||
| 175 | KEY(15, 2, KEY_F5), | ||
| 176 | KEY(15, 3, KEY_TAB), | ||
| 177 | KEY(15, 4, KEY_F1), | ||
| 178 | KEY(15, 5, KEY_F2), | ||
| 179 | KEY(15, 6, KEY_CAPSLOCK), | ||
| 180 | KEY(15, 7, KEY_F6), | ||
| 181 | }; | ||
| 182 | |||
| 183 | static const struct matrix_keymap_data tegra_kbc_default_keymap_data = { | ||
| 184 | .keymap = tegra_kbc_default_keymap, | ||
| 185 | .keymap_size = ARRAY_SIZE(tegra_kbc_default_keymap), | ||
| 186 | }; | ||
| 187 | |||
| 188 | static void tegra_kbc_report_released_keys(struct input_dev *input, | ||
| 189 | unsigned short old_keycodes[], | ||
| 190 | unsigned int old_num_keys, | ||
| 191 | unsigned short new_keycodes[], | ||
| 192 | unsigned int new_num_keys) | ||
| 193 | { | ||
| 194 | unsigned int i, j; | ||
| 195 | |||
| 196 | for (i = 0; i < old_num_keys; i++) { | ||
| 197 | for (j = 0; j < new_num_keys; j++) | ||
| 198 | if (old_keycodes[i] == new_keycodes[j]) | ||
| 199 | break; | ||
| 200 | |||
| 201 | if (j == new_num_keys) | ||
| 202 | input_report_key(input, old_keycodes[i], 0); | ||
| 203 | } | ||
| 204 | } | ||
| 205 | |||
| 206 | static void tegra_kbc_report_pressed_keys(struct input_dev *input, | ||
| 207 | unsigned char scancodes[], | ||
| 208 | unsigned short keycodes[], | ||
| 209 | unsigned int num_pressed_keys) | ||
| 210 | { | ||
| 211 | unsigned int i; | ||
| 212 | |||
| 213 | for (i = 0; i < num_pressed_keys; i++) { | ||
| 214 | input_event(input, EV_MSC, MSC_SCAN, scancodes[i]); | ||
| 215 | input_report_key(input, keycodes[i], 1); | ||
| 216 | } | ||
| 217 | } | ||
| 218 | |||
| 219 | static void tegra_kbc_report_keys(struct tegra_kbc *kbc) | ||
| 220 | { | ||
| 221 | unsigned char scancodes[KBC_MAX_KPENT]; | ||
| 222 | unsigned short keycodes[KBC_MAX_KPENT]; | ||
| 223 | u32 val = 0; | ||
| 224 | unsigned int i; | ||
| 225 | unsigned int num_down = 0; | ||
| 226 | unsigned long flags; | ||
| 227 | |||
| 228 | spin_lock_irqsave(&kbc->lock, flags); | ||
| 229 | for (i = 0; i < KBC_MAX_KPENT; i++) { | ||
| 230 | if ((i % 4) == 0) | ||
| 231 | val = readl(kbc->mmio + KBC_KP_ENT0_0 + i); | ||
| 232 | |||
| 233 | if (val & 0x80) { | ||
| 234 | unsigned int col = val & 0x07; | ||
| 235 | unsigned int row = (val >> 3) & 0x0f; | ||
| 236 | unsigned char scancode = | ||
| 237 | MATRIX_SCAN_CODE(row, col, KBC_ROW_SHIFT); | ||
| 238 | |||
| 239 | scancodes[num_down] = scancode; | ||
| 240 | keycodes[num_down++] = kbc->keycode[scancode]; | ||
| 241 | } | ||
| 242 | |||
| 243 | val >>= 8; | ||
| 244 | } | ||
| 245 | spin_unlock_irqrestore(&kbc->lock, flags); | ||
| 246 | |||
| 247 | tegra_kbc_report_released_keys(kbc->idev, | ||
| 248 | kbc->current_keys, kbc->num_pressed_keys, | ||
| 249 | keycodes, num_down); | ||
| 250 | tegra_kbc_report_pressed_keys(kbc->idev, scancodes, keycodes, num_down); | ||
| 251 | input_sync(kbc->idev); | ||
| 252 | |||
| 253 | memcpy(kbc->current_keys, keycodes, sizeof(kbc->current_keys)); | ||
| 254 | kbc->num_pressed_keys = num_down; | ||
| 255 | } | ||
| 256 | |||
| 257 | static void tegra_kbc_keypress_timer(unsigned long data) | ||
| 258 | { | ||
| 259 | struct tegra_kbc *kbc = (struct tegra_kbc *)data; | ||
| 260 | unsigned long flags; | ||
| 261 | u32 val; | ||
| 262 | unsigned int i; | ||
| 263 | |||
| 264 | val = (readl(kbc->mmio + KBC_INT_0) >> 4) & 0xf; | ||
| 265 | if (val) { | ||
| 266 | unsigned long dly; | ||
| 267 | |||
| 268 | tegra_kbc_report_keys(kbc); | ||
| 269 | |||
| 270 | /* | ||
| 271 | * If more than one keys are pressed we need not wait | ||
| 272 | * for the repoll delay. | ||
| 273 | */ | ||
| 274 | dly = (val == 1) ? kbc->repoll_dly : 1; | ||
| 275 | mod_timer(&kbc->timer, jiffies + msecs_to_jiffies(dly)); | ||
| 276 | } else { | ||
| 277 | /* Release any pressed keys and exit the polling loop */ | ||
| 278 | for (i = 0; i < kbc->num_pressed_keys; i++) | ||
| 279 | input_report_key(kbc->idev, kbc->current_keys[i], 0); | ||
| 280 | input_sync(kbc->idev); | ||
| 281 | |||
| 282 | kbc->num_pressed_keys = 0; | ||
| 283 | |||
| 284 | /* All keys are released so enable the keypress interrupt */ | ||
| 285 | spin_lock_irqsave(&kbc->lock, flags); | ||
| 286 | val = readl(kbc->mmio + KBC_CONTROL_0); | ||
| 287 | val |= KBC_CONTROL_FIFO_CNT_INT_EN; | ||
| 288 | writel(val, kbc->mmio + KBC_CONTROL_0); | ||
| 289 | spin_unlock_irqrestore(&kbc->lock, flags); | ||
| 290 | } | ||
| 291 | } | ||
| 292 | |||
| 293 | static irqreturn_t tegra_kbc_isr(int irq, void *args) | ||
| 294 | { | ||
| 295 | struct tegra_kbc *kbc = args; | ||
| 296 | u32 val, ctl; | ||
| 297 | |||
| 298 | /* | ||
| 299 | * Until all keys are released, defer further processing to | ||
| 300 | * the polling loop in tegra_kbc_keypress_timer | ||
| 301 | */ | ||
| 302 | ctl = readl(kbc->mmio + KBC_CONTROL_0); | ||
| 303 | ctl &= ~KBC_CONTROL_FIFO_CNT_INT_EN; | ||
| 304 | writel(ctl, kbc->mmio + KBC_CONTROL_0); | ||
| 305 | |||
| 306 | /* | ||
| 307 | * Quickly bail out & reenable interrupts if the fifo threshold | ||
| 308 | * count interrupt wasn't the interrupt source | ||
| 309 | */ | ||
| 310 | val = readl(kbc->mmio + KBC_INT_0); | ||
| 311 | writel(val, kbc->mmio + KBC_INT_0); | ||
| 312 | |||
| 313 | if (val & KBC_INT_FIFO_CNT_INT_STATUS) { | ||
| 314 | /* | ||
| 315 | * Schedule timer to run when hardware is in continuous | ||
| 316 | * polling mode. | ||
| 317 | */ | ||
| 318 | mod_timer(&kbc->timer, jiffies + kbc->cp_dly_jiffies); | ||
| 319 | } else { | ||
| 320 | ctl |= KBC_CONTROL_FIFO_CNT_INT_EN; | ||
| 321 | writel(ctl, kbc->mmio + KBC_CONTROL_0); | ||
| 322 | } | ||
| 323 | |||
| 324 | return IRQ_HANDLED; | ||
| 325 | } | ||
| 326 | |||
| 327 | static void tegra_kbc_setup_wakekeys(struct tegra_kbc *kbc, bool filter) | ||
| 328 | { | ||
| 329 | const struct tegra_kbc_platform_data *pdata = kbc->pdata; | ||
| 330 | int i; | ||
| 331 | unsigned int rst_val; | ||
| 332 | |||
| 333 | BUG_ON(pdata->wake_cnt > KBC_MAX_KEY); | ||
| 334 | rst_val = (filter && pdata->wake_cnt) ? ~0 : 0; | ||
| 335 | |||
| 336 | for (i = 0; i < KBC_MAX_ROW; i++) | ||
| 337 | writel(rst_val, kbc->mmio + KBC_ROW0_MASK_0 + i * 4); | ||
| 338 | |||
| 339 | if (filter) { | ||
| 340 | for (i = 0; i < pdata->wake_cnt; i++) { | ||
| 341 | u32 val, addr; | ||
| 342 | addr = pdata->wake_cfg[i].row * 4 + KBC_ROW0_MASK_0; | ||
| 343 | val = readl(kbc->mmio + addr); | ||
| 344 | val &= ~(1 << pdata->wake_cfg[i].col); | ||
| 345 | writel(val, kbc->mmio + addr); | ||
| 346 | } | ||
| 347 | } | ||
| 348 | } | ||
| 349 | |||
| 350 | static void tegra_kbc_config_pins(struct tegra_kbc *kbc) | ||
| 351 | { | ||
| 352 | const struct tegra_kbc_platform_data *pdata = kbc->pdata; | ||
| 353 | int i; | ||
| 354 | |||
| 355 | for (i = 0; i < KBC_MAX_GPIO; i++) { | ||
| 356 | u32 r_shft = 5 * (i % 6); | ||
| 357 | u32 c_shft = 4 * (i % 8); | ||
| 358 | u32 r_mask = 0x1f << r_shft; | ||
| 359 | u32 c_mask = 0x0f << c_shft; | ||
| 360 | u32 r_offs = (i / 6) * 4 + KBC_ROW_CFG0_0; | ||
| 361 | u32 c_offs = (i / 8) * 4 + KBC_COL_CFG0_0; | ||
| 362 | u32 row_cfg = readl(kbc->mmio + r_offs); | ||
| 363 | u32 col_cfg = readl(kbc->mmio + c_offs); | ||
| 364 | |||
| 365 | row_cfg &= ~r_mask; | ||
| 366 | col_cfg &= ~c_mask; | ||
| 367 | |||
| 368 | if (pdata->pin_cfg[i].is_row) | ||
| 369 | row_cfg |= ((pdata->pin_cfg[i].num << 1) | 1) << r_shft; | ||
| 370 | else | ||
| 371 | col_cfg |= ((pdata->pin_cfg[i].num << 1) | 1) << c_shft; | ||
| 372 | |||
| 373 | writel(row_cfg, kbc->mmio + r_offs); | ||
| 374 | writel(col_cfg, kbc->mmio + c_offs); | ||
| 375 | } | ||
| 376 | } | ||
| 377 | |||
| 378 | static int tegra_kbc_start(struct tegra_kbc *kbc) | ||
| 379 | { | ||
| 380 | const struct tegra_kbc_platform_data *pdata = kbc->pdata; | ||
| 381 | unsigned long flags; | ||
| 382 | unsigned int debounce_cnt; | ||
| 383 | u32 val = 0; | ||
| 384 | |||
| 385 | clk_enable(kbc->clk); | ||
| 386 | |||
| 387 | /* Reset the KBC controller to clear all previous status.*/ | ||
| 388 | tegra_periph_reset_assert(kbc->clk); | ||
| 389 | udelay(100); | ||
| 390 | tegra_periph_reset_deassert(kbc->clk); | ||
| 391 | udelay(100); | ||
| 392 | |||
| 393 | tegra_kbc_config_pins(kbc); | ||
| 394 | tegra_kbc_setup_wakekeys(kbc, false); | ||
| 395 | |||
| 396 | writel(pdata->repeat_cnt, kbc->mmio + KBC_RPT_DLY_0); | ||
| 397 | |||
| 398 | /* Keyboard debounce count is maximum of 12 bits. */ | ||
| 399 | debounce_cnt = min(pdata->debounce_cnt, KBC_MAX_DEBOUNCE_CNT); | ||
| 400 | val = KBC_DEBOUNCE_CNT_SHIFT(debounce_cnt); | ||
| 401 | val |= KBC_FIFO_TH_CNT_SHIFT(1); /* set fifo interrupt threshold to 1 */ | ||
| 402 | val |= KBC_CONTROL_FIFO_CNT_INT_EN; /* interrupt on FIFO threshold */ | ||
| 403 | val |= KBC_CONTROL_KBC_EN; /* enable */ | ||
| 404 | writel(val, kbc->mmio + KBC_CONTROL_0); | ||
| 405 | |||
| 406 | /* | ||
| 407 | * Compute the delay(ns) from interrupt mode to continuous polling | ||
| 408 | * mode so the timer routine is scheduled appropriately. | ||
| 409 | */ | ||
| 410 | val = readl(kbc->mmio + KBC_INIT_DLY_0); | ||
| 411 | kbc->cp_dly_jiffies = usecs_to_jiffies((val & 0xfffff) * 32); | ||
| 412 | |||
| 413 | kbc->num_pressed_keys = 0; | ||
| 414 | |||
| 415 | /* | ||
| 416 | * Atomically clear out any remaining entries in the key FIFO | ||
| 417 | * and enable keyboard interrupts. | ||
| 418 | */ | ||
| 419 | spin_lock_irqsave(&kbc->lock, flags); | ||
| 420 | while (1) { | ||
| 421 | val = readl(kbc->mmio + KBC_INT_0); | ||
| 422 | val >>= 4; | ||
| 423 | if (!val) | ||
| 424 | break; | ||
| 425 | |||
| 426 | val = readl(kbc->mmio + KBC_KP_ENT0_0); | ||
| 427 | val = readl(kbc->mmio + KBC_KP_ENT1_0); | ||
| 428 | } | ||
| 429 | writel(0x7, kbc->mmio + KBC_INT_0); | ||
| 430 | spin_unlock_irqrestore(&kbc->lock, flags); | ||
| 431 | |||
| 432 | enable_irq(kbc->irq); | ||
| 433 | |||
| 434 | return 0; | ||
| 435 | } | ||
| 436 | |||
| 437 | static void tegra_kbc_stop(struct tegra_kbc *kbc) | ||
| 438 | { | ||
| 439 | unsigned long flags; | ||
| 440 | u32 val; | ||
| 441 | |||
| 442 | spin_lock_irqsave(&kbc->lock, flags); | ||
| 443 | val = readl(kbc->mmio + KBC_CONTROL_0); | ||
| 444 | val &= ~1; | ||
| 445 | writel(val, kbc->mmio + KBC_CONTROL_0); | ||
| 446 | spin_unlock_irqrestore(&kbc->lock, flags); | ||
| 447 | |||
| 448 | disable_irq(kbc->irq); | ||
| 449 | del_timer_sync(&kbc->timer); | ||
| 450 | |||
| 451 | clk_disable(kbc->clk); | ||
| 452 | } | ||
| 453 | |||
| 454 | static int tegra_kbc_open(struct input_dev *dev) | ||
| 455 | { | ||
| 456 | struct tegra_kbc *kbc = input_get_drvdata(dev); | ||
| 457 | |||
| 458 | return tegra_kbc_start(kbc); | ||
| 459 | } | ||
| 460 | |||
| 461 | static void tegra_kbc_close(struct input_dev *dev) | ||
| 462 | { | ||
| 463 | struct tegra_kbc *kbc = input_get_drvdata(dev); | ||
| 464 | |||
| 465 | return tegra_kbc_stop(kbc); | ||
| 466 | } | ||
| 467 | |||
| 468 | static bool __devinit | ||
| 469 | tegra_kbc_check_pin_cfg(const struct tegra_kbc_platform_data *pdata, | ||
| 470 | struct device *dev, unsigned int *num_rows) | ||
| 471 | { | ||
| 472 | int i; | ||
| 473 | |||
| 474 | *num_rows = 0; | ||
| 475 | |||
| 476 | for (i = 0; i < KBC_MAX_GPIO; i++) { | ||
| 477 | const struct tegra_kbc_pin_cfg *pin_cfg = &pdata->pin_cfg[i]; | ||
| 478 | |||
| 479 | if (pin_cfg->is_row) { | ||
| 480 | if (pin_cfg->num >= KBC_MAX_ROW) { | ||
| 481 | dev_err(dev, | ||
| 482 | "pin_cfg[%d]: invalid row number %d\n", | ||
| 483 | i, pin_cfg->num); | ||
| 484 | return false; | ||
| 485 | } | ||
| 486 | (*num_rows)++; | ||
| 487 | } else { | ||
| 488 | if (pin_cfg->num >= KBC_MAX_COL) { | ||
| 489 | dev_err(dev, | ||
| 490 | "pin_cfg[%d]: invalid column number %d\n", | ||
| 491 | i, pin_cfg->num); | ||
| 492 | return false; | ||
| 493 | } | ||
| 494 | } | ||
| 495 | } | ||
| 496 | |||
| 497 | return true; | ||
| 498 | } | ||
| 499 | |||
| 500 | static int __devinit tegra_kbc_probe(struct platform_device *pdev) | ||
| 501 | { | ||
| 502 | const struct tegra_kbc_platform_data *pdata = pdev->dev.platform_data; | ||
| 503 | const struct matrix_keymap_data *keymap_data; | ||
| 504 | struct tegra_kbc *kbc; | ||
| 505 | struct input_dev *input_dev; | ||
| 506 | struct resource *res; | ||
| 507 | int irq; | ||
| 508 | int err; | ||
| 509 | int i; | ||
| 510 | int num_rows = 0; | ||
| 511 | unsigned int debounce_cnt; | ||
| 512 | unsigned int scan_time_rows; | ||
| 513 | |||
| 514 | if (!pdata) | ||
| 515 | return -EINVAL; | ||
| 516 | |||
| 517 | if (!tegra_kbc_check_pin_cfg(pdata, &pdev->dev, &num_rows)) | ||
| 518 | return -EINVAL; | ||
| 519 | |||
| 520 | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | ||
| 521 | if (!res) { | ||
| 522 | dev_err(&pdev->dev, "failed to get I/O memory\n"); | ||
| 523 | return -ENXIO; | ||
| 524 | } | ||
| 525 | |||
| 526 | irq = platform_get_irq(pdev, 0); | ||
| 527 | if (irq < 0) { | ||
| 528 | dev_err(&pdev->dev, "failed to get keyboard IRQ\n"); | ||
| 529 | return -ENXIO; | ||
| 530 | } | ||
| 531 | |||
| 532 | kbc = kzalloc(sizeof(*kbc), GFP_KERNEL); | ||
| 533 | input_dev = input_allocate_device(); | ||
| 534 | if (!kbc || !input_dev) { | ||
| 535 | err = -ENOMEM; | ||
| 536 | goto err_free_mem; | ||
| 537 | } | ||
| 538 | |||
| 539 | kbc->pdata = pdata; | ||
| 540 | kbc->idev = input_dev; | ||
| 541 | kbc->irq = irq; | ||
| 542 | spin_lock_init(&kbc->lock); | ||
| 543 | setup_timer(&kbc->timer, tegra_kbc_keypress_timer, (unsigned long)kbc); | ||
| 544 | |||
| 545 | res = request_mem_region(res->start, resource_size(res), pdev->name); | ||
| 546 | if (!res) { | ||
| 547 | dev_err(&pdev->dev, "failed to request I/O memory\n"); | ||
| 548 | err = -EBUSY; | ||
| 549 | goto err_free_mem; | ||
| 550 | } | ||
| 551 | |||
| 552 | kbc->mmio = ioremap(res->start, resource_size(res)); | ||
| 553 | if (!kbc->mmio) { | ||
| 554 | dev_err(&pdev->dev, "failed to remap I/O memory\n"); | ||
| 555 | err = -ENXIO; | ||
| 556 | goto err_free_mem_region; | ||
| 557 | } | ||
| 558 | |||
| 559 | kbc->clk = clk_get(&pdev->dev, NULL); | ||
| 560 | if (IS_ERR(kbc->clk)) { | ||
| 561 | dev_err(&pdev->dev, "failed to get keyboard clock\n"); | ||
| 562 | err = PTR_ERR(kbc->clk); | ||
| 563 | goto err_iounmap; | ||
| 564 | } | ||
| 565 | |||
| 566 | kbc->wake_enable_rows = 0; | ||
| 567 | kbc->wake_enable_cols = 0; | ||
| 568 | for (i = 0; i < pdata->wake_cnt; i++) { | ||
| 569 | kbc->wake_enable_rows |= (1 << pdata->wake_cfg[i].row); | ||
| 570 | kbc->wake_enable_cols |= (1 << pdata->wake_cfg[i].col); | ||
| 571 | } | ||
| 572 | |||
| 573 | /* | ||
| 574 | * The time delay between two consecutive reads of the FIFO is | ||
| 575 | * the sum of the repeat time and the time taken for scanning | ||
| 576 | * the rows. There is an additional delay before the row scanning | ||
| 577 | * starts. The repoll delay is computed in milliseconds. | ||
| 578 | */ | ||
| 579 | debounce_cnt = min(pdata->debounce_cnt, KBC_MAX_DEBOUNCE_CNT); | ||
| 580 | scan_time_rows = (KBC_ROW_SCAN_TIME + debounce_cnt) * num_rows; | ||
| 581 | kbc->repoll_dly = KBC_ROW_SCAN_DLY + scan_time_rows + pdata->repeat_cnt; | ||
| 582 | kbc->repoll_dly = ((kbc->repoll_dly * KBC_CYCLE_USEC) + 999) / 1000; | ||
| 583 | |||
| 584 | input_dev->name = pdev->name; | ||
| 585 | input_dev->id.bustype = BUS_HOST; | ||
| 586 | input_dev->dev.parent = &pdev->dev; | ||
| 587 | input_dev->open = tegra_kbc_open; | ||
| 588 | input_dev->close = tegra_kbc_close; | ||
| 589 | |||
| 590 | input_set_drvdata(input_dev, kbc); | ||
| 591 | |||
| 592 | input_dev->evbit[0] = BIT_MASK(EV_KEY); | ||
| 593 | input_set_capability(input_dev, EV_MSC, MSC_SCAN); | ||
| 594 | |||
| 595 | input_dev->keycode = kbc->keycode; | ||
| 596 | input_dev->keycodesize = sizeof(kbc->keycode[0]); | ||
| 597 | input_dev->keycodemax = ARRAY_SIZE(kbc->keycode); | ||
| 598 | |||
| 599 | keymap_data = pdata->keymap_data ?: &tegra_kbc_default_keymap_data; | ||
| 600 | matrix_keypad_build_keymap(keymap_data, KBC_ROW_SHIFT, | ||
| 601 | input_dev->keycode, input_dev->keybit); | ||
| 602 | |||
| 603 | err = request_irq(kbc->irq, tegra_kbc_isr, IRQF_TRIGGER_HIGH, | ||
| 604 | pdev->name, kbc); | ||
| 605 | if (err) { | ||
| 606 | dev_err(&pdev->dev, "failed to request keyboard IRQ\n"); | ||
| 607 | goto err_put_clk; | ||
| 608 | } | ||
| 609 | |||
| 610 | disable_irq(kbc->irq); | ||
| 611 | |||
| 612 | err = input_register_device(kbc->idev); | ||
| 613 | if (err) { | ||
| 614 | dev_err(&pdev->dev, "failed to register input device\n"); | ||
| 615 | goto err_free_irq; | ||
| 616 | } | ||
| 617 | |||
| 618 | platform_set_drvdata(pdev, kbc); | ||
| 619 | device_init_wakeup(&pdev->dev, pdata->wakeup); | ||
| 620 | |||
| 621 | return 0; | ||
| 622 | |||
| 623 | err_free_irq: | ||
| 624 | free_irq(kbc->irq, pdev); | ||
| 625 | err_put_clk: | ||
| 626 | clk_put(kbc->clk); | ||
| 627 | err_iounmap: | ||
| 628 | iounmap(kbc->mmio); | ||
| 629 | err_free_mem_region: | ||
| 630 | release_mem_region(res->start, resource_size(res)); | ||
| 631 | err_free_mem: | ||
| 632 | input_free_device(kbc->idev); | ||
| 633 | kfree(kbc); | ||
| 634 | |||
| 635 | return err; | ||
| 636 | } | ||
| 637 | |||
| 638 | static int __devexit tegra_kbc_remove(struct platform_device *pdev) | ||
| 639 | { | ||
| 640 | struct tegra_kbc *kbc = platform_get_drvdata(pdev); | ||
| 641 | struct resource *res; | ||
| 642 | |||
| 643 | free_irq(kbc->irq, pdev); | ||
| 644 | clk_put(kbc->clk); | ||
| 645 | |||
| 646 | input_unregister_device(kbc->idev); | ||
| 647 | iounmap(kbc->mmio); | ||
| 648 | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | ||
| 649 | release_mem_region(res->start, resource_size(res)); | ||
| 650 | |||
| 651 | kfree(kbc); | ||
| 652 | |||
| 653 | platform_set_drvdata(pdev, NULL); | ||
| 654 | |||
| 655 | return 0; | ||
| 656 | } | ||
| 657 | |||
| 658 | #ifdef CONFIG_PM_SLEEP | ||
| 659 | static int tegra_kbc_suspend(struct device *dev) | ||
| 660 | { | ||
| 661 | struct platform_device *pdev = to_platform_device(dev); | ||
| 662 | struct tegra_kbc *kbc = platform_get_drvdata(pdev); | ||
| 663 | |||
| 664 | if (device_may_wakeup(&pdev->dev)) { | ||
| 665 | tegra_kbc_setup_wakekeys(kbc, true); | ||
| 666 | enable_irq_wake(kbc->irq); | ||
| 667 | /* Forcefully clear the interrupt status */ | ||
| 668 | writel(0x7, kbc->mmio + KBC_INT_0); | ||
| 669 | msleep(30); | ||
| 670 | } else { | ||
| 671 | mutex_lock(&kbc->idev->mutex); | ||
| 672 | if (kbc->idev->users) | ||
| 673 | tegra_kbc_stop(kbc); | ||
| 674 | mutex_unlock(&kbc->idev->mutex); | ||
| 675 | } | ||
| 676 | |||
| 677 | return 0; | ||
| 678 | } | ||
| 679 | |||
| 680 | static int tegra_kbc_resume(struct device *dev) | ||
| 681 | { | ||
| 682 | struct platform_device *pdev = to_platform_device(dev); | ||
| 683 | struct tegra_kbc *kbc = platform_get_drvdata(pdev); | ||
| 684 | int err = 0; | ||
| 685 | |||
| 686 | if (device_may_wakeup(&pdev->dev)) { | ||
| 687 | disable_irq_wake(kbc->irq); | ||
| 688 | tegra_kbc_setup_wakekeys(kbc, false); | ||
| 689 | } else { | ||
| 690 | mutex_lock(&kbc->idev->mutex); | ||
| 691 | if (kbc->idev->users) | ||
| 692 | err = tegra_kbc_start(kbc); | ||
| 693 | mutex_unlock(&kbc->idev->mutex); | ||
| 694 | } | ||
| 695 | |||
| 696 | return err; | ||
| 697 | } | ||
| 698 | #endif | ||
| 699 | |||
| 700 | static SIMPLE_DEV_PM_OPS(tegra_kbc_pm_ops, tegra_kbc_suspend, tegra_kbc_resume); | ||
| 701 | |||
| 702 | static struct platform_driver tegra_kbc_driver = { | ||
| 703 | .probe = tegra_kbc_probe, | ||
| 704 | .remove = __devexit_p(tegra_kbc_remove), | ||
| 705 | .driver = { | ||
| 706 | .name = "tegra-kbc", | ||
| 707 | .owner = THIS_MODULE, | ||
| 708 | .pm = &tegra_kbc_pm_ops, | ||
| 709 | }, | ||
| 710 | }; | ||
| 711 | |||
| 712 | static void __exit tegra_kbc_exit(void) | ||
| 713 | { | ||
| 714 | platform_driver_unregister(&tegra_kbc_driver); | ||
| 715 | } | ||
| 716 | module_exit(tegra_kbc_exit); | ||
| 717 | |||
| 718 | static int __init tegra_kbc_init(void) | ||
| 719 | { | ||
| 720 | return platform_driver_register(&tegra_kbc_driver); | ||
| 721 | } | ||
| 722 | module_init(tegra_kbc_init); | ||
| 723 | |||
| 724 | MODULE_LICENSE("GPL"); | ||
| 725 | MODULE_AUTHOR("Rakesh Iyer <riyer@nvidia.com>"); | ||
| 726 | MODULE_DESCRIPTION("Tegra matrix keyboard controller driver"); | ||
| 727 | MODULE_ALIAS("platform:tegra-kbc"); | ||
diff --git a/drivers/input/keyboard/tnetv107x-keypad.c b/drivers/input/keyboard/tnetv107x-keypad.c index b4a81ebfab9..c8f097a15d8 100644 --- a/drivers/input/keyboard/tnetv107x-keypad.c +++ b/drivers/input/keyboard/tnetv107x-keypad.c | |||
| @@ -14,6 +14,7 @@ | |||
| 14 | */ | 14 | */ |
| 15 | 15 | ||
| 16 | #include <linux/kernel.h> | 16 | #include <linux/kernel.h> |
| 17 | #include <linux/err.h> | ||
| 17 | #include <linux/errno.h> | 18 | #include <linux/errno.h> |
| 18 | #include <linux/input.h> | 19 | #include <linux/input.h> |
| 19 | #include <linux/platform_device.h> | 20 | #include <linux/platform_device.h> |
| @@ -219,9 +220,9 @@ static int __devinit keypad_probe(struct platform_device *pdev) | |||
| 219 | } | 220 | } |
| 220 | 221 | ||
| 221 | kp->clk = clk_get(dev, NULL); | 222 | kp->clk = clk_get(dev, NULL); |
| 222 | if (!kp->clk) { | 223 | if (IS_ERR(kp->clk)) { |
| 223 | dev_err(dev, "cannot claim device clock\n"); | 224 | dev_err(dev, "cannot claim device clock\n"); |
| 224 | error = -EINVAL; | 225 | error = PTR_ERR(kp->clk); |
| 225 | goto error_clk; | 226 | goto error_clk; |
| 226 | } | 227 | } |
| 227 | 228 | ||
diff --git a/drivers/input/mouse/synaptics.c b/drivers/input/mouse/synaptics.c index da392c22fc6..aa186cf6c51 100644 --- a/drivers/input/mouse/synaptics.c +++ b/drivers/input/mouse/synaptics.c | |||
| @@ -755,23 +755,26 @@ static int synaptics_reconnect(struct psmouse *psmouse) | |||
| 755 | { | 755 | { |
| 756 | struct synaptics_data *priv = psmouse->private; | 756 | struct synaptics_data *priv = psmouse->private; |
| 757 | struct synaptics_data old_priv = *priv; | 757 | struct synaptics_data old_priv = *priv; |
| 758 | int retry = 0; | ||
| 759 | int error; | ||
| 758 | 760 | ||
| 759 | psmouse_reset(psmouse); | 761 | do { |
| 762 | psmouse_reset(psmouse); | ||
| 763 | error = synaptics_detect(psmouse, 0); | ||
| 764 | } while (error && ++retry < 3); | ||
| 760 | 765 | ||
| 761 | if (synaptics_detect(psmouse, 0)) | 766 | if (error) |
| 762 | return -1; | 767 | return -1; |
| 763 | 768 | ||
| 769 | if (retry > 1) | ||
| 770 | printk(KERN_DEBUG "Synaptics reconnected after %d tries\n", | ||
| 771 | retry); | ||
| 772 | |||
| 764 | if (synaptics_query_hardware(psmouse)) { | 773 | if (synaptics_query_hardware(psmouse)) { |
| 765 | printk(KERN_ERR "Unable to query Synaptics hardware.\n"); | 774 | printk(KERN_ERR "Unable to query Synaptics hardware.\n"); |
| 766 | return -1; | 775 | return -1; |
| 767 | } | 776 | } |
| 768 | 777 | ||
| 769 | if (old_priv.identity != priv->identity || | ||
| 770 | old_priv.model_id != priv->model_id || | ||
| 771 | old_priv.capabilities != priv->capabilities || | ||
| 772 | old_priv.ext_cap != priv->ext_cap) | ||
| 773 | return -1; | ||
| 774 | |||
| 775 | if (synaptics_set_absolute_mode(psmouse)) { | 778 | if (synaptics_set_absolute_mode(psmouse)) { |
| 776 | printk(KERN_ERR "Unable to initialize Synaptics hardware.\n"); | 779 | printk(KERN_ERR "Unable to initialize Synaptics hardware.\n"); |
| 777 | return -1; | 780 | return -1; |
| @@ -782,6 +785,19 @@ static int synaptics_reconnect(struct psmouse *psmouse) | |||
| 782 | return -1; | 785 | return -1; |
| 783 | } | 786 | } |
| 784 | 787 | ||
| 788 | if (old_priv.identity != priv->identity || | ||
| 789 | old_priv.model_id != priv->model_id || | ||
| 790 | old_priv.capabilities != priv->capabilities || | ||
| 791 | old_priv.ext_cap != priv->ext_cap) { | ||
| 792 | printk(KERN_ERR "Synaptics hardware appears to be different: " | ||
| 793 | "id(%ld-%ld), model(%ld-%ld), caps(%lx-%lx), ext(%lx-%lx).\n", | ||
| 794 | old_priv.identity, priv->identity, | ||
| 795 | old_priv.model_id, priv->model_id, | ||
| 796 | old_priv.capabilities, priv->capabilities, | ||
| 797 | old_priv.ext_cap, priv->ext_cap); | ||
| 798 | return -1; | ||
| 799 | } | ||
| 800 | |||
| 785 | return 0; | 801 | return 0; |
| 786 | } | 802 | } |
| 787 | 803 | ||
diff --git a/drivers/input/serio/ct82c710.c b/drivers/input/serio/ct82c710.c index 448c7724beb..85281656724 100644 --- a/drivers/input/serio/ct82c710.c +++ b/drivers/input/serio/ct82c710.c | |||
| @@ -111,9 +111,11 @@ static void ct82c710_close(struct serio *serio) | |||
| 111 | static int ct82c710_open(struct serio *serio) | 111 | static int ct82c710_open(struct serio *serio) |
| 112 | { | 112 | { |
| 113 | unsigned char status; | 113 | unsigned char status; |
| 114 | int err; | ||
| 114 | 115 | ||
| 115 | if (request_irq(CT82C710_IRQ, ct82c710_interrupt, 0, "ct82c710", NULL)) | 116 | err = request_irq(CT82C710_IRQ, ct82c710_interrupt, 0, "ct82c710", NULL); |
| 116 | return -1; | 117 | if (err) |
| 118 | return err; | ||
| 117 | 119 | ||
| 118 | status = inb_p(CT82C710_STATUS); | 120 | status = inb_p(CT82C710_STATUS); |
| 119 | 121 | ||
| @@ -131,7 +133,7 @@ static int ct82c710_open(struct serio *serio) | |||
| 131 | status &= ~(CT82C710_ENABLE | CT82C710_INTS_ON); | 133 | status &= ~(CT82C710_ENABLE | CT82C710_INTS_ON); |
| 132 | outb_p(status, CT82C710_STATUS); | 134 | outb_p(status, CT82C710_STATUS); |
| 133 | free_irq(CT82C710_IRQ, NULL); | 135 | free_irq(CT82C710_IRQ, NULL); |
| 134 | return -1; | 136 | return -EBUSY; |
| 135 | } | 137 | } |
| 136 | 138 | ||
| 137 | return 0; | 139 | return 0; |
diff --git a/drivers/input/serio/serport.c b/drivers/input/serio/serport.c index 6e362de3f41..8755f5f3ad3 100644 --- a/drivers/input/serio/serport.c +++ b/drivers/input/serio/serport.c | |||
| @@ -116,14 +116,15 @@ static void serport_ldisc_close(struct tty_struct *tty) | |||
| 116 | 116 | ||
| 117 | /* | 117 | /* |
| 118 | * serport_ldisc_receive() is called by the low level tty driver when characters | 118 | * serport_ldisc_receive() is called by the low level tty driver when characters |
| 119 | * are ready for us. We forward the characters, one by one to the 'interrupt' | 119 | * are ready for us. We forward the characters and flags, one by one to the |
| 120 | * routine. | 120 | * 'interrupt' routine. |
| 121 | */ | 121 | */ |
| 122 | 122 | ||
| 123 | static void serport_ldisc_receive(struct tty_struct *tty, const unsigned char *cp, char *fp, int count) | 123 | static void serport_ldisc_receive(struct tty_struct *tty, const unsigned char *cp, char *fp, int count) |
| 124 | { | 124 | { |
| 125 | struct serport *serport = (struct serport*) tty->disc_data; | 125 | struct serport *serport = (struct serport*) tty->disc_data; |
| 126 | unsigned long flags; | 126 | unsigned long flags; |
| 127 | unsigned int ch_flags; | ||
| 127 | int i; | 128 | int i; |
| 128 | 129 | ||
| 129 | spin_lock_irqsave(&serport->lock, flags); | 130 | spin_lock_irqsave(&serport->lock, flags); |
| @@ -131,8 +132,23 @@ static void serport_ldisc_receive(struct tty_struct *tty, const unsigned char *c | |||
| 131 | if (!test_bit(SERPORT_ACTIVE, &serport->flags)) | 132 | if (!test_bit(SERPORT_ACTIVE, &serport->flags)) |
| 132 | goto out; | 133 | goto out; |
| 133 | 134 | ||
| 134 | for (i = 0; i < count; i++) | 135 | for (i = 0; i < count; i++) { |
| 135 | serio_interrupt(serport->serio, cp[i], 0); | 136 | switch (fp[i]) { |
| 137 | case TTY_FRAME: | ||
| 138 | ch_flags = SERIO_FRAME; | ||
| 139 | break; | ||
| 140 | |||
| 141 | case TTY_PARITY: | ||
| 142 | ch_flags = SERIO_PARITY; | ||
| 143 | break; | ||
| 144 | |||
| 145 | default: | ||
| 146 | ch_flags = 0; | ||
| 147 | break; | ||
| 148 | } | ||
| 149 | |||
| 150 | serio_interrupt(serport->serio, cp[i], ch_flags); | ||
| 151 | } | ||
| 136 | 152 | ||
| 137 | out: | 153 | out: |
| 138 | spin_unlock_irqrestore(&serport->lock, flags); | 154 | spin_unlock_irqrestore(&serport->lock, flags); |
diff --git a/drivers/input/sparse-keymap.c b/drivers/input/sparse-keymap.c index a29a7812bd4..7729e547ba6 100644 --- a/drivers/input/sparse-keymap.c +++ b/drivers/input/sparse-keymap.c | |||
| @@ -201,6 +201,7 @@ int sparse_keymap_setup(struct input_dev *dev, | |||
| 201 | break; | 201 | break; |
| 202 | 202 | ||
| 203 | case KE_SW: | 203 | case KE_SW: |
| 204 | case KE_VSW: | ||
| 204 | __set_bit(EV_SW, dev->evbit); | 205 | __set_bit(EV_SW, dev->evbit); |
| 205 | __set_bit(entry->sw.code, dev->swbit); | 206 | __set_bit(entry->sw.code, dev->swbit); |
| 206 | break; | 207 | break; |
diff --git a/drivers/input/tablet/wacom_wac.c b/drivers/input/tablet/wacom_wac.c index 518782999fe..367fa82a607 100644 --- a/drivers/input/tablet/wacom_wac.c +++ b/drivers/input/tablet/wacom_wac.c | |||
| @@ -1101,6 +1101,13 @@ void wacom_setup_device_quirks(struct wacom_features *features) | |||
| 1101 | } | 1101 | } |
| 1102 | } | 1102 | } |
| 1103 | 1103 | ||
| 1104 | static unsigned int wacom_calculate_touch_res(unsigned int logical_max, | ||
| 1105 | unsigned int physical_max) | ||
| 1106 | { | ||
| 1107 | /* Touch physical dimensions are in 100th of mm */ | ||
| 1108 | return (logical_max * 100) / physical_max; | ||
| 1109 | } | ||
| 1110 | |||
| 1104 | void wacom_setup_input_capabilities(struct input_dev *input_dev, | 1111 | void wacom_setup_input_capabilities(struct input_dev *input_dev, |
| 1105 | struct wacom_wac *wacom_wac) | 1112 | struct wacom_wac *wacom_wac) |
| 1106 | { | 1113 | { |
| @@ -1228,8 +1235,12 @@ void wacom_setup_input_capabilities(struct input_dev *input_dev, | |||
| 1228 | case TABLETPC: | 1235 | case TABLETPC: |
| 1229 | if (features->device_type == BTN_TOOL_DOUBLETAP || | 1236 | if (features->device_type == BTN_TOOL_DOUBLETAP || |
| 1230 | features->device_type == BTN_TOOL_TRIPLETAP) { | 1237 | features->device_type == BTN_TOOL_TRIPLETAP) { |
| 1231 | input_set_abs_params(input_dev, ABS_RX, 0, features->x_phy, 0, 0); | 1238 | input_abs_set_res(input_dev, ABS_X, |
| 1232 | input_set_abs_params(input_dev, ABS_RY, 0, features->y_phy, 0, 0); | 1239 | wacom_calculate_touch_res(features->x_max, |
| 1240 | features->x_phy)); | ||
| 1241 | input_abs_set_res(input_dev, ABS_Y, | ||
| 1242 | wacom_calculate_touch_res(features->y_max, | ||
| 1243 | features->y_phy)); | ||
| 1233 | __set_bit(BTN_TOOL_DOUBLETAP, input_dev->keybit); | 1244 | __set_bit(BTN_TOOL_DOUBLETAP, input_dev->keybit); |
| 1234 | } | 1245 | } |
| 1235 | 1246 | ||
| @@ -1272,6 +1283,12 @@ void wacom_setup_input_capabilities(struct input_dev *input_dev, | |||
| 1272 | input_set_abs_params(input_dev, ABS_MT_PRESSURE, | 1283 | input_set_abs_params(input_dev, ABS_MT_PRESSURE, |
| 1273 | 0, features->pressure_max, | 1284 | 0, features->pressure_max, |
| 1274 | features->pressure_fuzz, 0); | 1285 | features->pressure_fuzz, 0); |
| 1286 | input_abs_set_res(input_dev, ABS_X, | ||
| 1287 | wacom_calculate_touch_res(features->x_max, | ||
| 1288 | features->x_phy)); | ||
| 1289 | input_abs_set_res(input_dev, ABS_Y, | ||
| 1290 | wacom_calculate_touch_res(features->y_max, | ||
| 1291 | features->y_phy)); | ||
| 1275 | } else if (features->device_type == BTN_TOOL_PEN) { | 1292 | } else if (features->device_type == BTN_TOOL_PEN) { |
| 1276 | __set_bit(BTN_TOOL_RUBBER, input_dev->keybit); | 1293 | __set_bit(BTN_TOOL_RUBBER, input_dev->keybit); |
| 1277 | __set_bit(BTN_TOOL_PEN, input_dev->keybit); | 1294 | __set_bit(BTN_TOOL_PEN, input_dev->keybit); |
| @@ -1426,6 +1443,10 @@ static struct wacom_features wacom_features_0xD3 = | |||
| 1426 | { "Wacom Bamboo 2FG 6x8", WACOM_PKGLEN_BBFUN, 21648, 13530, 1023, 63, BAMBOO_PT }; | 1443 | { "Wacom Bamboo 2FG 6x8", WACOM_PKGLEN_BBFUN, 21648, 13530, 1023, 63, BAMBOO_PT }; |
| 1427 | static const struct wacom_features wacom_features_0xD4 = | 1444 | static const struct wacom_features wacom_features_0xD4 = |
| 1428 | { "Wacom Bamboo Pen", WACOM_PKGLEN_BBFUN, 14720, 9200, 255, 63, BAMBOO_PT }; | 1445 | { "Wacom Bamboo Pen", WACOM_PKGLEN_BBFUN, 14720, 9200, 255, 63, BAMBOO_PT }; |
| 1446 | static struct wacom_features wacom_features_0xD6 = | ||
| 1447 | { "Wacom BambooPT 2FG 4x5", WACOM_PKGLEN_BBFUN, 14720, 9200, 1023, 63, BAMBOO_PT }; | ||
| 1448 | static struct wacom_features wacom_features_0xD7 = | ||
| 1449 | { "Wacom BambooPT 2FG Small", WACOM_PKGLEN_BBFUN, 14720, 9200, 1023, 63, BAMBOO_PT }; | ||
| 1429 | static struct wacom_features wacom_features_0xD8 = | 1450 | static struct wacom_features wacom_features_0xD8 = |
| 1430 | { "Wacom Bamboo Comic 2FG", WACOM_PKGLEN_BBFUN, 21648, 13530, 1023, 63, BAMBOO_PT }; | 1451 | { "Wacom Bamboo Comic 2FG", WACOM_PKGLEN_BBFUN, 21648, 13530, 1023, 63, BAMBOO_PT }; |
| 1431 | static struct wacom_features wacom_features_0xDA = | 1452 | static struct wacom_features wacom_features_0xDA = |
| @@ -1507,6 +1528,8 @@ const struct usb_device_id wacom_ids[] = { | |||
| 1507 | { USB_DEVICE_WACOM(0xD2) }, | 1528 | { USB_DEVICE_WACOM(0xD2) }, |
| 1508 | { USB_DEVICE_WACOM(0xD3) }, | 1529 | { USB_DEVICE_WACOM(0xD3) }, |
| 1509 | { USB_DEVICE_WACOM(0xD4) }, | 1530 | { USB_DEVICE_WACOM(0xD4) }, |
| 1531 | { USB_DEVICE_WACOM(0xD6) }, | ||
| 1532 | { USB_DEVICE_WACOM(0xD7) }, | ||
| 1510 | { USB_DEVICE_WACOM(0xD8) }, | 1533 | { USB_DEVICE_WACOM(0xD8) }, |
| 1511 | { USB_DEVICE_WACOM(0xDA) }, | 1534 | { USB_DEVICE_WACOM(0xDA) }, |
| 1512 | { USB_DEVICE_WACOM(0xDB) }, | 1535 | { USB_DEVICE_WACOM(0xDB) }, |
diff --git a/drivers/input/touchscreen/bu21013_ts.c b/drivers/input/touchscreen/bu21013_ts.c index f7fa9ef4cd6..1507ce108d5 100644 --- a/drivers/input/touchscreen/bu21013_ts.c +++ b/drivers/input/touchscreen/bu21013_ts.c | |||
| @@ -12,6 +12,7 @@ | |||
| 12 | #include <linux/input.h> | 12 | #include <linux/input.h> |
| 13 | #include <linux/input/bu21013.h> | 13 | #include <linux/input/bu21013.h> |
| 14 | #include <linux/slab.h> | 14 | #include <linux/slab.h> |
| 15 | #include <linux/regulator/consumer.h> | ||
| 15 | 16 | ||
| 16 | #define PEN_DOWN_INTR 0 | 17 | #define PEN_DOWN_INTR 0 |
| 17 | #define MAX_FINGERS 2 | 18 | #define MAX_FINGERS 2 |
| @@ -139,6 +140,7 @@ | |||
| 139 | * @chip: pointer to the touch panel controller | 140 | * @chip: pointer to the touch panel controller |
| 140 | * @in_dev: pointer to the input device structure | 141 | * @in_dev: pointer to the input device structure |
| 141 | * @intr_pin: interrupt pin value | 142 | * @intr_pin: interrupt pin value |
| 143 | * @regulator: pointer to the Regulator used for touch screen | ||
| 142 | * | 144 | * |
| 143 | * Touch panel device data structure | 145 | * Touch panel device data structure |
| 144 | */ | 146 | */ |
| @@ -149,6 +151,7 @@ struct bu21013_ts_data { | |||
| 149 | const struct bu21013_platform_device *chip; | 151 | const struct bu21013_platform_device *chip; |
| 150 | struct input_dev *in_dev; | 152 | struct input_dev *in_dev; |
| 151 | unsigned int intr_pin; | 153 | unsigned int intr_pin; |
| 154 | struct regulator *regulator; | ||
| 152 | }; | 155 | }; |
| 153 | 156 | ||
| 154 | /** | 157 | /** |
| @@ -456,6 +459,20 @@ static int __devinit bu21013_probe(struct i2c_client *client, | |||
| 456 | bu21013_data->in_dev = in_dev; | 459 | bu21013_data->in_dev = in_dev; |
| 457 | bu21013_data->chip = pdata; | 460 | bu21013_data->chip = pdata; |
| 458 | bu21013_data->client = client; | 461 | bu21013_data->client = client; |
| 462 | |||
| 463 | bu21013_data->regulator = regulator_get(&client->dev, "V-TOUCH"); | ||
| 464 | if (IS_ERR(bu21013_data->regulator)) { | ||
| 465 | dev_err(&client->dev, "regulator_get failed\n"); | ||
| 466 | error = PTR_ERR(bu21013_data->regulator); | ||
| 467 | goto err_free_mem; | ||
| 468 | } | ||
| 469 | |||
| 470 | error = regulator_enable(bu21013_data->regulator); | ||
| 471 | if (error < 0) { | ||
| 472 | dev_err(&client->dev, "regulator enable failed\n"); | ||
| 473 | goto err_put_regulator; | ||
| 474 | } | ||
| 475 | |||
| 459 | bu21013_data->touch_stopped = false; | 476 | bu21013_data->touch_stopped = false; |
| 460 | init_waitqueue_head(&bu21013_data->wait); | 477 | init_waitqueue_head(&bu21013_data->wait); |
| 461 | 478 | ||
| @@ -464,7 +481,7 @@ static int __devinit bu21013_probe(struct i2c_client *client, | |||
| 464 | error = pdata->cs_en(pdata->cs_pin); | 481 | error = pdata->cs_en(pdata->cs_pin); |
| 465 | if (error < 0) { | 482 | if (error < 0) { |
| 466 | dev_err(&client->dev, "chip init failed\n"); | 483 | dev_err(&client->dev, "chip init failed\n"); |
| 467 | goto err_free_mem; | 484 | goto err_disable_regulator; |
| 468 | } | 485 | } |
| 469 | } | 486 | } |
| 470 | 487 | ||
| @@ -485,9 +502,9 @@ static int __devinit bu21013_probe(struct i2c_client *client, | |||
| 485 | __set_bit(EV_ABS, in_dev->evbit); | 502 | __set_bit(EV_ABS, in_dev->evbit); |
| 486 | 503 | ||
| 487 | input_set_abs_params(in_dev, ABS_MT_POSITION_X, 0, | 504 | input_set_abs_params(in_dev, ABS_MT_POSITION_X, 0, |
| 488 | pdata->x_max_res, 0, 0); | 505 | pdata->touch_x_max, 0, 0); |
| 489 | input_set_abs_params(in_dev, ABS_MT_POSITION_Y, 0, | 506 | input_set_abs_params(in_dev, ABS_MT_POSITION_Y, 0, |
| 490 | pdata->y_max_res, 0, 0); | 507 | pdata->touch_y_max, 0, 0); |
| 491 | input_set_drvdata(in_dev, bu21013_data); | 508 | input_set_drvdata(in_dev, bu21013_data); |
| 492 | 509 | ||
| 493 | error = request_threaded_irq(pdata->irq, NULL, bu21013_gpio_irq, | 510 | error = request_threaded_irq(pdata->irq, NULL, bu21013_gpio_irq, |
| @@ -513,6 +530,10 @@ err_free_irq: | |||
| 513 | bu21013_free_irq(bu21013_data); | 530 | bu21013_free_irq(bu21013_data); |
| 514 | err_cs_disable: | 531 | err_cs_disable: |
| 515 | pdata->cs_dis(pdata->cs_pin); | 532 | pdata->cs_dis(pdata->cs_pin); |
| 533 | err_disable_regulator: | ||
| 534 | regulator_disable(bu21013_data->regulator); | ||
| 535 | err_put_regulator: | ||
| 536 | regulator_put(bu21013_data->regulator); | ||
| 516 | err_free_mem: | 537 | err_free_mem: |
| 517 | input_free_device(in_dev); | 538 | input_free_device(in_dev); |
| 518 | kfree(bu21013_data); | 539 | kfree(bu21013_data); |
| @@ -535,6 +556,10 @@ static int __devexit bu21013_remove(struct i2c_client *client) | |||
| 535 | bu21013_data->chip->cs_dis(bu21013_data->chip->cs_pin); | 556 | bu21013_data->chip->cs_dis(bu21013_data->chip->cs_pin); |
| 536 | 557 | ||
| 537 | input_unregister_device(bu21013_data->in_dev); | 558 | input_unregister_device(bu21013_data->in_dev); |
| 559 | |||
| 560 | regulator_disable(bu21013_data->regulator); | ||
| 561 | regulator_put(bu21013_data->regulator); | ||
| 562 | |||
| 538 | kfree(bu21013_data); | 563 | kfree(bu21013_data); |
| 539 | 564 | ||
| 540 | device_init_wakeup(&client->dev, false); | 565 | device_init_wakeup(&client->dev, false); |
| @@ -561,6 +586,8 @@ static int bu21013_suspend(struct device *dev) | |||
| 561 | else | 586 | else |
| 562 | disable_irq(bu21013_data->chip->irq); | 587 | disable_irq(bu21013_data->chip->irq); |
| 563 | 588 | ||
| 589 | regulator_disable(bu21013_data->regulator); | ||
| 590 | |||
| 564 | return 0; | 591 | return 0; |
| 565 | } | 592 | } |
| 566 | 593 | ||
| @@ -577,6 +604,12 @@ static int bu21013_resume(struct device *dev) | |||
| 577 | struct i2c_client *client = bu21013_data->client; | 604 | struct i2c_client *client = bu21013_data->client; |
| 578 | int retval; | 605 | int retval; |
| 579 | 606 | ||
| 607 | retval = regulator_enable(bu21013_data->regulator); | ||
| 608 | if (retval < 0) { | ||
| 609 | dev_err(&client->dev, "bu21013 regulator enable failed\n"); | ||
| 610 | return retval; | ||
| 611 | } | ||
| 612 | |||
| 580 | retval = bu21013_init_chip(bu21013_data); | 613 | retval = bu21013_init_chip(bu21013_data); |
| 581 | if (retval < 0) { | 614 | if (retval < 0) { |
| 582 | dev_err(&client->dev, "bu21013 controller config failed\n"); | 615 | dev_err(&client->dev, "bu21013 controller config failed\n"); |
diff --git a/drivers/input/touchscreen/tnetv107x-ts.c b/drivers/input/touchscreen/tnetv107x-ts.c index cf1dba2e267..22a3411e93c 100644 --- a/drivers/input/touchscreen/tnetv107x-ts.c +++ b/drivers/input/touchscreen/tnetv107x-ts.c | |||
| @@ -14,6 +14,7 @@ | |||
| 14 | */ | 14 | */ |
| 15 | 15 | ||
| 16 | #include <linux/kernel.h> | 16 | #include <linux/kernel.h> |
| 17 | #include <linux/err.h> | ||
| 17 | #include <linux/errno.h> | 18 | #include <linux/errno.h> |
| 18 | #include <linux/input.h> | 19 | #include <linux/input.h> |
| 19 | #include <linux/platform_device.h> | 20 | #include <linux/platform_device.h> |
| @@ -289,9 +290,9 @@ static int __devinit tsc_probe(struct platform_device *pdev) | |||
| 289 | } | 290 | } |
| 290 | 291 | ||
| 291 | ts->clk = clk_get(dev, NULL); | 292 | ts->clk = clk_get(dev, NULL); |
| 292 | if (!ts->clk) { | 293 | if (IS_ERR(ts->clk)) { |
| 293 | dev_err(dev, "cannot claim device clock\n"); | 294 | dev_err(dev, "cannot claim device clock\n"); |
| 294 | error = -EINVAL; | 295 | error = PTR_ERR(ts->clk); |
| 295 | goto error_clk; | 296 | goto error_clk; |
| 296 | } | 297 | } |
| 297 | 298 | ||
diff --git a/drivers/isdn/icn/icn.c b/drivers/isdn/icn/icn.c index f2b5bab5e6a..1f355bb85e5 100644 --- a/drivers/isdn/icn/icn.c +++ b/drivers/isdn/icn/icn.c | |||
| @@ -1627,7 +1627,7 @@ __setup("icn=", icn_setup); | |||
| 1627 | static int __init icn_init(void) | 1627 | static int __init icn_init(void) |
| 1628 | { | 1628 | { |
| 1629 | char *p; | 1629 | char *p; |
| 1630 | char rev[20]; | 1630 | char rev[21]; |
| 1631 | 1631 | ||
| 1632 | memset(&dev, 0, sizeof(icn_dev)); | 1632 | memset(&dev, 0, sizeof(icn_dev)); |
| 1633 | dev.memaddr = (membase & 0x0ffc000); | 1633 | dev.memaddr = (membase & 0x0ffc000); |
| @@ -1638,6 +1638,7 @@ static int __init icn_init(void) | |||
| 1638 | 1638 | ||
| 1639 | if ((p = strchr(revision, ':'))) { | 1639 | if ((p = strchr(revision, ':'))) { |
| 1640 | strncpy(rev, p + 1, 20); | 1640 | strncpy(rev, p + 1, 20); |
| 1641 | rev[20] = '\0'; | ||
| 1641 | p = strchr(rev, '$'); | 1642 | p = strchr(rev, '$'); |
| 1642 | if (p) | 1643 | if (p) |
| 1643 | *p = 0; | 1644 | *p = 0; |
diff --git a/drivers/leds/leds-pwm.c b/drivers/leds/leds-pwm.c index da3fa8dcdf5..666daf77872 100644 --- a/drivers/leds/leds-pwm.c +++ b/drivers/leds/leds-pwm.c | |||
| @@ -69,6 +69,7 @@ static int led_pwm_probe(struct platform_device *pdev) | |||
| 69 | led_dat->pwm = pwm_request(cur_led->pwm_id, | 69 | led_dat->pwm = pwm_request(cur_led->pwm_id, |
| 70 | cur_led->name); | 70 | cur_led->name); |
| 71 | if (IS_ERR(led_dat->pwm)) { | 71 | if (IS_ERR(led_dat->pwm)) { |
| 72 | ret = PTR_ERR(led_dat->pwm); | ||
| 72 | dev_err(&pdev->dev, "unable to request PWM %d\n", | 73 | dev_err(&pdev->dev, "unable to request PWM %d\n", |
| 73 | cur_led->pwm_id); | 74 | cur_led->pwm_id); |
| 74 | goto err; | 75 | goto err; |
diff --git a/drivers/media/rc/ir-lirc-codec.c b/drivers/media/rc/ir-lirc-codec.c index f011c5d9dea..1c5cc65ea1e 100644 --- a/drivers/media/rc/ir-lirc-codec.c +++ b/drivers/media/rc/ir-lirc-codec.c | |||
| @@ -1,4 +1,4 @@ | |||
| 1 | /* ir-lirc-codec.c - ir-core to classic lirc interface bridge | 1 | /* ir-lirc-codec.c - rc-core to classic lirc interface bridge |
| 2 | * | 2 | * |
| 3 | * Copyright (C) 2010 by Jarod Wilson <jarod@redhat.com> | 3 | * Copyright (C) 2010 by Jarod Wilson <jarod@redhat.com> |
| 4 | * | 4 | * |
| @@ -47,6 +47,7 @@ static int ir_lirc_decode(struct rc_dev *dev, struct ir_raw_event ev) | |||
| 47 | /* Carrier reports */ | 47 | /* Carrier reports */ |
| 48 | if (ev.carrier_report) { | 48 | if (ev.carrier_report) { |
| 49 | sample = LIRC_FREQUENCY(ev.carrier); | 49 | sample = LIRC_FREQUENCY(ev.carrier); |
| 50 | IR_dprintk(2, "carrier report (freq: %d)\n", sample); | ||
| 50 | 51 | ||
| 51 | /* Packet end */ | 52 | /* Packet end */ |
| 52 | } else if (ev.timeout) { | 53 | } else if (ev.timeout) { |
| @@ -62,6 +63,7 @@ static int ir_lirc_decode(struct rc_dev *dev, struct ir_raw_event ev) | |||
| 62 | return 0; | 63 | return 0; |
| 63 | 64 | ||
| 64 | sample = LIRC_TIMEOUT(ev.duration / 1000); | 65 | sample = LIRC_TIMEOUT(ev.duration / 1000); |
| 66 | IR_dprintk(2, "timeout report (duration: %d)\n", sample); | ||
| 65 | 67 | ||
| 66 | /* Normal sample */ | 68 | /* Normal sample */ |
| 67 | } else { | 69 | } else { |
| @@ -85,6 +87,8 @@ static int ir_lirc_decode(struct rc_dev *dev, struct ir_raw_event ev) | |||
| 85 | 87 | ||
| 86 | sample = ev.pulse ? LIRC_PULSE(ev.duration / 1000) : | 88 | sample = ev.pulse ? LIRC_PULSE(ev.duration / 1000) : |
| 87 | LIRC_SPACE(ev.duration / 1000); | 89 | LIRC_SPACE(ev.duration / 1000); |
| 90 | IR_dprintk(2, "delivering %uus %s to lirc_dev\n", | ||
| 91 | TO_US(ev.duration), TO_STR(ev.pulse)); | ||
| 88 | } | 92 | } |
| 89 | 93 | ||
| 90 | lirc_buffer_write(dev->raw->lirc.drv->rbuf, | 94 | lirc_buffer_write(dev->raw->lirc.drv->rbuf, |
diff --git a/drivers/media/rc/keymaps/rc-rc6-mce.c b/drivers/media/rc/keymaps/rc-rc6-mce.c index 3bf3337875d..2f5dc0622b9 100644 --- a/drivers/media/rc/keymaps/rc-rc6-mce.c +++ b/drivers/media/rc/keymaps/rc-rc6-mce.c | |||
| @@ -3,6 +3,9 @@ | |||
| 3 | * | 3 | * |
| 4 | * Copyright (c) 2010 by Jarod Wilson <jarod@redhat.com> | 4 | * Copyright (c) 2010 by Jarod Wilson <jarod@redhat.com> |
| 5 | * | 5 | * |
| 6 | * See http://mediacenterguides.com/book/export/html/31 for details on | ||
| 7 | * key mappings. | ||
| 8 | * | ||
| 6 | * This program is free software; you can redistribute it and/or modify | 9 | * This program is free software; you can redistribute it and/or modify |
| 7 | * it under the terms of the GNU General Public License as published by | 10 | * it under the terms of the GNU General Public License as published by |
| 8 | * the Free Software Foundation; either version 2 of the License, or | 11 | * the Free Software Foundation; either version 2 of the License, or |
| @@ -60,6 +63,9 @@ static struct rc_map_table rc6_mce[] = { | |||
| 60 | { 0x800f0426, KEY_EPG }, /* Guide */ | 63 | { 0x800f0426, KEY_EPG }, /* Guide */ |
| 61 | { 0x800f0427, KEY_ZOOM }, /* Aspect */ | 64 | { 0x800f0427, KEY_ZOOM }, /* Aspect */ |
| 62 | 65 | ||
| 66 | { 0x800f0432, KEY_MODE }, /* Visualization */ | ||
| 67 | { 0x800f0433, KEY_PRESENTATION }, /* Slide Show */ | ||
| 68 | { 0x800f0434, KEY_EJECTCD }, | ||
| 63 | { 0x800f043a, KEY_BRIGHTNESSUP }, | 69 | { 0x800f043a, KEY_BRIGHTNESSUP }, |
| 64 | 70 | ||
| 65 | { 0x800f0446, KEY_TV }, | 71 | { 0x800f0446, KEY_TV }, |
diff --git a/drivers/media/rc/mceusb.c b/drivers/media/rc/mceusb.c index 079353e5d55..6df0a498064 100644 --- a/drivers/media/rc/mceusb.c +++ b/drivers/media/rc/mceusb.c | |||
| @@ -816,7 +816,7 @@ static void mceusb_handle_command(struct mceusb_dev *ir, int index) | |||
| 816 | switch (ir->buf_in[index]) { | 816 | switch (ir->buf_in[index]) { |
| 817 | /* 2-byte return value commands */ | 817 | /* 2-byte return value commands */ |
| 818 | case MCE_CMD_S_TIMEOUT: | 818 | case MCE_CMD_S_TIMEOUT: |
| 819 | ir->rc->timeout = MS_TO_NS((hi << 8 | lo) / 2); | 819 | ir->rc->timeout = US_TO_NS((hi << 8 | lo) / 2); |
| 820 | break; | 820 | break; |
| 821 | 821 | ||
| 822 | /* 1-byte return value commands */ | 822 | /* 1-byte return value commands */ |
| @@ -855,9 +855,10 @@ static void mceusb_process_ir_data(struct mceusb_dev *ir, int buf_len) | |||
| 855 | break; | 855 | break; |
| 856 | case PARSE_IRDATA: | 856 | case PARSE_IRDATA: |
| 857 | ir->rem--; | 857 | ir->rem--; |
| 858 | init_ir_raw_event(&rawir); | ||
| 858 | rawir.pulse = ((ir->buf_in[i] & MCE_PULSE_BIT) != 0); | 859 | rawir.pulse = ((ir->buf_in[i] & MCE_PULSE_BIT) != 0); |
| 859 | rawir.duration = (ir->buf_in[i] & MCE_PULSE_MASK) | 860 | rawir.duration = (ir->buf_in[i] & MCE_PULSE_MASK) |
| 860 | * MS_TO_US(MCE_TIME_UNIT); | 861 | * US_TO_NS(MCE_TIME_UNIT); |
| 861 | 862 | ||
| 862 | dev_dbg(ir->dev, "Storing %s with duration %d\n", | 863 | dev_dbg(ir->dev, "Storing %s with duration %d\n", |
| 863 | rawir.pulse ? "pulse" : "space", | 864 | rawir.pulse ? "pulse" : "space", |
| @@ -883,6 +884,8 @@ static void mceusb_process_ir_data(struct mceusb_dev *ir, int buf_len) | |||
| 883 | i, ir->rem + 1, false); | 884 | i, ir->rem + 1, false); |
| 884 | if (ir->rem) | 885 | if (ir->rem) |
| 885 | ir->parser_state = PARSE_IRDATA; | 886 | ir->parser_state = PARSE_IRDATA; |
| 887 | else | ||
| 888 | ir_raw_event_reset(ir->rc); | ||
| 886 | break; | 889 | break; |
| 887 | } | 890 | } |
| 888 | 891 | ||
| @@ -1060,7 +1063,7 @@ static struct rc_dev *mceusb_init_rc_dev(struct mceusb_dev *ir) | |||
| 1060 | rc->priv = ir; | 1063 | rc->priv = ir; |
| 1061 | rc->driver_type = RC_DRIVER_IR_RAW; | 1064 | rc->driver_type = RC_DRIVER_IR_RAW; |
| 1062 | rc->allowed_protos = RC_TYPE_ALL; | 1065 | rc->allowed_protos = RC_TYPE_ALL; |
| 1063 | rc->timeout = MS_TO_NS(1000); | 1066 | rc->timeout = US_TO_NS(1000); |
| 1064 | if (!ir->flags.no_tx) { | 1067 | if (!ir->flags.no_tx) { |
| 1065 | rc->s_tx_mask = mceusb_set_tx_mask; | 1068 | rc->s_tx_mask = mceusb_set_tx_mask; |
| 1066 | rc->s_tx_carrier = mceusb_set_tx_carrier; | 1069 | rc->s_tx_carrier = mceusb_set_tx_carrier; |
diff --git a/drivers/media/rc/nuvoton-cir.c b/drivers/media/rc/nuvoton-cir.c index dd4caf8ef80..273d9d67479 100644 --- a/drivers/media/rc/nuvoton-cir.c +++ b/drivers/media/rc/nuvoton-cir.c | |||
| @@ -460,7 +460,7 @@ static u32 nvt_rx_carrier_detect(struct nvt_dev *nvt) | |||
| 460 | return 0; | 460 | return 0; |
| 461 | } | 461 | } |
| 462 | 462 | ||
| 463 | carrier = (count * 1000000) / duration; | 463 | carrier = MS_TO_NS(count) / duration; |
| 464 | 464 | ||
| 465 | if ((carrier > MAX_CARRIER) || (carrier < MIN_CARRIER)) | 465 | if ((carrier > MAX_CARRIER) || (carrier < MIN_CARRIER)) |
| 466 | nvt_dbg("WTF? Carrier frequency out of range!"); | 466 | nvt_dbg("WTF? Carrier frequency out of range!"); |
| @@ -612,8 +612,8 @@ static void nvt_process_rx_ir_data(struct nvt_dev *nvt) | |||
| 612 | sample = nvt->buf[i]; | 612 | sample = nvt->buf[i]; |
| 613 | 613 | ||
| 614 | rawir.pulse = ((sample & BUF_PULSE_BIT) != 0); | 614 | rawir.pulse = ((sample & BUF_PULSE_BIT) != 0); |
| 615 | rawir.duration = (sample & BUF_LEN_MASK) | 615 | rawir.duration = US_TO_NS((sample & BUF_LEN_MASK) |
| 616 | * SAMPLE_PERIOD * 1000; | 616 | * SAMPLE_PERIOD); |
| 617 | 617 | ||
| 618 | if ((sample & BUF_LEN_MASK) == BUF_LEN_MASK) { | 618 | if ((sample & BUF_LEN_MASK) == BUF_LEN_MASK) { |
| 619 | if (nvt->rawir.pulse == rawir.pulse) | 619 | if (nvt->rawir.pulse == rawir.pulse) |
diff --git a/drivers/media/rc/rc-main.c b/drivers/media/rc/rc-main.c index 72be8a02118..512a2f4ada0 100644 --- a/drivers/media/rc/rc-main.c +++ b/drivers/media/rc/rc-main.c | |||
| @@ -458,21 +458,27 @@ static int ir_getkeycode(struct input_dev *idev, | |||
| 458 | index = ir_lookup_by_scancode(rc_map, scancode); | 458 | index = ir_lookup_by_scancode(rc_map, scancode); |
| 459 | } | 459 | } |
| 460 | 460 | ||
| 461 | if (index >= rc_map->len) { | 461 | if (index < rc_map->len) { |
| 462 | if (!(ke->flags & INPUT_KEYMAP_BY_INDEX)) | 462 | entry = &rc_map->scan[index]; |
| 463 | IR_dprintk(1, "unknown key for scancode 0x%04x\n", | 463 | |
| 464 | scancode); | 464 | ke->index = index; |
| 465 | ke->keycode = entry->keycode; | ||
| 466 | ke->len = sizeof(entry->scancode); | ||
| 467 | memcpy(ke->scancode, &entry->scancode, sizeof(entry->scancode)); | ||
| 468 | |||
| 469 | } else if (!(ke->flags & INPUT_KEYMAP_BY_INDEX)) { | ||
| 470 | /* | ||
| 471 | * We do not really know the valid range of scancodes | ||
| 472 | * so let's respond with KEY_RESERVED to anything we | ||
| 473 | * do not have mapping for [yet]. | ||
| 474 | */ | ||
| 475 | ke->index = index; | ||
| 476 | ke->keycode = KEY_RESERVED; | ||
| 477 | } else { | ||
| 465 | retval = -EINVAL; | 478 | retval = -EINVAL; |
| 466 | goto out; | 479 | goto out; |
| 467 | } | 480 | } |
| 468 | 481 | ||
| 469 | entry = &rc_map->scan[index]; | ||
| 470 | |||
| 471 | ke->index = index; | ||
| 472 | ke->keycode = entry->keycode; | ||
| 473 | ke->len = sizeof(entry->scancode); | ||
| 474 | memcpy(ke->scancode, &entry->scancode, sizeof(entry->scancode)); | ||
| 475 | |||
| 476 | retval = 0; | 482 | retval = 0; |
| 477 | 483 | ||
| 478 | out: | 484 | out: |
diff --git a/drivers/media/rc/streamzap.c b/drivers/media/rc/streamzap.c index 6e2911c2abf..e435d94c077 100644 --- a/drivers/media/rc/streamzap.c +++ b/drivers/media/rc/streamzap.c | |||
| @@ -164,7 +164,7 @@ static void sz_push_full_pulse(struct streamzap_ir *sz, | |||
| 164 | sz->signal_start.tv_usec - | 164 | sz->signal_start.tv_usec - |
| 165 | sz->signal_last.tv_usec); | 165 | sz->signal_last.tv_usec); |
| 166 | rawir.duration -= sz->sum; | 166 | rawir.duration -= sz->sum; |
| 167 | rawir.duration *= 1000; | 167 | rawir.duration = US_TO_NS(rawir.duration); |
| 168 | rawir.duration &= IR_MAX_DURATION; | 168 | rawir.duration &= IR_MAX_DURATION; |
| 169 | } | 169 | } |
| 170 | sz_push(sz, rawir); | 170 | sz_push(sz, rawir); |
| @@ -177,7 +177,7 @@ static void sz_push_full_pulse(struct streamzap_ir *sz, | |||
| 177 | rawir.duration = ((int) value) * SZ_RESOLUTION; | 177 | rawir.duration = ((int) value) * SZ_RESOLUTION; |
| 178 | rawir.duration += SZ_RESOLUTION / 2; | 178 | rawir.duration += SZ_RESOLUTION / 2; |
| 179 | sz->sum += rawir.duration; | 179 | sz->sum += rawir.duration; |
| 180 | rawir.duration *= 1000; | 180 | rawir.duration = US_TO_NS(rawir.duration); |
| 181 | rawir.duration &= IR_MAX_DURATION; | 181 | rawir.duration &= IR_MAX_DURATION; |
| 182 | sz_push(sz, rawir); | 182 | sz_push(sz, rawir); |
| 183 | } | 183 | } |
| @@ -197,7 +197,7 @@ static void sz_push_full_space(struct streamzap_ir *sz, | |||
| 197 | rawir.duration = ((int) value) * SZ_RESOLUTION; | 197 | rawir.duration = ((int) value) * SZ_RESOLUTION; |
| 198 | rawir.duration += SZ_RESOLUTION / 2; | 198 | rawir.duration += SZ_RESOLUTION / 2; |
| 199 | sz->sum += rawir.duration; | 199 | sz->sum += rawir.duration; |
| 200 | rawir.duration *= 1000; | 200 | rawir.duration = US_TO_NS(rawir.duration); |
| 201 | sz_push(sz, rawir); | 201 | sz_push(sz, rawir); |
| 202 | } | 202 | } |
| 203 | 203 | ||
| @@ -273,6 +273,7 @@ static void streamzap_callback(struct urb *urb) | |||
| 273 | if (sz->timeout_enabled) | 273 | if (sz->timeout_enabled) |
| 274 | sz_push(sz, rawir); | 274 | sz_push(sz, rawir); |
| 275 | ir_raw_event_handle(sz->rdev); | 275 | ir_raw_event_handle(sz->rdev); |
| 276 | ir_raw_event_reset(sz->rdev); | ||
| 276 | } else { | 277 | } else { |
| 277 | sz_push_full_space(sz, sz->buf_in[i]); | 278 | sz_push_full_space(sz, sz->buf_in[i]); |
| 278 | } | 279 | } |
| @@ -290,6 +291,7 @@ static void streamzap_callback(struct urb *urb) | |||
| 290 | } | 291 | } |
| 291 | } | 292 | } |
| 292 | 293 | ||
| 294 | ir_raw_event_handle(sz->rdev); | ||
| 293 | usb_submit_urb(urb, GFP_ATOMIC); | 295 | usb_submit_urb(urb, GFP_ATOMIC); |
| 294 | 296 | ||
| 295 | return; | 297 | return; |
| @@ -430,13 +432,13 @@ static int __devinit streamzap_probe(struct usb_interface *intf, | |||
| 430 | sz->decoder_state = PulseSpace; | 432 | sz->decoder_state = PulseSpace; |
| 431 | /* FIXME: don't yet have a way to set this */ | 433 | /* FIXME: don't yet have a way to set this */ |
| 432 | sz->timeout_enabled = true; | 434 | sz->timeout_enabled = true; |
| 433 | sz->rdev->timeout = (((SZ_TIMEOUT * SZ_RESOLUTION * 1000) & | 435 | sz->rdev->timeout = ((US_TO_NS(SZ_TIMEOUT * SZ_RESOLUTION) & |
| 434 | IR_MAX_DURATION) | 0x03000000); | 436 | IR_MAX_DURATION) | 0x03000000); |
| 435 | #if 0 | 437 | #if 0 |
| 436 | /* not yet supported, depends on patches from maxim */ | 438 | /* not yet supported, depends on patches from maxim */ |
| 437 | /* see also: LIRC_GET_REC_RESOLUTION and LIRC_SET_REC_TIMEOUT */ | 439 | /* see also: LIRC_GET_REC_RESOLUTION and LIRC_SET_REC_TIMEOUT */ |
| 438 | sz->min_timeout = SZ_TIMEOUT * SZ_RESOLUTION * 1000; | 440 | sz->min_timeout = US_TO_NS(SZ_TIMEOUT * SZ_RESOLUTION); |
| 439 | sz->max_timeout = SZ_TIMEOUT * SZ_RESOLUTION * 1000; | 441 | sz->max_timeout = US_TO_NS(SZ_TIMEOUT * SZ_RESOLUTION); |
| 440 | #endif | 442 | #endif |
| 441 | 443 | ||
| 442 | do_gettimeofday(&sz->signal_start); | 444 | do_gettimeofday(&sz->signal_start); |
diff --git a/drivers/media/video/gspca/zc3xx.c b/drivers/media/video/gspca/zc3xx.c index 865216e9362..47236a58bf3 100644 --- a/drivers/media/video/gspca/zc3xx.c +++ b/drivers/media/video/gspca/zc3xx.c | |||
| @@ -5793,7 +5793,7 @@ static void usb_exchange(struct gspca_dev *gspca_dev, | |||
| 5793 | break; | 5793 | break; |
| 5794 | default: | 5794 | default: |
| 5795 | /* case 0xdd: * delay */ | 5795 | /* case 0xdd: * delay */ |
| 5796 | msleep(action->val / 64 + 10); | 5796 | msleep(action->idx); |
| 5797 | break; | 5797 | break; |
| 5798 | } | 5798 | } |
| 5799 | action++; | 5799 | action++; |
| @@ -5830,7 +5830,7 @@ static void setmatrix(struct gspca_dev *gspca_dev) | |||
| 5830 | [SENSOR_GC0305] = gc0305_matrix, | 5830 | [SENSOR_GC0305] = gc0305_matrix, |
| 5831 | [SENSOR_HDCS2020b] = NULL, | 5831 | [SENSOR_HDCS2020b] = NULL, |
| 5832 | [SENSOR_HV7131B] = NULL, | 5832 | [SENSOR_HV7131B] = NULL, |
| 5833 | [SENSOR_HV7131R] = NULL, | 5833 | [SENSOR_HV7131R] = po2030_matrix, |
| 5834 | [SENSOR_ICM105A] = po2030_matrix, | 5834 | [SENSOR_ICM105A] = po2030_matrix, |
| 5835 | [SENSOR_MC501CB] = NULL, | 5835 | [SENSOR_MC501CB] = NULL, |
| 5836 | [SENSOR_MT9V111_1] = gc0305_matrix, | 5836 | [SENSOR_MT9V111_1] = gc0305_matrix, |
| @@ -5936,6 +5936,7 @@ static void setquality(struct gspca_dev *gspca_dev) | |||
| 5936 | case SENSOR_ADCM2700: | 5936 | case SENSOR_ADCM2700: |
| 5937 | case SENSOR_GC0305: | 5937 | case SENSOR_GC0305: |
| 5938 | case SENSOR_HV7131B: | 5938 | case SENSOR_HV7131B: |
| 5939 | case SENSOR_HV7131R: | ||
| 5939 | case SENSOR_OV7620: | 5940 | case SENSOR_OV7620: |
| 5940 | case SENSOR_PAS202B: | 5941 | case SENSOR_PAS202B: |
| 5941 | case SENSOR_PO2030: | 5942 | case SENSOR_PO2030: |
| @@ -6108,11 +6109,13 @@ static void send_unknown(struct gspca_dev *gspca_dev, int sensor) | |||
| 6108 | reg_w(gspca_dev, 0x02, 0x003b); | 6109 | reg_w(gspca_dev, 0x02, 0x003b); |
| 6109 | reg_w(gspca_dev, 0x00, 0x0038); | 6110 | reg_w(gspca_dev, 0x00, 0x0038); |
| 6110 | break; | 6111 | break; |
| 6112 | case SENSOR_HV7131R: | ||
| 6111 | case SENSOR_PAS202B: | 6113 | case SENSOR_PAS202B: |
| 6112 | reg_w(gspca_dev, 0x03, 0x003b); | 6114 | reg_w(gspca_dev, 0x03, 0x003b); |
| 6113 | reg_w(gspca_dev, 0x0c, 0x003a); | 6115 | reg_w(gspca_dev, 0x0c, 0x003a); |
| 6114 | reg_w(gspca_dev, 0x0b, 0x0039); | 6116 | reg_w(gspca_dev, 0x0b, 0x0039); |
| 6115 | reg_w(gspca_dev, 0x0b, 0x0038); | 6117 | if (sensor == SENSOR_PAS202B) |
| 6118 | reg_w(gspca_dev, 0x0b, 0x0038); | ||
| 6116 | break; | 6119 | break; |
| 6117 | } | 6120 | } |
| 6118 | } | 6121 | } |
| @@ -6704,10 +6707,13 @@ static int sd_start(struct gspca_dev *gspca_dev) | |||
| 6704 | reg_w(gspca_dev, 0x02, 0x003b); | 6707 | reg_w(gspca_dev, 0x02, 0x003b); |
| 6705 | reg_w(gspca_dev, 0x00, 0x0038); | 6708 | reg_w(gspca_dev, 0x00, 0x0038); |
| 6706 | break; | 6709 | break; |
| 6710 | case SENSOR_HV7131R: | ||
| 6707 | case SENSOR_PAS202B: | 6711 | case SENSOR_PAS202B: |
| 6708 | reg_w(gspca_dev, 0x03, 0x003b); | 6712 | reg_w(gspca_dev, 0x03, 0x003b); |
| 6709 | reg_w(gspca_dev, 0x0c, 0x003a); | 6713 | reg_w(gspca_dev, 0x0c, 0x003a); |
| 6710 | reg_w(gspca_dev, 0x0b, 0x0039); | 6714 | reg_w(gspca_dev, 0x0b, 0x0039); |
| 6715 | if (sd->sensor == SENSOR_HV7131R) | ||
| 6716 | reg_w(gspca_dev, 0x50, ZC3XX_R11D_GLOBALGAIN); | ||
| 6711 | break; | 6717 | break; |
| 6712 | } | 6718 | } |
| 6713 | 6719 | ||
| @@ -6720,6 +6726,7 @@ static int sd_start(struct gspca_dev *gspca_dev) | |||
| 6720 | break; | 6726 | break; |
| 6721 | case SENSOR_PAS202B: | 6727 | case SENSOR_PAS202B: |
| 6722 | case SENSOR_GC0305: | 6728 | case SENSOR_GC0305: |
| 6729 | case SENSOR_HV7131R: | ||
| 6723 | case SENSOR_TAS5130C: | 6730 | case SENSOR_TAS5130C: |
| 6724 | reg_r(gspca_dev, 0x0008); | 6731 | reg_r(gspca_dev, 0x0008); |
| 6725 | /* fall thru */ | 6732 | /* fall thru */ |
| @@ -6760,6 +6767,12 @@ static int sd_start(struct gspca_dev *gspca_dev) | |||
| 6760 | /* ms-win + */ | 6767 | /* ms-win + */ |
| 6761 | reg_w(gspca_dev, 0x40, 0x0117); | 6768 | reg_w(gspca_dev, 0x40, 0x0117); |
| 6762 | break; | 6769 | break; |
| 6770 | case SENSOR_HV7131R: | ||
| 6771 | i2c_write(gspca_dev, 0x25, 0x04, 0x00); /* exposure */ | ||
| 6772 | i2c_write(gspca_dev, 0x26, 0x93, 0x00); | ||
| 6773 | i2c_write(gspca_dev, 0x27, 0xe0, 0x00); | ||
| 6774 | reg_w(gspca_dev, 0x00, ZC3XX_R1A7_CALCGLOBALMEAN); | ||
| 6775 | break; | ||
| 6763 | case SENSOR_GC0305: | 6776 | case SENSOR_GC0305: |
| 6764 | case SENSOR_TAS5130C: | 6777 | case SENSOR_TAS5130C: |
| 6765 | reg_w(gspca_dev, 0x09, 0x01ad); /* (from win traces) */ | 6778 | reg_w(gspca_dev, 0x09, 0x01ad); /* (from win traces) */ |
| @@ -6808,9 +6821,17 @@ static void sd_pkt_scan(struct gspca_dev *gspca_dev, | |||
| 6808 | { | 6821 | { |
| 6809 | struct sd *sd = (struct sd *) gspca_dev; | 6822 | struct sd *sd = (struct sd *) gspca_dev; |
| 6810 | 6823 | ||
| 6811 | if (data[0] == 0xff && data[1] == 0xd8) { /* start of frame */ | 6824 | /* check the JPEG end of frame */ |
| 6825 | if (len >= 3 | ||
| 6826 | && data[len - 3] == 0xff && data[len - 2] == 0xd9) { | ||
| 6827 | /*fixme: what does the last byte mean?*/ | ||
| 6812 | gspca_frame_add(gspca_dev, LAST_PACKET, | 6828 | gspca_frame_add(gspca_dev, LAST_PACKET, |
| 6813 | NULL, 0); | 6829 | data, len - 1); |
| 6830 | return; | ||
| 6831 | } | ||
| 6832 | |||
| 6833 | /* check the JPEG start of a frame */ | ||
| 6834 | if (data[0] == 0xff && data[1] == 0xd8) { | ||
| 6814 | /* put the JPEG header in the new frame */ | 6835 | /* put the JPEG header in the new frame */ |
| 6815 | gspca_frame_add(gspca_dev, FIRST_PACKET, | 6836 | gspca_frame_add(gspca_dev, FIRST_PACKET, |
| 6816 | sd->jpeg_hdr, JPEG_HDR_SZ); | 6837 | sd->jpeg_hdr, JPEG_HDR_SZ); |
diff --git a/drivers/media/video/hdpvr/hdpvr-core.c b/drivers/media/video/hdpvr/hdpvr-core.c index a6572e5ae36..a27d93b503a 100644 --- a/drivers/media/video/hdpvr/hdpvr-core.c +++ b/drivers/media/video/hdpvr/hdpvr-core.c | |||
| @@ -283,6 +283,7 @@ static int hdpvr_probe(struct usb_interface *interface, | |||
| 283 | struct hdpvr_device *dev; | 283 | struct hdpvr_device *dev; |
| 284 | struct usb_host_interface *iface_desc; | 284 | struct usb_host_interface *iface_desc; |
| 285 | struct usb_endpoint_descriptor *endpoint; | 285 | struct usb_endpoint_descriptor *endpoint; |
| 286 | struct i2c_client *client; | ||
| 286 | size_t buffer_size; | 287 | size_t buffer_size; |
| 287 | int i; | 288 | int i; |
| 288 | int retval = -ENOMEM; | 289 | int retval = -ENOMEM; |
| @@ -381,13 +382,21 @@ static int hdpvr_probe(struct usb_interface *interface, | |||
| 381 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | 382 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) |
| 382 | retval = hdpvr_register_i2c_adapter(dev); | 383 | retval = hdpvr_register_i2c_adapter(dev); |
| 383 | if (retval < 0) { | 384 | if (retval < 0) { |
| 384 | v4l2_err(&dev->v4l2_dev, "registering i2c adapter failed\n"); | 385 | v4l2_err(&dev->v4l2_dev, "i2c adapter register failed\n"); |
| 385 | goto error; | 386 | goto error; |
| 386 | } | 387 | } |
| 387 | 388 | ||
| 388 | retval = hdpvr_register_i2c_ir(dev); | 389 | client = hdpvr_register_ir_rx_i2c(dev); |
| 389 | if (retval < 0) | 390 | if (!client) { |
| 390 | v4l2_err(&dev->v4l2_dev, "registering i2c IR devices failed\n"); | 391 | v4l2_err(&dev->v4l2_dev, "i2c IR RX device register failed\n"); |
| 392 | goto reg_fail; | ||
| 393 | } | ||
| 394 | |||
| 395 | client = hdpvr_register_ir_tx_i2c(dev); | ||
| 396 | if (!client) { | ||
| 397 | v4l2_err(&dev->v4l2_dev, "i2c IR TX device register failed\n"); | ||
| 398 | goto reg_fail; | ||
| 399 | } | ||
| 391 | #endif | 400 | #endif |
| 392 | 401 | ||
| 393 | /* let the user know what node this device is now attached to */ | 402 | /* let the user know what node this device is now attached to */ |
| @@ -395,6 +404,10 @@ static int hdpvr_probe(struct usb_interface *interface, | |||
| 395 | video_device_node_name(dev->video_dev)); | 404 | video_device_node_name(dev->video_dev)); |
| 396 | return 0; | 405 | return 0; |
| 397 | 406 | ||
| 407 | reg_fail: | ||
| 408 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | ||
| 409 | i2c_del_adapter(&dev->i2c_adapter); | ||
| 410 | #endif | ||
| 398 | error: | 411 | error: |
| 399 | if (dev) { | 412 | if (dev) { |
| 400 | /* Destroy single thread */ | 413 | /* Destroy single thread */ |
| @@ -424,6 +437,9 @@ static void hdpvr_disconnect(struct usb_interface *interface) | |||
| 424 | mutex_lock(&dev->io_mutex); | 437 | mutex_lock(&dev->io_mutex); |
| 425 | hdpvr_cancel_queue(dev); | 438 | hdpvr_cancel_queue(dev); |
| 426 | mutex_unlock(&dev->io_mutex); | 439 | mutex_unlock(&dev->io_mutex); |
| 440 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | ||
| 441 | i2c_del_adapter(&dev->i2c_adapter); | ||
| 442 | #endif | ||
| 427 | video_unregister_device(dev->video_dev); | 443 | video_unregister_device(dev->video_dev); |
| 428 | atomic_dec(&dev_nr); | 444 | atomic_dec(&dev_nr); |
| 429 | } | 445 | } |
diff --git a/drivers/media/video/hdpvr/hdpvr-i2c.c b/drivers/media/video/hdpvr/hdpvr-i2c.c index 89b71faeaac..e53fa55d56a 100644 --- a/drivers/media/video/hdpvr/hdpvr-i2c.c +++ b/drivers/media/video/hdpvr/hdpvr-i2c.c | |||
| @@ -31,26 +31,34 @@ | |||
| 31 | #define Z8F0811_IR_RX_I2C_ADDR 0x71 | 31 | #define Z8F0811_IR_RX_I2C_ADDR 0x71 |
| 32 | 32 | ||
| 33 | 33 | ||
| 34 | static struct i2c_board_info hdpvr_i2c_board_info = { | 34 | struct i2c_client *hdpvr_register_ir_tx_i2c(struct hdpvr_device *dev) |
| 35 | I2C_BOARD_INFO("ir_tx_z8f0811_hdpvr", Z8F0811_IR_TX_I2C_ADDR), | 35 | { |
| 36 | I2C_BOARD_INFO("ir_rx_z8f0811_hdpvr", Z8F0811_IR_RX_I2C_ADDR), | 36 | struct IR_i2c_init_data *init_data = &dev->ir_i2c_init_data; |
| 37 | }; | 37 | struct i2c_board_info hdpvr_ir_tx_i2c_board_info = { |
| 38 | I2C_BOARD_INFO("ir_tx_z8f0811_hdpvr", Z8F0811_IR_TX_I2C_ADDR), | ||
| 39 | }; | ||
| 40 | |||
| 41 | init_data->name = "HD-PVR"; | ||
| 42 | hdpvr_ir_tx_i2c_board_info.platform_data = init_data; | ||
| 38 | 43 | ||
| 39 | int hdpvr_register_i2c_ir(struct hdpvr_device *dev) | 44 | return i2c_new_device(&dev->i2c_adapter, &hdpvr_ir_tx_i2c_board_info); |
| 45 | } | ||
| 46 | |||
| 47 | struct i2c_client *hdpvr_register_ir_rx_i2c(struct hdpvr_device *dev) | ||
| 40 | { | 48 | { |
| 41 | struct i2c_client *c; | ||
| 42 | struct IR_i2c_init_data *init_data = &dev->ir_i2c_init_data; | 49 | struct IR_i2c_init_data *init_data = &dev->ir_i2c_init_data; |
| 50 | struct i2c_board_info hdpvr_ir_rx_i2c_board_info = { | ||
| 51 | I2C_BOARD_INFO("ir_rx_z8f0811_hdpvr", Z8F0811_IR_RX_I2C_ADDR), | ||
| 52 | }; | ||
| 43 | 53 | ||
| 44 | /* Our default information for ir-kbd-i2c.c to use */ | 54 | /* Our default information for ir-kbd-i2c.c to use */ |
| 45 | init_data->ir_codes = RC_MAP_HAUPPAUGE_NEW; | 55 | init_data->ir_codes = RC_MAP_HAUPPAUGE_NEW; |
| 46 | init_data->internal_get_key_func = IR_KBD_GET_KEY_HAUP_XVR; | 56 | init_data->internal_get_key_func = IR_KBD_GET_KEY_HAUP_XVR; |
| 47 | init_data->type = RC_TYPE_RC5; | 57 | init_data->type = RC_TYPE_RC5; |
| 48 | init_data->name = "HD PVR"; | 58 | init_data->name = "HD-PVR"; |
| 49 | hdpvr_i2c_board_info.platform_data = init_data; | 59 | hdpvr_ir_rx_i2c_board_info.platform_data = init_data; |
| 50 | |||
| 51 | c = i2c_new_device(&dev->i2c_adapter, &hdpvr_i2c_board_info); | ||
| 52 | 60 | ||
| 53 | return (c == NULL) ? -ENODEV : 0; | 61 | return i2c_new_device(&dev->i2c_adapter, &hdpvr_ir_rx_i2c_board_info); |
| 54 | } | 62 | } |
| 55 | 63 | ||
| 56 | static int hdpvr_i2c_read(struct hdpvr_device *dev, int bus, | 64 | static int hdpvr_i2c_read(struct hdpvr_device *dev, int bus, |
diff --git a/drivers/media/video/hdpvr/hdpvr.h b/drivers/media/video/hdpvr/hdpvr.h index ee74e3be9a6..072f23c570f 100644 --- a/drivers/media/video/hdpvr/hdpvr.h +++ b/drivers/media/video/hdpvr/hdpvr.h | |||
| @@ -313,7 +313,8 @@ int hdpvr_cancel_queue(struct hdpvr_device *dev); | |||
| 313 | /* i2c adapter registration */ | 313 | /* i2c adapter registration */ |
| 314 | int hdpvr_register_i2c_adapter(struct hdpvr_device *dev); | 314 | int hdpvr_register_i2c_adapter(struct hdpvr_device *dev); |
| 315 | 315 | ||
| 316 | int hdpvr_register_i2c_ir(struct hdpvr_device *dev); | 316 | struct i2c_client *hdpvr_register_ir_rx_i2c(struct hdpvr_device *dev); |
| 317 | struct i2c_client *hdpvr_register_ir_tx_i2c(struct hdpvr_device *dev); | ||
| 317 | 318 | ||
| 318 | /*========================================================================*/ | 319 | /*========================================================================*/ |
| 319 | /* buffer management */ | 320 | /* buffer management */ |
diff --git a/drivers/media/video/ir-kbd-i2c.c b/drivers/media/video/ir-kbd-i2c.c index d2b20ad383a..a221ad68b33 100644 --- a/drivers/media/video/ir-kbd-i2c.c +++ b/drivers/media/video/ir-kbd-i2c.c | |||
| @@ -128,6 +128,19 @@ static int get_key_haup(struct IR_i2c *ir, u32 *ir_key, u32 *ir_raw) | |||
| 128 | 128 | ||
| 129 | static int get_key_haup_xvr(struct IR_i2c *ir, u32 *ir_key, u32 *ir_raw) | 129 | static int get_key_haup_xvr(struct IR_i2c *ir, u32 *ir_key, u32 *ir_raw) |
| 130 | { | 130 | { |
| 131 | int ret; | ||
| 132 | unsigned char buf[1] = { 0 }; | ||
| 133 | |||
| 134 | /* | ||
| 135 | * This is the same apparent "are you ready?" poll command observed | ||
| 136 | * watching Windows driver traffic and implemented in lirc_zilog. With | ||
| 137 | * this added, we get far saner remote behavior with z8 chips on usb | ||
| 138 | * connected devices, even with the default polling interval of 100ms. | ||
| 139 | */ | ||
| 140 | ret = i2c_master_send(ir->c, buf, 1); | ||
| 141 | if (ret != 1) | ||
| 142 | return (ret < 0) ? ret : -EINVAL; | ||
| 143 | |||
| 131 | return get_key_haup_common (ir, ir_key, ir_raw, 6, 3); | 144 | return get_key_haup_common (ir, ir_key, ir_raw, 6, 3); |
| 132 | } | 145 | } |
| 133 | 146 | ||
diff --git a/drivers/media/video/pvrusb2/pvrusb2-i2c-core.c b/drivers/media/video/pvrusb2/pvrusb2-i2c-core.c index ccc884948f3..451ecd485f9 100644 --- a/drivers/media/video/pvrusb2/pvrusb2-i2c-core.c +++ b/drivers/media/video/pvrusb2/pvrusb2-i2c-core.c | |||
| @@ -597,7 +597,6 @@ static void pvr2_i2c_register_ir(struct pvr2_hdw *hdw) | |||
| 597 | init_data->internal_get_key_func = IR_KBD_GET_KEY_HAUP_XVR; | 597 | init_data->internal_get_key_func = IR_KBD_GET_KEY_HAUP_XVR; |
| 598 | init_data->type = RC_TYPE_RC5; | 598 | init_data->type = RC_TYPE_RC5; |
| 599 | init_data->name = hdw->hdw_desc->description; | 599 | init_data->name = hdw->hdw_desc->description; |
| 600 | init_data->polling_interval = 260; /* ms From lirc_zilog */ | ||
| 601 | /* IR Receiver */ | 600 | /* IR Receiver */ |
| 602 | info.addr = 0x71; | 601 | info.addr = 0x71; |
| 603 | info.platform_data = init_data; | 602 | info.platform_data = init_data; |
diff --git a/drivers/media/video/saa7115.c b/drivers/media/video/saa7115.c index f35459d1f42..0db90922ee9 100644 --- a/drivers/media/video/saa7115.c +++ b/drivers/media/video/saa7115.c | |||
| @@ -1565,7 +1565,7 @@ static int saa711x_probe(struct i2c_client *client, | |||
| 1565 | chip_id = name[5]; | 1565 | chip_id = name[5]; |
| 1566 | 1566 | ||
| 1567 | /* Check whether this chip is part of the saa711x series */ | 1567 | /* Check whether this chip is part of the saa711x series */ |
| 1568 | if (memcmp(name, "1f711", 5)) { | 1568 | if (memcmp(name + 1, "f711", 4)) { |
| 1569 | v4l_dbg(1, debug, client, "chip found @ 0x%x (ID %s) does not match a known saa711x chip.\n", | 1569 | v4l_dbg(1, debug, client, "chip found @ 0x%x (ID %s) does not match a known saa711x chip.\n", |
| 1570 | client->addr << 1, name); | 1570 | client->addr << 1, name); |
| 1571 | return -ENODEV; | 1571 | return -ENODEV; |
diff --git a/drivers/mmc/host/bfin_sdh.c b/drivers/mmc/host/bfin_sdh.c index bac7d62866b..0371bf50224 100644 --- a/drivers/mmc/host/bfin_sdh.c +++ b/drivers/mmc/host/bfin_sdh.c | |||
| @@ -462,7 +462,7 @@ static int __devinit sdh_probe(struct platform_device *pdev) | |||
| 462 | goto out; | 462 | goto out; |
| 463 | } | 463 | } |
| 464 | 464 | ||
| 465 | mmc = mmc_alloc_host(sizeof(*mmc), &pdev->dev); | 465 | mmc = mmc_alloc_host(sizeof(struct sdh_host), &pdev->dev); |
| 466 | if (!mmc) { | 466 | if (!mmc) { |
| 467 | ret = -ENOMEM; | 467 | ret = -ENOMEM; |
| 468 | goto out; | 468 | goto out; |
diff --git a/drivers/mmc/host/jz4740_mmc.c b/drivers/mmc/host/jz4740_mmc.c index b3a0ab0e4c2..74218ad677e 100644 --- a/drivers/mmc/host/jz4740_mmc.c +++ b/drivers/mmc/host/jz4740_mmc.c | |||
| @@ -14,6 +14,7 @@ | |||
| 14 | */ | 14 | */ |
| 15 | 15 | ||
| 16 | #include <linux/mmc/host.h> | 16 | #include <linux/mmc/host.h> |
| 17 | #include <linux/err.h> | ||
| 17 | #include <linux/io.h> | 18 | #include <linux/io.h> |
| 18 | #include <linux/irq.h> | 19 | #include <linux/irq.h> |
| 19 | #include <linux/interrupt.h> | 20 | #include <linux/interrupt.h> |
| @@ -827,8 +828,8 @@ static int __devinit jz4740_mmc_probe(struct platform_device* pdev) | |||
| 827 | } | 828 | } |
| 828 | 829 | ||
| 829 | host->clk = clk_get(&pdev->dev, "mmc"); | 830 | host->clk = clk_get(&pdev->dev, "mmc"); |
| 830 | if (!host->clk) { | 831 | if (IS_ERR(host->clk)) { |
| 831 | ret = -ENOENT; | 832 | ret = PTR_ERR(host->clk); |
| 832 | dev_err(&pdev->dev, "Failed to get mmc clock\n"); | 833 | dev_err(&pdev->dev, "Failed to get mmc clock\n"); |
| 833 | goto err_free_host; | 834 | goto err_free_host; |
| 834 | } | 835 | } |
diff --git a/drivers/mmc/host/mmci.c b/drivers/mmc/host/mmci.c index 56302282566..2d6de3e03e2 100644 --- a/drivers/mmc/host/mmci.c +++ b/drivers/mmc/host/mmci.c | |||
| @@ -14,6 +14,7 @@ | |||
| 14 | #include <linux/ioport.h> | 14 | #include <linux/ioport.h> |
| 15 | #include <linux/device.h> | 15 | #include <linux/device.h> |
| 16 | #include <linux/interrupt.h> | 16 | #include <linux/interrupt.h> |
| 17 | #include <linux/kernel.h> | ||
| 17 | #include <linux/delay.h> | 18 | #include <linux/delay.h> |
| 18 | #include <linux/err.h> | 19 | #include <linux/err.h> |
| 19 | #include <linux/highmem.h> | 20 | #include <linux/highmem.h> |
| @@ -46,10 +47,6 @@ static unsigned int fmax = 515633; | |||
| 46 | * is asserted (likewise for RX) | 47 | * is asserted (likewise for RX) |
| 47 | * @fifohalfsize: number of bytes that can be written when MCI_TXFIFOHALFEMPTY | 48 | * @fifohalfsize: number of bytes that can be written when MCI_TXFIFOHALFEMPTY |
| 48 | * is asserted (likewise for RX) | 49 | * is asserted (likewise for RX) |
| 49 | * @broken_blockend: the MCI_DATABLOCKEND is broken on the hardware | ||
| 50 | * and will not work at all. | ||
| 51 | * @broken_blockend_dma: the MCI_DATABLOCKEND is broken on the hardware when | ||
| 52 | * using DMA. | ||
| 53 | * @sdio: variant supports SDIO | 50 | * @sdio: variant supports SDIO |
| 54 | * @st_clkdiv: true if using a ST-specific clock divider algorithm | 51 | * @st_clkdiv: true if using a ST-specific clock divider algorithm |
| 55 | */ | 52 | */ |
| @@ -59,8 +56,6 @@ struct variant_data { | |||
| 59 | unsigned int datalength_bits; | 56 | unsigned int datalength_bits; |
| 60 | unsigned int fifosize; | 57 | unsigned int fifosize; |
| 61 | unsigned int fifohalfsize; | 58 | unsigned int fifohalfsize; |
| 62 | bool broken_blockend; | ||
| 63 | bool broken_blockend_dma; | ||
| 64 | bool sdio; | 59 | bool sdio; |
| 65 | bool st_clkdiv; | 60 | bool st_clkdiv; |
| 66 | }; | 61 | }; |
| @@ -76,7 +71,6 @@ static struct variant_data variant_u300 = { | |||
| 76 | .fifohalfsize = 8 * 4, | 71 | .fifohalfsize = 8 * 4, |
| 77 | .clkreg_enable = 1 << 13, /* HWFCEN */ | 72 | .clkreg_enable = 1 << 13, /* HWFCEN */ |
| 78 | .datalength_bits = 16, | 73 | .datalength_bits = 16, |
| 79 | .broken_blockend_dma = true, | ||
| 80 | .sdio = true, | 74 | .sdio = true, |
| 81 | }; | 75 | }; |
| 82 | 76 | ||
| @@ -86,7 +80,6 @@ static struct variant_data variant_ux500 = { | |||
| 86 | .clkreg = MCI_CLK_ENABLE, | 80 | .clkreg = MCI_CLK_ENABLE, |
| 87 | .clkreg_enable = 1 << 14, /* HWFCEN */ | 81 | .clkreg_enable = 1 << 14, /* HWFCEN */ |
| 88 | .datalength_bits = 24, | 82 | .datalength_bits = 24, |
| 89 | .broken_blockend = true, | ||
| 90 | .sdio = true, | 83 | .sdio = true, |
| 91 | .st_clkdiv = true, | 84 | .st_clkdiv = true, |
| 92 | }; | 85 | }; |
| @@ -210,8 +203,6 @@ static void mmci_start_data(struct mmci_host *host, struct mmc_data *data) | |||
| 210 | host->data = data; | 203 | host->data = data; |
| 211 | host->size = data->blksz * data->blocks; | 204 | host->size = data->blksz * data->blocks; |
| 212 | host->data_xfered = 0; | 205 | host->data_xfered = 0; |
| 213 | host->blockend = false; | ||
| 214 | host->dataend = false; | ||
| 215 | 206 | ||
| 216 | mmci_init_sg(host, data); | 207 | mmci_init_sg(host, data); |
| 217 | 208 | ||
| @@ -288,21 +279,26 @@ static void | |||
| 288 | mmci_data_irq(struct mmci_host *host, struct mmc_data *data, | 279 | mmci_data_irq(struct mmci_host *host, struct mmc_data *data, |
| 289 | unsigned int status) | 280 | unsigned int status) |
| 290 | { | 281 | { |
| 291 | struct variant_data *variant = host->variant; | ||
| 292 | |||
| 293 | /* First check for errors */ | 282 | /* First check for errors */ |
| 294 | if (status & (MCI_DATACRCFAIL|MCI_DATATIMEOUT|MCI_TXUNDERRUN|MCI_RXOVERRUN)) { | 283 | if (status & (MCI_DATACRCFAIL|MCI_DATATIMEOUT|MCI_TXUNDERRUN|MCI_RXOVERRUN)) { |
| 284 | u32 remain, success; | ||
| 285 | |||
| 286 | /* Calculate how far we are into the transfer */ | ||
| 287 | remain = readl(host->base + MMCIDATACNT); | ||
| 288 | success = data->blksz * data->blocks - remain; | ||
| 289 | |||
| 295 | dev_dbg(mmc_dev(host->mmc), "MCI ERROR IRQ (status %08x)\n", status); | 290 | dev_dbg(mmc_dev(host->mmc), "MCI ERROR IRQ (status %08x)\n", status); |
| 296 | if (status & MCI_DATACRCFAIL) | 291 | if (status & MCI_DATACRCFAIL) { |
| 292 | /* Last block was not successful */ | ||
| 293 | host->data_xfered = round_down(success - 1, data->blksz); | ||
| 297 | data->error = -EILSEQ; | 294 | data->error = -EILSEQ; |
| 298 | else if (status & MCI_DATATIMEOUT) | 295 | } else if (status & MCI_DATATIMEOUT) { |
| 296 | host->data_xfered = round_down(success, data->blksz); | ||
| 299 | data->error = -ETIMEDOUT; | 297 | data->error = -ETIMEDOUT; |
| 300 | else if (status & (MCI_TXUNDERRUN|MCI_RXOVERRUN)) | 298 | } else if (status & (MCI_TXUNDERRUN|MCI_RXOVERRUN)) { |
| 299 | host->data_xfered = round_down(success, data->blksz); | ||
| 301 | data->error = -EIO; | 300 | data->error = -EIO; |
| 302 | 301 | } | |
| 303 | /* Force-complete the transaction */ | ||
| 304 | host->blockend = true; | ||
| 305 | host->dataend = true; | ||
| 306 | 302 | ||
| 307 | /* | 303 | /* |
| 308 | * We hit an error condition. Ensure that any data | 304 | * We hit an error condition. Ensure that any data |
| @@ -321,61 +317,14 @@ mmci_data_irq(struct mmci_host *host, struct mmc_data *data, | |||
| 321 | } | 317 | } |
| 322 | } | 318 | } |
| 323 | 319 | ||
| 324 | /* | 320 | if (status & MCI_DATABLOCKEND) |
| 325 | * On ARM variants in PIO mode, MCI_DATABLOCKEND | 321 | dev_err(mmc_dev(host->mmc), "stray MCI_DATABLOCKEND interrupt\n"); |
| 326 | * is always sent first, and we increase the | ||
| 327 | * transfered number of bytes for that IRQ. Then | ||
| 328 | * MCI_DATAEND follows and we conclude the transaction. | ||
| 329 | * | ||
| 330 | * On the Ux500 single-IRQ variant MCI_DATABLOCKEND | ||
| 331 | * doesn't seem to immediately clear from the status, | ||
| 332 | * so we can't use it keep count when only one irq is | ||
| 333 | * used because the irq will hit for other reasons, and | ||
| 334 | * then the flag is still up. So we use the MCI_DATAEND | ||
| 335 | * IRQ at the end of the entire transfer because | ||
| 336 | * MCI_DATABLOCKEND is broken. | ||
| 337 | * | ||
| 338 | * In the U300, the IRQs can arrive out-of-order, | ||
| 339 | * e.g. MCI_DATABLOCKEND sometimes arrives after MCI_DATAEND, | ||
| 340 | * so for this case we use the flags "blockend" and | ||
| 341 | * "dataend" to make sure both IRQs have arrived before | ||
| 342 | * concluding the transaction. (This does not apply | ||
| 343 | * to the Ux500 which doesn't fire MCI_DATABLOCKEND | ||
| 344 | * at all.) In DMA mode it suffers from the same problem | ||
| 345 | * as the Ux500. | ||
| 346 | */ | ||
| 347 | if (status & MCI_DATABLOCKEND) { | ||
| 348 | /* | ||
| 349 | * Just being a little over-cautious, we do not | ||
| 350 | * use this progressive update if the hardware blockend | ||
| 351 | * flag is unreliable: since it can stay high between | ||
| 352 | * IRQs it will corrupt the transfer counter. | ||
| 353 | */ | ||
| 354 | if (!variant->broken_blockend) | ||
| 355 | host->data_xfered += data->blksz; | ||
| 356 | host->blockend = true; | ||
| 357 | } | ||
| 358 | |||
| 359 | if (status & MCI_DATAEND) | ||
| 360 | host->dataend = true; | ||
| 361 | 322 | ||
| 362 | /* | 323 | if (status & MCI_DATAEND || data->error) { |
| 363 | * On variants with broken blockend we shall only wait for dataend, | ||
| 364 | * on others we must sync with the blockend signal since they can | ||
| 365 | * appear out-of-order. | ||
| 366 | */ | ||
| 367 | if (host->dataend && (host->blockend || variant->broken_blockend)) { | ||
| 368 | mmci_stop_data(host); | 324 | mmci_stop_data(host); |
| 369 | 325 | ||
| 370 | /* Reset these flags */ | 326 | if (!data->error) |
| 371 | host->blockend = false; | 327 | /* The error clause is handled above, success! */ |
| 372 | host->dataend = false; | ||
| 373 | |||
| 374 | /* | ||
| 375 | * Variants with broken blockend flags need to handle the | ||
| 376 | * end of the entire transfer here. | ||
| 377 | */ | ||
| 378 | if (variant->broken_blockend && !data->error) | ||
| 379 | host->data_xfered += data->blksz * data->blocks; | 328 | host->data_xfered += data->blksz * data->blocks; |
| 380 | 329 | ||
| 381 | if (!data->stop) { | 330 | if (!data->stop) { |
| @@ -394,15 +343,15 @@ mmci_cmd_irq(struct mmci_host *host, struct mmc_command *cmd, | |||
| 394 | 343 | ||
| 395 | host->cmd = NULL; | 344 | host->cmd = NULL; |
| 396 | 345 | ||
| 397 | cmd->resp[0] = readl(base + MMCIRESPONSE0); | ||
| 398 | cmd->resp[1] = readl(base + MMCIRESPONSE1); | ||
| 399 | cmd->resp[2] = readl(base + MMCIRESPONSE2); | ||
| 400 | cmd->resp[3] = readl(base + MMCIRESPONSE3); | ||
| 401 | |||
| 402 | if (status & MCI_CMDTIMEOUT) { | 346 | if (status & MCI_CMDTIMEOUT) { |
| 403 | cmd->error = -ETIMEDOUT; | 347 | cmd->error = -ETIMEDOUT; |
| 404 | } else if (status & MCI_CMDCRCFAIL && cmd->flags & MMC_RSP_CRC) { | 348 | } else if (status & MCI_CMDCRCFAIL && cmd->flags & MMC_RSP_CRC) { |
| 405 | cmd->error = -EILSEQ; | 349 | cmd->error = -EILSEQ; |
| 350 | } else { | ||
| 351 | cmd->resp[0] = readl(base + MMCIRESPONSE0); | ||
| 352 | cmd->resp[1] = readl(base + MMCIRESPONSE1); | ||
| 353 | cmd->resp[2] = readl(base + MMCIRESPONSE2); | ||
| 354 | cmd->resp[3] = readl(base + MMCIRESPONSE3); | ||
| 406 | } | 355 | } |
| 407 | 356 | ||
| 408 | if (!cmd->data || cmd->error) { | 357 | if (!cmd->data || cmd->error) { |
| @@ -770,7 +719,6 @@ static int __devinit mmci_probe(struct amba_device *dev, struct amba_id *id) | |||
| 770 | struct variant_data *variant = id->data; | 719 | struct variant_data *variant = id->data; |
| 771 | struct mmci_host *host; | 720 | struct mmci_host *host; |
| 772 | struct mmc_host *mmc; | 721 | struct mmc_host *mmc; |
| 773 | unsigned int mask; | ||
| 774 | int ret; | 722 | int ret; |
| 775 | 723 | ||
| 776 | /* must have platform data */ | 724 | /* must have platform data */ |
| @@ -951,12 +899,7 @@ static int __devinit mmci_probe(struct amba_device *dev, struct amba_id *id) | |||
| 951 | goto irq0_free; | 899 | goto irq0_free; |
| 952 | } | 900 | } |
| 953 | 901 | ||
| 954 | mask = MCI_IRQENABLE; | 902 | writel(MCI_IRQENABLE, host->base + MMCIMASK0); |
| 955 | /* Don't use the datablockend flag if it's broken */ | ||
| 956 | if (variant->broken_blockend) | ||
| 957 | mask &= ~MCI_DATABLOCKEND; | ||
| 958 | |||
| 959 | writel(mask, host->base + MMCIMASK0); | ||
| 960 | 903 | ||
| 961 | amba_set_drvdata(dev, mmc); | 904 | amba_set_drvdata(dev, mmc); |
| 962 | 905 | ||
diff --git a/drivers/mmc/host/mmci.h b/drivers/mmc/host/mmci.h index df06f01aac8..c1df7b82d36 100644 --- a/drivers/mmc/host/mmci.h +++ b/drivers/mmc/host/mmci.h | |||
| @@ -137,7 +137,7 @@ | |||
| 137 | #define MCI_IRQENABLE \ | 137 | #define MCI_IRQENABLE \ |
| 138 | (MCI_CMDCRCFAILMASK|MCI_DATACRCFAILMASK|MCI_CMDTIMEOUTMASK| \ | 138 | (MCI_CMDCRCFAILMASK|MCI_DATACRCFAILMASK|MCI_CMDTIMEOUTMASK| \ |
| 139 | MCI_DATATIMEOUTMASK|MCI_TXUNDERRUNMASK|MCI_RXOVERRUNMASK| \ | 139 | MCI_DATATIMEOUTMASK|MCI_TXUNDERRUNMASK|MCI_RXOVERRUNMASK| \ |
| 140 | MCI_CMDRESPENDMASK|MCI_CMDSENTMASK|MCI_DATABLOCKENDMASK) | 140 | MCI_CMDRESPENDMASK|MCI_CMDSENTMASK) |
| 141 | 141 | ||
| 142 | /* These interrupts are directed to IRQ1 when two IRQ lines are available */ | 142 | /* These interrupts are directed to IRQ1 when two IRQ lines are available */ |
| 143 | #define MCI_IRQ1MASK \ | 143 | #define MCI_IRQ1MASK \ |
| @@ -177,9 +177,6 @@ struct mmci_host { | |||
| 177 | struct timer_list timer; | 177 | struct timer_list timer; |
| 178 | unsigned int oldstat; | 178 | unsigned int oldstat; |
| 179 | 179 | ||
| 180 | bool blockend; | ||
| 181 | bool dataend; | ||
| 182 | |||
| 183 | /* pio stuff */ | 180 | /* pio stuff */ |
| 184 | struct sg_mapping_iter sg_miter; | 181 | struct sg_mapping_iter sg_miter; |
| 185 | unsigned int size; | 182 | unsigned int size; |
diff --git a/drivers/mmc/host/msm_sdcc.c b/drivers/mmc/host/msm_sdcc.c index 5decfd0bd61..153ab977a01 100644 --- a/drivers/mmc/host/msm_sdcc.c +++ b/drivers/mmc/host/msm_sdcc.c | |||
| @@ -383,14 +383,30 @@ static int msmsdcc_config_dma(struct msmsdcc_host *host, struct mmc_data *data) | |||
| 383 | host->curr.user_pages = 0; | 383 | host->curr.user_pages = 0; |
| 384 | 384 | ||
| 385 | box = &nc->cmd[0]; | 385 | box = &nc->cmd[0]; |
| 386 | for (i = 0; i < host->dma.num_ents; i++) { | ||
| 387 | box->cmd = CMD_MODE_BOX; | ||
| 388 | 386 | ||
| 389 | /* Initialize sg dma address */ | 387 | /* location of command block must be 64 bit aligned */ |
| 390 | sg->dma_address = page_to_dma(mmc_dev(host->mmc), sg_page(sg)) | 388 | BUG_ON(host->dma.cmd_busaddr & 0x07); |
| 391 | + sg->offset; | ||
| 392 | 389 | ||
| 393 | if (i == (host->dma.num_ents - 1)) | 390 | nc->cmdptr = (host->dma.cmd_busaddr >> 3) | CMD_PTR_LP; |
| 391 | host->dma.hdr.cmdptr = DMOV_CMD_PTR_LIST | | ||
| 392 | DMOV_CMD_ADDR(host->dma.cmdptr_busaddr); | ||
| 393 | host->dma.hdr.complete_func = msmsdcc_dma_complete_func; | ||
| 394 | |||
| 395 | n = dma_map_sg(mmc_dev(host->mmc), host->dma.sg, | ||
| 396 | host->dma.num_ents, host->dma.dir); | ||
| 397 | if (n == 0) { | ||
| 398 | printk(KERN_ERR "%s: Unable to map in all sg elements\n", | ||
| 399 | mmc_hostname(host->mmc)); | ||
| 400 | host->dma.sg = NULL; | ||
| 401 | host->dma.num_ents = 0; | ||
| 402 | return -ENOMEM; | ||
| 403 | } | ||
| 404 | |||
| 405 | for_each_sg(host->dma.sg, sg, n, i) { | ||
| 406 | |||
| 407 | box->cmd = CMD_MODE_BOX; | ||
| 408 | |||
| 409 | if (i == n - 1) | ||
| 394 | box->cmd |= CMD_LC; | 410 | box->cmd |= CMD_LC; |
| 395 | rows = (sg_dma_len(sg) % MCI_FIFOSIZE) ? | 411 | rows = (sg_dma_len(sg) % MCI_FIFOSIZE) ? |
| 396 | (sg_dma_len(sg) / MCI_FIFOSIZE) + 1 : | 412 | (sg_dma_len(sg) / MCI_FIFOSIZE) + 1 : |
| @@ -418,27 +434,6 @@ static int msmsdcc_config_dma(struct msmsdcc_host *host, struct mmc_data *data) | |||
| 418 | box->cmd |= CMD_DST_CRCI(crci); | 434 | box->cmd |= CMD_DST_CRCI(crci); |
| 419 | } | 435 | } |
| 420 | box++; | 436 | box++; |
| 421 | sg++; | ||
| 422 | } | ||
| 423 | |||
| 424 | /* location of command block must be 64 bit aligned */ | ||
| 425 | BUG_ON(host->dma.cmd_busaddr & 0x07); | ||
| 426 | |||
| 427 | nc->cmdptr = (host->dma.cmd_busaddr >> 3) | CMD_PTR_LP; | ||
| 428 | host->dma.hdr.cmdptr = DMOV_CMD_PTR_LIST | | ||
| 429 | DMOV_CMD_ADDR(host->dma.cmdptr_busaddr); | ||
| 430 | host->dma.hdr.complete_func = msmsdcc_dma_complete_func; | ||
| 431 | |||
| 432 | n = dma_map_sg(mmc_dev(host->mmc), host->dma.sg, | ||
| 433 | host->dma.num_ents, host->dma.dir); | ||
| 434 | /* dsb inside dma_map_sg will write nc out to mem as well */ | ||
| 435 | |||
| 436 | if (n != host->dma.num_ents) { | ||
| 437 | printk(KERN_ERR "%s: Unable to map in all sg elements\n", | ||
| 438 | mmc_hostname(host->mmc)); | ||
| 439 | host->dma.sg = NULL; | ||
| 440 | host->dma.num_ents = 0; | ||
| 441 | return -ENOMEM; | ||
| 442 | } | 437 | } |
| 443 | 438 | ||
| 444 | return 0; | 439 | return 0; |
| @@ -1331,9 +1326,6 @@ msmsdcc_probe(struct platform_device *pdev) | |||
| 1331 | if (host->timer.function) | 1326 | if (host->timer.function) |
| 1332 | pr_info("%s: Polling status mode enabled\n", mmc_hostname(mmc)); | 1327 | pr_info("%s: Polling status mode enabled\n", mmc_hostname(mmc)); |
| 1333 | 1328 | ||
| 1334 | #if BUSCLK_PWRSAVE | ||
| 1335 | msmsdcc_disable_clocks(host, 1); | ||
| 1336 | #endif | ||
| 1337 | return 0; | 1329 | return 0; |
| 1338 | cmd_irq_free: | 1330 | cmd_irq_free: |
| 1339 | free_irq(cmd_irqres->start, host); | 1331 | free_irq(cmd_irqres->start, host); |
diff --git a/drivers/mmc/host/sdhci-s3c.c b/drivers/mmc/host/sdhci-s3c.c index 17203586305..5309ab95aad 100644 --- a/drivers/mmc/host/sdhci-s3c.c +++ b/drivers/mmc/host/sdhci-s3c.c | |||
| @@ -277,10 +277,43 @@ static void sdhci_cmu_set_clock(struct sdhci_host *host, unsigned int clock) | |||
| 277 | host->clock = clock; | 277 | host->clock = clock; |
| 278 | } | 278 | } |
| 279 | 279 | ||
| 280 | /** | ||
| 281 | * sdhci_s3c_platform_8bit_width - support 8bit buswidth | ||
| 282 | * @host: The SDHCI host being queried | ||
| 283 | * @width: MMC_BUS_WIDTH_ macro for the bus width being requested | ||
| 284 | * | ||
| 285 | * We have 8-bit width support but is not a v3 controller. | ||
| 286 | * So we add platform_8bit_width() and support 8bit width. | ||
| 287 | */ | ||
| 288 | static int sdhci_s3c_platform_8bit_width(struct sdhci_host *host, int width) | ||
| 289 | { | ||
| 290 | u8 ctrl; | ||
| 291 | |||
| 292 | ctrl = sdhci_readb(host, SDHCI_HOST_CONTROL); | ||
| 293 | |||
| 294 | switch (width) { | ||
| 295 | case MMC_BUS_WIDTH_8: | ||
| 296 | ctrl |= SDHCI_CTRL_8BITBUS; | ||
| 297 | ctrl &= ~SDHCI_CTRL_4BITBUS; | ||
| 298 | break; | ||
| 299 | case MMC_BUS_WIDTH_4: | ||
| 300 | ctrl |= SDHCI_CTRL_4BITBUS; | ||
| 301 | ctrl &= ~SDHCI_CTRL_8BITBUS; | ||
| 302 | break; | ||
| 303 | default: | ||
| 304 | break; | ||
| 305 | } | ||
| 306 | |||
| 307 | sdhci_writeb(host, ctrl, SDHCI_HOST_CONTROL); | ||
| 308 | |||
| 309 | return 0; | ||
| 310 | } | ||
| 311 | |||
| 280 | static struct sdhci_ops sdhci_s3c_ops = { | 312 | static struct sdhci_ops sdhci_s3c_ops = { |
| 281 | .get_max_clock = sdhci_s3c_get_max_clk, | 313 | .get_max_clock = sdhci_s3c_get_max_clk, |
| 282 | .set_clock = sdhci_s3c_set_clock, | 314 | .set_clock = sdhci_s3c_set_clock, |
| 283 | .get_min_clock = sdhci_s3c_get_min_clock, | 315 | .get_min_clock = sdhci_s3c_get_min_clock, |
| 316 | .platform_8bit_width = sdhci_s3c_platform_8bit_width, | ||
| 284 | }; | 317 | }; |
| 285 | 318 | ||
| 286 | static void sdhci_s3c_notify_change(struct platform_device *dev, int state) | 319 | static void sdhci_s3c_notify_change(struct platform_device *dev, int state) |
| @@ -473,6 +506,9 @@ static int __devinit sdhci_s3c_probe(struct platform_device *pdev) | |||
| 473 | if (pdata->cd_type == S3C_SDHCI_CD_PERMANENT) | 506 | if (pdata->cd_type == S3C_SDHCI_CD_PERMANENT) |
| 474 | host->mmc->caps = MMC_CAP_NONREMOVABLE; | 507 | host->mmc->caps = MMC_CAP_NONREMOVABLE; |
| 475 | 508 | ||
| 509 | if (pdata->host_caps) | ||
| 510 | host->mmc->caps |= pdata->host_caps; | ||
| 511 | |||
| 476 | host->quirks |= (SDHCI_QUIRK_32BIT_DMA_ADDR | | 512 | host->quirks |= (SDHCI_QUIRK_32BIT_DMA_ADDR | |
| 477 | SDHCI_QUIRK_32BIT_DMA_SIZE); | 513 | SDHCI_QUIRK_32BIT_DMA_SIZE); |
| 478 | 514 | ||
diff --git a/drivers/mmc/host/ushc.c b/drivers/mmc/host/ushc.c index f8f65df9b01..f08f944ac53 100644 --- a/drivers/mmc/host/ushc.c +++ b/drivers/mmc/host/ushc.c | |||
| @@ -19,7 +19,6 @@ | |||
| 19 | #include <linux/module.h> | 19 | #include <linux/module.h> |
| 20 | #include <linux/usb.h> | 20 | #include <linux/usb.h> |
| 21 | #include <linux/kernel.h> | 21 | #include <linux/kernel.h> |
| 22 | #include <linux/usb.h> | ||
| 23 | #include <linux/slab.h> | 22 | #include <linux/slab.h> |
| 24 | #include <linux/dma-mapping.h> | 23 | #include <linux/dma-mapping.h> |
| 25 | #include <linux/mmc/host.h> | 24 | #include <linux/mmc/host.h> |
diff --git a/drivers/mtd/ubi/build.c b/drivers/mtd/ubi/build.c index f49e49dc592..5ebe280225d 100644 --- a/drivers/mtd/ubi/build.c +++ b/drivers/mtd/ubi/build.c | |||
| @@ -672,33 +672,7 @@ static int io_init(struct ubi_device *ubi) | |||
| 672 | ubi->nor_flash = 1; | 672 | ubi->nor_flash = 1; |
| 673 | } | 673 | } |
| 674 | 674 | ||
| 675 | /* | 675 | ubi->min_io_size = ubi->mtd->writesize; |
| 676 | * Set UBI min. I/O size (@ubi->min_io_size). We use @mtd->writebufsize | ||
| 677 | * for these purposes, not @mtd->writesize. At the moment this does not | ||
| 678 | * matter for NAND, because currently @mtd->writebufsize is equivalent to | ||
| 679 | * @mtd->writesize for all NANDs. However, some CFI NOR flashes may | ||
| 680 | * have @mtd->writebufsize which is multiple of @mtd->writesize. | ||
| 681 | * | ||
| 682 | * The reason we use @mtd->writebufsize for @ubi->min_io_size is that | ||
| 683 | * UBI and UBIFS recovery algorithms rely on the fact that if there was | ||
| 684 | * an unclean power cut, then we can find offset of the last corrupted | ||
| 685 | * node, align the offset to @ubi->min_io_size, read the rest of the | ||
| 686 | * eraseblock starting from this offset, and check whether there are | ||
| 687 | * only 0xFF bytes. If yes, then we are probably dealing with a | ||
| 688 | * corruption caused by a power cut, if not, then this is probably some | ||
| 689 | * severe corruption. | ||
| 690 | * | ||
| 691 | * Thus, we have to use the maximum write unit size of the flash, which | ||
| 692 | * is @mtd->writebufsize, because @mtd->writesize is the minimum write | ||
| 693 | * size, not the maximum. | ||
| 694 | */ | ||
| 695 | if (ubi->mtd->type == MTD_NANDFLASH) | ||
| 696 | ubi_assert(ubi->mtd->writebufsize == ubi->mtd->writesize); | ||
| 697 | else if (ubi->mtd->type == MTD_NORFLASH) | ||
| 698 | ubi_assert(ubi->mtd->writebufsize % ubi->mtd->writesize == 0); | ||
| 699 | |||
| 700 | ubi->min_io_size = ubi->mtd->writebufsize; | ||
| 701 | |||
| 702 | ubi->hdrs_min_io_size = ubi->mtd->writesize >> ubi->mtd->subpage_sft; | 676 | ubi->hdrs_min_io_size = ubi->mtd->writesize >> ubi->mtd->subpage_sft; |
| 703 | 677 | ||
| 704 | /* | 678 | /* |
diff --git a/drivers/net/arm/ks8695net.c b/drivers/net/arm/ks8695net.c index 62d6f88cbab..aa07657744c 100644 --- a/drivers/net/arm/ks8695net.c +++ b/drivers/net/arm/ks8695net.c | |||
| @@ -1644,7 +1644,7 @@ ks8695_cleanup(void) | |||
| 1644 | module_init(ks8695_init); | 1644 | module_init(ks8695_init); |
| 1645 | module_exit(ks8695_cleanup); | 1645 | module_exit(ks8695_cleanup); |
| 1646 | 1646 | ||
| 1647 | MODULE_AUTHOR("Simtec Electronics") | 1647 | MODULE_AUTHOR("Simtec Electronics"); |
| 1648 | MODULE_DESCRIPTION("Micrel KS8695 (Centaur) Ethernet driver"); | 1648 | MODULE_DESCRIPTION("Micrel KS8695 (Centaur) Ethernet driver"); |
| 1649 | MODULE_LICENSE("GPL"); | 1649 | MODULE_LICENSE("GPL"); |
| 1650 | MODULE_ALIAS("platform:" MODULENAME); | 1650 | MODULE_ALIAS("platform:" MODULENAME); |
diff --git a/drivers/net/atl1c/atl1c_main.c b/drivers/net/atl1c/atl1c_main.c index a699bbf20eb..3824382faec 100644 --- a/drivers/net/atl1c/atl1c_main.c +++ b/drivers/net/atl1c/atl1c_main.c | |||
| @@ -48,6 +48,7 @@ static DEFINE_PCI_DEVICE_TABLE(atl1c_pci_tbl) = { | |||
| 48 | {PCI_DEVICE(PCI_VENDOR_ID_ATTANSIC, PCI_DEVICE_ID_ATHEROS_L2C_B)}, | 48 | {PCI_DEVICE(PCI_VENDOR_ID_ATTANSIC, PCI_DEVICE_ID_ATHEROS_L2C_B)}, |
| 49 | {PCI_DEVICE(PCI_VENDOR_ID_ATTANSIC, PCI_DEVICE_ID_ATHEROS_L2C_B2)}, | 49 | {PCI_DEVICE(PCI_VENDOR_ID_ATTANSIC, PCI_DEVICE_ID_ATHEROS_L2C_B2)}, |
| 50 | {PCI_DEVICE(PCI_VENDOR_ID_ATTANSIC, PCI_DEVICE_ID_ATHEROS_L1D)}, | 50 | {PCI_DEVICE(PCI_VENDOR_ID_ATTANSIC, PCI_DEVICE_ID_ATHEROS_L1D)}, |
| 51 | {PCI_DEVICE(PCI_VENDOR_ID_ATTANSIC, PCI_DEVICE_ID_ATHEROS_L1D_2_0)}, | ||
| 51 | /* required last entry */ | 52 | /* required last entry */ |
| 52 | { 0 } | 53 | { 0 } |
| 53 | }; | 54 | }; |
diff --git a/drivers/net/benet/be_main.c b/drivers/net/benet/be_main.c index de40d3b7152..28a32a6c8bf 100644 --- a/drivers/net/benet/be_main.c +++ b/drivers/net/benet/be_main.c | |||
| @@ -312,11 +312,9 @@ void be_link_status_update(struct be_adapter *adapter, bool link_up) | |||
| 312 | if (adapter->link_up != link_up) { | 312 | if (adapter->link_up != link_up) { |
| 313 | adapter->link_speed = -1; | 313 | adapter->link_speed = -1; |
| 314 | if (link_up) { | 314 | if (link_up) { |
| 315 | netif_start_queue(netdev); | ||
| 316 | netif_carrier_on(netdev); | 315 | netif_carrier_on(netdev); |
| 317 | printk(KERN_INFO "%s: Link up\n", netdev->name); | 316 | printk(KERN_INFO "%s: Link up\n", netdev->name); |
| 318 | } else { | 317 | } else { |
| 319 | netif_stop_queue(netdev); | ||
| 320 | netif_carrier_off(netdev); | 318 | netif_carrier_off(netdev); |
| 321 | printk(KERN_INFO "%s: Link down\n", netdev->name); | 319 | printk(KERN_INFO "%s: Link down\n", netdev->name); |
| 322 | } | 320 | } |
| @@ -2628,8 +2626,6 @@ static void be_netdev_init(struct net_device *netdev) | |||
| 2628 | 2626 | ||
| 2629 | netif_napi_add(netdev, &adapter->tx_eq.napi, be_poll_tx_mcc, | 2627 | netif_napi_add(netdev, &adapter->tx_eq.napi, be_poll_tx_mcc, |
| 2630 | BE_NAPI_WEIGHT); | 2628 | BE_NAPI_WEIGHT); |
| 2631 | |||
| 2632 | netif_stop_queue(netdev); | ||
| 2633 | } | 2629 | } |
| 2634 | 2630 | ||
| 2635 | static void be_unmap_pci_bars(struct be_adapter *adapter) | 2631 | static void be_unmap_pci_bars(struct be_adapter *adapter) |
diff --git a/drivers/net/bnx2.c b/drivers/net/bnx2.c index df99edf3464..0ba59d5aeb7 100644 --- a/drivers/net/bnx2.c +++ b/drivers/net/bnx2.c | |||
| @@ -7553,6 +7553,10 @@ bnx2_set_flags(struct net_device *dev, u32 data) | |||
| 7553 | !(data & ETH_FLAG_RXVLAN)) | 7553 | !(data & ETH_FLAG_RXVLAN)) |
| 7554 | return -EINVAL; | 7554 | return -EINVAL; |
| 7555 | 7555 | ||
| 7556 | /* TSO with VLAN tag won't work with current firmware */ | ||
| 7557 | if (!(data & ETH_FLAG_TXVLAN)) | ||
| 7558 | return -EINVAL; | ||
| 7559 | |||
| 7556 | rc = ethtool_op_set_flags(dev, data, ETH_FLAG_RXHASH | ETH_FLAG_RXVLAN | | 7560 | rc = ethtool_op_set_flags(dev, data, ETH_FLAG_RXHASH | ETH_FLAG_RXVLAN | |
| 7557 | ETH_FLAG_TXVLAN); | 7561 | ETH_FLAG_TXVLAN); |
| 7558 | if (rc) | 7562 | if (rc) |
| @@ -7962,11 +7966,8 @@ bnx2_init_board(struct pci_dev *pdev, struct net_device *dev) | |||
| 7962 | 7966 | ||
| 7963 | /* AER (Advanced Error Reporting) hooks */ | 7967 | /* AER (Advanced Error Reporting) hooks */ |
| 7964 | err = pci_enable_pcie_error_reporting(pdev); | 7968 | err = pci_enable_pcie_error_reporting(pdev); |
| 7965 | if (err) { | 7969 | if (!err) |
| 7966 | dev_err(&pdev->dev, "pci_enable_pcie_error_reporting " | 7970 | bp->flags |= BNX2_FLAG_AER_ENABLED; |
| 7967 | "failed 0x%x\n", err); | ||
| 7968 | /* non-fatal, continue */ | ||
| 7969 | } | ||
| 7970 | 7971 | ||
| 7971 | } else { | 7972 | } else { |
| 7972 | bp->pcix_cap = pci_find_capability(pdev, PCI_CAP_ID_PCIX); | 7973 | bp->pcix_cap = pci_find_capability(pdev, PCI_CAP_ID_PCIX); |
| @@ -8229,8 +8230,10 @@ bnx2_init_board(struct pci_dev *pdev, struct net_device *dev) | |||
| 8229 | return 0; | 8230 | return 0; |
| 8230 | 8231 | ||
| 8231 | err_out_unmap: | 8232 | err_out_unmap: |
| 8232 | if (bp->flags & BNX2_FLAG_PCIE) | 8233 | if (bp->flags & BNX2_FLAG_AER_ENABLED) { |
| 8233 | pci_disable_pcie_error_reporting(pdev); | 8234 | pci_disable_pcie_error_reporting(pdev); |
| 8235 | bp->flags &= ~BNX2_FLAG_AER_ENABLED; | ||
| 8236 | } | ||
| 8234 | 8237 | ||
| 8235 | if (bp->regview) { | 8238 | if (bp->regview) { |
| 8236 | iounmap(bp->regview); | 8239 | iounmap(bp->regview); |
| @@ -8418,8 +8421,10 @@ bnx2_remove_one(struct pci_dev *pdev) | |||
| 8418 | 8421 | ||
| 8419 | kfree(bp->temp_stats_blk); | 8422 | kfree(bp->temp_stats_blk); |
| 8420 | 8423 | ||
| 8421 | if (bp->flags & BNX2_FLAG_PCIE) | 8424 | if (bp->flags & BNX2_FLAG_AER_ENABLED) { |
| 8422 | pci_disable_pcie_error_reporting(pdev); | 8425 | pci_disable_pcie_error_reporting(pdev); |
| 8426 | bp->flags &= ~BNX2_FLAG_AER_ENABLED; | ||
| 8427 | } | ||
| 8423 | 8428 | ||
| 8424 | free_netdev(dev); | 8429 | free_netdev(dev); |
| 8425 | 8430 | ||
| @@ -8535,7 +8540,7 @@ static pci_ers_result_t bnx2_io_slot_reset(struct pci_dev *pdev) | |||
| 8535 | } | 8540 | } |
| 8536 | rtnl_unlock(); | 8541 | rtnl_unlock(); |
| 8537 | 8542 | ||
| 8538 | if (!(bp->flags & BNX2_FLAG_PCIE)) | 8543 | if (!(bp->flags & BNX2_FLAG_AER_ENABLED)) |
| 8539 | return result; | 8544 | return result; |
| 8540 | 8545 | ||
| 8541 | err = pci_cleanup_aer_uncorrect_error_status(pdev); | 8546 | err = pci_cleanup_aer_uncorrect_error_status(pdev); |
diff --git a/drivers/net/bnx2.h b/drivers/net/bnx2.h index 5488a2e82fe..f459fb2f9ad 100644 --- a/drivers/net/bnx2.h +++ b/drivers/net/bnx2.h | |||
| @@ -6741,6 +6741,7 @@ struct bnx2 { | |||
| 6741 | #define BNX2_FLAG_JUMBO_BROKEN 0x00000800 | 6741 | #define BNX2_FLAG_JUMBO_BROKEN 0x00000800 |
| 6742 | #define BNX2_FLAG_CAN_KEEP_VLAN 0x00001000 | 6742 | #define BNX2_FLAG_CAN_KEEP_VLAN 0x00001000 |
| 6743 | #define BNX2_FLAG_BROKEN_STATS 0x00002000 | 6743 | #define BNX2_FLAG_BROKEN_STATS 0x00002000 |
| 6744 | #define BNX2_FLAG_AER_ENABLED 0x00004000 | ||
| 6744 | 6745 | ||
| 6745 | struct bnx2_napi bnx2_napi[BNX2_MAX_MSIX_VEC]; | 6746 | struct bnx2_napi bnx2_napi[BNX2_MAX_MSIX_VEC]; |
| 6746 | 6747 | ||
diff --git a/drivers/net/bnx2x/bnx2x.h b/drivers/net/bnx2x/bnx2x.h index 8e4183717d9..653c62475cb 100644 --- a/drivers/net/bnx2x/bnx2x.h +++ b/drivers/net/bnx2x/bnx2x.h | |||
| @@ -22,8 +22,8 @@ | |||
| 22 | * (you will need to reboot afterwards) */ | 22 | * (you will need to reboot afterwards) */ |
| 23 | /* #define BNX2X_STOP_ON_ERROR */ | 23 | /* #define BNX2X_STOP_ON_ERROR */ |
| 24 | 24 | ||
| 25 | #define DRV_MODULE_VERSION "1.62.00-4" | 25 | #define DRV_MODULE_VERSION "1.62.00-5" |
| 26 | #define DRV_MODULE_RELDATE "2011/01/18" | 26 | #define DRV_MODULE_RELDATE "2011/01/30" |
| 27 | #define BNX2X_BC_VER 0x040200 | 27 | #define BNX2X_BC_VER 0x040200 |
| 28 | 28 | ||
| 29 | #define BNX2X_MULTI_QUEUE | 29 | #define BNX2X_MULTI_QUEUE |
diff --git a/drivers/net/bnx2x/bnx2x_link.c b/drivers/net/bnx2x/bnx2x_link.c index 7160ec51093..dd1210fddff 100644 --- a/drivers/net/bnx2x/bnx2x_link.c +++ b/drivers/net/bnx2x/bnx2x_link.c | |||
| @@ -3948,48 +3948,6 @@ static u8 bnx2x_8073_8727_external_rom_boot(struct bnx2x *bp, | |||
| 3948 | return rc; | 3948 | return rc; |
| 3949 | } | 3949 | } |
| 3950 | 3950 | ||
| 3951 | static void bnx2x_8073_set_xaui_low_power_mode(struct bnx2x *bp, | ||
| 3952 | struct bnx2x_phy *phy) | ||
| 3953 | { | ||
| 3954 | u16 val; | ||
| 3955 | bnx2x_cl45_read(bp, phy, | ||
| 3956 | MDIO_PMA_DEVAD, MDIO_PMA_REG_8073_CHIP_REV, &val); | ||
| 3957 | |||
| 3958 | if (val == 0) { | ||
| 3959 | /* Mustn't set low power mode in 8073 A0 */ | ||
| 3960 | return; | ||
| 3961 | } | ||
| 3962 | |||
| 3963 | /* Disable PLL sequencer (use read-modify-write to clear bit 13) */ | ||
| 3964 | bnx2x_cl45_read(bp, phy, | ||
| 3965 | MDIO_XS_DEVAD, MDIO_XS_PLL_SEQUENCER, &val); | ||
| 3966 | val &= ~(1<<13); | ||
| 3967 | bnx2x_cl45_write(bp, phy, | ||
| 3968 | MDIO_XS_DEVAD, MDIO_XS_PLL_SEQUENCER, val); | ||
| 3969 | |||
| 3970 | /* PLL controls */ | ||
| 3971 | bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x805E, 0x1077); | ||
| 3972 | bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x805D, 0x0000); | ||
| 3973 | bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x805C, 0x030B); | ||
| 3974 | bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x805B, 0x1240); | ||
| 3975 | bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x805A, 0x2490); | ||
| 3976 | |||
| 3977 | /* Tx Controls */ | ||
| 3978 | bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x80A7, 0x0C74); | ||
| 3979 | bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x80A6, 0x9041); | ||
| 3980 | bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x80A5, 0x4640); | ||
| 3981 | |||
| 3982 | /* Rx Controls */ | ||
| 3983 | bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x80FE, 0x01C4); | ||
| 3984 | bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x80FD, 0x9249); | ||
| 3985 | bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x80FC, 0x2015); | ||
| 3986 | |||
| 3987 | /* Enable PLL sequencer (use read-modify-write to set bit 13) */ | ||
| 3988 | bnx2x_cl45_read(bp, phy, MDIO_XS_DEVAD, MDIO_XS_PLL_SEQUENCER, &val); | ||
| 3989 | val |= (1<<13); | ||
| 3990 | bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, MDIO_XS_PLL_SEQUENCER, val); | ||
| 3991 | } | ||
| 3992 | |||
| 3993 | /******************************************************************/ | 3951 | /******************************************************************/ |
| 3994 | /* BCM8073 PHY SECTION */ | 3952 | /* BCM8073 PHY SECTION */ |
| 3995 | /******************************************************************/ | 3953 | /******************************************************************/ |
| @@ -4148,8 +4106,6 @@ static u8 bnx2x_8073_config_init(struct bnx2x_phy *phy, | |||
| 4148 | 4106 | ||
| 4149 | bnx2x_8073_set_pause_cl37(params, phy, vars); | 4107 | bnx2x_8073_set_pause_cl37(params, phy, vars); |
| 4150 | 4108 | ||
| 4151 | bnx2x_8073_set_xaui_low_power_mode(bp, phy); | ||
| 4152 | |||
| 4153 | bnx2x_cl45_read(bp, phy, | 4109 | bnx2x_cl45_read(bp, phy, |
| 4154 | MDIO_PMA_DEVAD, MDIO_PMA_REG_M8051_MSGOUT_REG, &tmp1); | 4110 | MDIO_PMA_DEVAD, MDIO_PMA_REG_M8051_MSGOUT_REG, &tmp1); |
| 4155 | 4111 | ||
| @@ -6519,6 +6475,18 @@ static void bnx2x_848xx_set_link_led(struct bnx2x_phy *phy, | |||
| 6519 | MDIO_PMA_DEVAD, | 6475 | MDIO_PMA_DEVAD, |
| 6520 | MDIO_PMA_REG_8481_LED1_MASK, | 6476 | MDIO_PMA_REG_8481_LED1_MASK, |
| 6521 | 0x80); | 6477 | 0x80); |
| 6478 | |||
| 6479 | /* Tell LED3 to blink on source */ | ||
| 6480 | bnx2x_cl45_read(bp, phy, | ||
| 6481 | MDIO_PMA_DEVAD, | ||
| 6482 | MDIO_PMA_REG_8481_LINK_SIGNAL, | ||
| 6483 | &val); | ||
| 6484 | val &= ~(7<<6); | ||
| 6485 | val |= (1<<6); /* A83B[8:6]= 1 */ | ||
| 6486 | bnx2x_cl45_write(bp, phy, | ||
| 6487 | MDIO_PMA_DEVAD, | ||
| 6488 | MDIO_PMA_REG_8481_LINK_SIGNAL, | ||
| 6489 | val); | ||
| 6522 | } | 6490 | } |
| 6523 | break; | 6491 | break; |
| 6524 | } | 6492 | } |
| @@ -7720,10 +7688,13 @@ static u8 bnx2x_8073_common_init_phy(struct bnx2x *bp, | |||
| 7720 | struct bnx2x_phy phy[PORT_MAX]; | 7688 | struct bnx2x_phy phy[PORT_MAX]; |
| 7721 | struct bnx2x_phy *phy_blk[PORT_MAX]; | 7689 | struct bnx2x_phy *phy_blk[PORT_MAX]; |
| 7722 | u16 val; | 7690 | u16 val; |
| 7723 | s8 port; | 7691 | s8 port = 0; |
| 7724 | s8 port_of_path = 0; | 7692 | s8 port_of_path = 0; |
| 7725 | 7693 | u32 swap_val, swap_override; | |
| 7726 | bnx2x_ext_phy_hw_reset(bp, 0); | 7694 | swap_val = REG_RD(bp, NIG_REG_PORT_SWAP); |
| 7695 | swap_override = REG_RD(bp, NIG_REG_STRAP_OVERRIDE); | ||
| 7696 | port ^= (swap_val && swap_override); | ||
| 7697 | bnx2x_ext_phy_hw_reset(bp, port); | ||
| 7727 | /* PART1 - Reset both phys */ | 7698 | /* PART1 - Reset both phys */ |
| 7728 | for (port = PORT_MAX - 1; port >= PORT_0; port--) { | 7699 | for (port = PORT_MAX - 1; port >= PORT_0; port--) { |
| 7729 | u32 shmem_base, shmem2_base; | 7700 | u32 shmem_base, shmem2_base; |
diff --git a/drivers/net/bnx2x/bnx2x_main.c b/drivers/net/bnx2x/bnx2x_main.c index 8cdcf5b39d1..f40740e68ea 100644 --- a/drivers/net/bnx2x/bnx2x_main.c +++ b/drivers/net/bnx2x/bnx2x_main.c | |||
| @@ -2301,15 +2301,10 @@ static void bnx2x_rxq_set_mac_filters(struct bnx2x *bp, u16 cl_id, u32 filters) | |||
| 2301 | /* accept matched ucast */ | 2301 | /* accept matched ucast */ |
| 2302 | drop_all_ucast = 0; | 2302 | drop_all_ucast = 0; |
| 2303 | } | 2303 | } |
| 2304 | if (filters & BNX2X_ACCEPT_MULTICAST) { | 2304 | if (filters & BNX2X_ACCEPT_MULTICAST) |
| 2305 | /* accept matched mcast */ | 2305 | /* accept matched mcast */ |
| 2306 | drop_all_mcast = 0; | 2306 | drop_all_mcast = 0; |
| 2307 | if (IS_MF_SI(bp)) | 2307 | |
| 2308 | /* since mcast addresses won't arrive with ovlan, | ||
| 2309 | * fw needs to accept all of them in | ||
| 2310 | * switch-independent mode */ | ||
| 2311 | accp_all_mcast = 1; | ||
| 2312 | } | ||
| 2313 | if (filters & BNX2X_ACCEPT_ALL_UNICAST) { | 2308 | if (filters & BNX2X_ACCEPT_ALL_UNICAST) { |
| 2314 | /* accept all mcast */ | 2309 | /* accept all mcast */ |
| 2315 | drop_all_ucast = 0; | 2310 | drop_all_ucast = 0; |
| @@ -5296,10 +5291,6 @@ static int bnx2x_init_hw_common(struct bnx2x *bp, u32 load_code) | |||
| 5296 | } | 5291 | } |
| 5297 | } | 5292 | } |
| 5298 | 5293 | ||
| 5299 | bp->port.need_hw_lock = bnx2x_hw_lock_required(bp, | ||
| 5300 | bp->common.shmem_base, | ||
| 5301 | bp->common.shmem2_base); | ||
| 5302 | |||
| 5303 | bnx2x_setup_fan_failure_detection(bp); | 5294 | bnx2x_setup_fan_failure_detection(bp); |
| 5304 | 5295 | ||
| 5305 | /* clear PXP2 attentions */ | 5296 | /* clear PXP2 attentions */ |
| @@ -5503,9 +5494,6 @@ static int bnx2x_init_hw_port(struct bnx2x *bp) | |||
| 5503 | 5494 | ||
| 5504 | bnx2x_init_block(bp, MCP_BLOCK, init_stage); | 5495 | bnx2x_init_block(bp, MCP_BLOCK, init_stage); |
| 5505 | bnx2x_init_block(bp, DMAE_BLOCK, init_stage); | 5496 | bnx2x_init_block(bp, DMAE_BLOCK, init_stage); |
| 5506 | bp->port.need_hw_lock = bnx2x_hw_lock_required(bp, | ||
| 5507 | bp->common.shmem_base, | ||
| 5508 | bp->common.shmem2_base); | ||
| 5509 | if (bnx2x_fan_failure_det_req(bp, bp->common.shmem_base, | 5497 | if (bnx2x_fan_failure_det_req(bp, bp->common.shmem_base, |
| 5510 | bp->common.shmem2_base, port)) { | 5498 | bp->common.shmem2_base, port)) { |
| 5511 | u32 reg_addr = (port ? MISC_REG_AEU_ENABLE1_FUNC_1_OUT_0 : | 5499 | u32 reg_addr = (port ? MISC_REG_AEU_ENABLE1_FUNC_1_OUT_0 : |
| @@ -8379,6 +8367,17 @@ static void __devinit bnx2x_get_port_hwinfo(struct bnx2x *bp) | |||
| 8379 | (ext_phy_type != PORT_HW_CFG_XGXS_EXT_PHY_TYPE_NOT_CONN)) | 8367 | (ext_phy_type != PORT_HW_CFG_XGXS_EXT_PHY_TYPE_NOT_CONN)) |
| 8380 | bp->mdio.prtad = | 8368 | bp->mdio.prtad = |
| 8381 | XGXS_EXT_PHY_ADDR(ext_phy_config); | 8369 | XGXS_EXT_PHY_ADDR(ext_phy_config); |
| 8370 | |||
| 8371 | /* | ||
| 8372 | * Check if hw lock is required to access MDC/MDIO bus to the PHY(s) | ||
| 8373 | * In MF mode, it is set to cover self test cases | ||
| 8374 | */ | ||
| 8375 | if (IS_MF(bp)) | ||
| 8376 | bp->port.need_hw_lock = 1; | ||
| 8377 | else | ||
| 8378 | bp->port.need_hw_lock = bnx2x_hw_lock_required(bp, | ||
| 8379 | bp->common.shmem_base, | ||
| 8380 | bp->common.shmem2_base); | ||
| 8382 | } | 8381 | } |
| 8383 | 8382 | ||
| 8384 | static void __devinit bnx2x_get_mac_hwinfo(struct bnx2x *bp) | 8383 | static void __devinit bnx2x_get_mac_hwinfo(struct bnx2x *bp) |
diff --git a/drivers/net/bonding/bond_3ad.c b/drivers/net/bonding/bond_3ad.c index 171782e2bb3..1024ae15822 100644 --- a/drivers/net/bonding/bond_3ad.c +++ b/drivers/net/bonding/bond_3ad.c | |||
| @@ -2470,6 +2470,10 @@ int bond_3ad_lacpdu_recv(struct sk_buff *skb, struct net_device *dev, struct pac | |||
| 2470 | if (!(dev->flags & IFF_MASTER)) | 2470 | if (!(dev->flags & IFF_MASTER)) |
| 2471 | goto out; | 2471 | goto out; |
| 2472 | 2472 | ||
| 2473 | skb = skb_share_check(skb, GFP_ATOMIC); | ||
| 2474 | if (!skb) | ||
| 2475 | goto out; | ||
| 2476 | |||
| 2473 | if (!pskb_may_pull(skb, sizeof(struct lacpdu))) | 2477 | if (!pskb_may_pull(skb, sizeof(struct lacpdu))) |
| 2474 | goto out; | 2478 | goto out; |
| 2475 | 2479 | ||
diff --git a/drivers/net/bonding/bond_alb.c b/drivers/net/bonding/bond_alb.c index f4e638c6512..5c6fba802f2 100644 --- a/drivers/net/bonding/bond_alb.c +++ b/drivers/net/bonding/bond_alb.c | |||
| @@ -326,6 +326,10 @@ static int rlb_arp_recv(struct sk_buff *skb, struct net_device *bond_dev, struct | |||
| 326 | goto out; | 326 | goto out; |
| 327 | } | 327 | } |
| 328 | 328 | ||
| 329 | skb = skb_share_check(skb, GFP_ATOMIC); | ||
| 330 | if (!skb) | ||
| 331 | goto out; | ||
| 332 | |||
| 329 | if (!pskb_may_pull(skb, arp_hdr_len(bond_dev))) | 333 | if (!pskb_may_pull(skb, arp_hdr_len(bond_dev))) |
| 330 | goto out; | 334 | goto out; |
| 331 | 335 | ||
diff --git a/drivers/net/bonding/bond_main.c b/drivers/net/bonding/bond_main.c index b1025b85acf..163e0b06eaa 100644 --- a/drivers/net/bonding/bond_main.c +++ b/drivers/net/bonding/bond_main.c | |||
| @@ -2733,6 +2733,10 @@ static int bond_arp_rcv(struct sk_buff *skb, struct net_device *dev, struct pack | |||
| 2733 | if (!slave || !slave_do_arp_validate(bond, slave)) | 2733 | if (!slave || !slave_do_arp_validate(bond, slave)) |
| 2734 | goto out_unlock; | 2734 | goto out_unlock; |
| 2735 | 2735 | ||
| 2736 | skb = skb_share_check(skb, GFP_ATOMIC); | ||
| 2737 | if (!skb) | ||
| 2738 | goto out_unlock; | ||
| 2739 | |||
| 2736 | if (!pskb_may_pull(skb, arp_hdr_len(dev))) | 2740 | if (!pskb_may_pull(skb, arp_hdr_len(dev))) |
| 2737 | goto out_unlock; | 2741 | goto out_unlock; |
| 2738 | 2742 | ||
diff --git a/drivers/net/can/Kconfig b/drivers/net/can/Kconfig index d5a9db60ade..5dec456fd4a 100644 --- a/drivers/net/can/Kconfig +++ b/drivers/net/can/Kconfig | |||
| @@ -23,7 +23,7 @@ config CAN_SLCAN | |||
| 23 | 23 | ||
| 24 | As only the sending and receiving of CAN frames is implemented, this | 24 | As only the sending and receiving of CAN frames is implemented, this |
| 25 | driver should work with the (serial/USB) CAN hardware from: | 25 | driver should work with the (serial/USB) CAN hardware from: |
| 26 | www.canusb.com / www.can232.com / www.mictronic.com / www.canhack.de | 26 | www.canusb.com / www.can232.com / www.mictronics.de / www.canhack.de |
| 27 | 27 | ||
| 28 | Userspace tools to attach the SLCAN line discipline (slcan_attach, | 28 | Userspace tools to attach the SLCAN line discipline (slcan_attach, |
| 29 | slcand) can be found in the can-utils at the SocketCAN SVN, see | 29 | slcand) can be found in the can-utils at the SocketCAN SVN, see |
| @@ -117,6 +117,8 @@ source "drivers/net/can/sja1000/Kconfig" | |||
| 117 | 117 | ||
| 118 | source "drivers/net/can/usb/Kconfig" | 118 | source "drivers/net/can/usb/Kconfig" |
| 119 | 119 | ||
| 120 | source "drivers/net/can/softing/Kconfig" | ||
| 121 | |||
| 120 | config CAN_DEBUG_DEVICES | 122 | config CAN_DEBUG_DEVICES |
| 121 | bool "CAN devices debugging messages" | 123 | bool "CAN devices debugging messages" |
| 122 | depends on CAN | 124 | depends on CAN |
diff --git a/drivers/net/can/Makefile b/drivers/net/can/Makefile index 07ca159ba3f..53c82a71778 100644 --- a/drivers/net/can/Makefile +++ b/drivers/net/can/Makefile | |||
| @@ -9,6 +9,7 @@ obj-$(CONFIG_CAN_DEV) += can-dev.o | |||
| 9 | can-dev-y := dev.o | 9 | can-dev-y := dev.o |
| 10 | 10 | ||
| 11 | obj-y += usb/ | 11 | obj-y += usb/ |
| 12 | obj-y += softing/ | ||
| 12 | 13 | ||
| 13 | obj-$(CONFIG_CAN_SJA1000) += sja1000/ | 14 | obj-$(CONFIG_CAN_SJA1000) += sja1000/ |
| 14 | obj-$(CONFIG_CAN_MSCAN) += mscan/ | 15 | obj-$(CONFIG_CAN_MSCAN) += mscan/ |
diff --git a/drivers/net/can/at91_can.c b/drivers/net/can/at91_can.c index 7ef83d06f7e..57d2ffbbb43 100644 --- a/drivers/net/can/at91_can.c +++ b/drivers/net/can/at91_can.c | |||
| @@ -2,7 +2,7 @@ | |||
| 2 | * at91_can.c - CAN network driver for AT91 SoC CAN controller | 2 | * at91_can.c - CAN network driver for AT91 SoC CAN controller |
| 3 | * | 3 | * |
| 4 | * (C) 2007 by Hans J. Koch <hjk@hansjkoch.de> | 4 | * (C) 2007 by Hans J. Koch <hjk@hansjkoch.de> |
| 5 | * (C) 2008, 2009, 2010 by Marc Kleine-Budde <kernel@pengutronix.de> | 5 | * (C) 2008, 2009, 2010, 2011 by Marc Kleine-Budde <kernel@pengutronix.de> |
| 6 | * | 6 | * |
| 7 | * This software may be distributed under the terms of the GNU General | 7 | * This software may be distributed under the terms of the GNU General |
| 8 | * Public License ("GPL") version 2 as distributed in the 'COPYING' | 8 | * Public License ("GPL") version 2 as distributed in the 'COPYING' |
| @@ -30,6 +30,7 @@ | |||
| 30 | #include <linux/module.h> | 30 | #include <linux/module.h> |
| 31 | #include <linux/netdevice.h> | 31 | #include <linux/netdevice.h> |
| 32 | #include <linux/platform_device.h> | 32 | #include <linux/platform_device.h> |
| 33 | #include <linux/rtnetlink.h> | ||
| 33 | #include <linux/skbuff.h> | 34 | #include <linux/skbuff.h> |
| 34 | #include <linux/spinlock.h> | 35 | #include <linux/spinlock.h> |
| 35 | #include <linux/string.h> | 36 | #include <linux/string.h> |
| @@ -40,22 +41,23 @@ | |||
| 40 | 41 | ||
| 41 | #include <mach/board.h> | 42 | #include <mach/board.h> |
| 42 | 43 | ||
| 43 | #define AT91_NAPI_WEIGHT 12 | 44 | #define AT91_NAPI_WEIGHT 11 |
| 44 | 45 | ||
| 45 | /* | 46 | /* |
| 46 | * RX/TX Mailbox split | 47 | * RX/TX Mailbox split |
| 47 | * don't dare to touch | 48 | * don't dare to touch |
| 48 | */ | 49 | */ |
| 49 | #define AT91_MB_RX_NUM 12 | 50 | #define AT91_MB_RX_NUM 11 |
| 50 | #define AT91_MB_TX_SHIFT 2 | 51 | #define AT91_MB_TX_SHIFT 2 |
| 51 | 52 | ||
| 52 | #define AT91_MB_RX_FIRST 0 | 53 | #define AT91_MB_RX_FIRST 1 |
| 53 | #define AT91_MB_RX_LAST (AT91_MB_RX_FIRST + AT91_MB_RX_NUM - 1) | 54 | #define AT91_MB_RX_LAST (AT91_MB_RX_FIRST + AT91_MB_RX_NUM - 1) |
| 54 | 55 | ||
| 55 | #define AT91_MB_RX_MASK(i) ((1 << (i)) - 1) | 56 | #define AT91_MB_RX_MASK(i) ((1 << (i)) - 1) |
| 56 | #define AT91_MB_RX_SPLIT 8 | 57 | #define AT91_MB_RX_SPLIT 8 |
| 57 | #define AT91_MB_RX_LOW_LAST (AT91_MB_RX_SPLIT - 1) | 58 | #define AT91_MB_RX_LOW_LAST (AT91_MB_RX_SPLIT - 1) |
| 58 | #define AT91_MB_RX_LOW_MASK (AT91_MB_RX_MASK(AT91_MB_RX_SPLIT)) | 59 | #define AT91_MB_RX_LOW_MASK (AT91_MB_RX_MASK(AT91_MB_RX_SPLIT) & \ |
| 60 | ~AT91_MB_RX_MASK(AT91_MB_RX_FIRST)) | ||
| 59 | 61 | ||
| 60 | #define AT91_MB_TX_NUM (1 << AT91_MB_TX_SHIFT) | 62 | #define AT91_MB_TX_NUM (1 << AT91_MB_TX_SHIFT) |
| 61 | #define AT91_MB_TX_FIRST (AT91_MB_RX_LAST + 1) | 63 | #define AT91_MB_TX_FIRST (AT91_MB_RX_LAST + 1) |
| @@ -168,6 +170,8 @@ struct at91_priv { | |||
| 168 | 170 | ||
| 169 | struct clk *clk; | 171 | struct clk *clk; |
| 170 | struct at91_can_data *pdata; | 172 | struct at91_can_data *pdata; |
| 173 | |||
| 174 | canid_t mb0_id; | ||
| 171 | }; | 175 | }; |
| 172 | 176 | ||
| 173 | static struct can_bittiming_const at91_bittiming_const = { | 177 | static struct can_bittiming_const at91_bittiming_const = { |
| @@ -220,6 +224,18 @@ static inline void set_mb_mode(const struct at91_priv *priv, unsigned int mb, | |||
| 220 | set_mb_mode_prio(priv, mb, mode, 0); | 224 | set_mb_mode_prio(priv, mb, mode, 0); |
| 221 | } | 225 | } |
| 222 | 226 | ||
| 227 | static inline u32 at91_can_id_to_reg_mid(canid_t can_id) | ||
| 228 | { | ||
| 229 | u32 reg_mid; | ||
| 230 | |||
| 231 | if (can_id & CAN_EFF_FLAG) | ||
| 232 | reg_mid = (can_id & CAN_EFF_MASK) | AT91_MID_MIDE; | ||
| 233 | else | ||
| 234 | reg_mid = (can_id & CAN_SFF_MASK) << 18; | ||
| 235 | |||
| 236 | return reg_mid; | ||
| 237 | } | ||
| 238 | |||
| 223 | /* | 239 | /* |
| 224 | * Swtich transceiver on or off | 240 | * Swtich transceiver on or off |
| 225 | */ | 241 | */ |
| @@ -233,12 +249,22 @@ static void at91_setup_mailboxes(struct net_device *dev) | |||
| 233 | { | 249 | { |
| 234 | struct at91_priv *priv = netdev_priv(dev); | 250 | struct at91_priv *priv = netdev_priv(dev); |
| 235 | unsigned int i; | 251 | unsigned int i; |
| 252 | u32 reg_mid; | ||
| 236 | 253 | ||
| 237 | /* | 254 | /* |
| 238 | * The first 12 mailboxes are used as a reception FIFO. The | 255 | * Due to a chip bug (errata 50.2.6.3 & 50.3.5.3) the first |
| 239 | * last mailbox is configured with overwrite option. The | 256 | * mailbox is disabled. The next 11 mailboxes are used as a |
| 240 | * overwrite flag indicates a FIFO overflow. | 257 | * reception FIFO. The last mailbox is configured with |
| 258 | * overwrite option. The overwrite flag indicates a FIFO | ||
| 259 | * overflow. | ||
| 241 | */ | 260 | */ |
| 261 | reg_mid = at91_can_id_to_reg_mid(priv->mb0_id); | ||
| 262 | for (i = 0; i < AT91_MB_RX_FIRST; i++) { | ||
| 263 | set_mb_mode(priv, i, AT91_MB_MODE_DISABLED); | ||
| 264 | at91_write(priv, AT91_MID(i), reg_mid); | ||
| 265 | at91_write(priv, AT91_MCR(i), 0x0); /* clear dlc */ | ||
| 266 | } | ||
| 267 | |||
| 242 | for (i = AT91_MB_RX_FIRST; i < AT91_MB_RX_LAST; i++) | 268 | for (i = AT91_MB_RX_FIRST; i < AT91_MB_RX_LAST; i++) |
| 243 | set_mb_mode(priv, i, AT91_MB_MODE_RX); | 269 | set_mb_mode(priv, i, AT91_MB_MODE_RX); |
| 244 | set_mb_mode(priv, AT91_MB_RX_LAST, AT91_MB_MODE_RX_OVRWR); | 270 | set_mb_mode(priv, AT91_MB_RX_LAST, AT91_MB_MODE_RX_OVRWR); |
| @@ -254,7 +280,8 @@ static void at91_setup_mailboxes(struct net_device *dev) | |||
| 254 | set_mb_mode_prio(priv, i, AT91_MB_MODE_TX, 0); | 280 | set_mb_mode_prio(priv, i, AT91_MB_MODE_TX, 0); |
| 255 | 281 | ||
| 256 | /* Reset tx and rx helper pointers */ | 282 | /* Reset tx and rx helper pointers */ |
| 257 | priv->tx_next = priv->tx_echo = priv->rx_next = 0; | 283 | priv->tx_next = priv->tx_echo = 0; |
| 284 | priv->rx_next = AT91_MB_RX_FIRST; | ||
| 258 | } | 285 | } |
| 259 | 286 | ||
| 260 | static int at91_set_bittiming(struct net_device *dev) | 287 | static int at91_set_bittiming(struct net_device *dev) |
| @@ -372,12 +399,7 @@ static netdev_tx_t at91_start_xmit(struct sk_buff *skb, struct net_device *dev) | |||
| 372 | netdev_err(dev, "BUG! TX buffer full when queue awake!\n"); | 399 | netdev_err(dev, "BUG! TX buffer full when queue awake!\n"); |
| 373 | return NETDEV_TX_BUSY; | 400 | return NETDEV_TX_BUSY; |
| 374 | } | 401 | } |
| 375 | 402 | reg_mid = at91_can_id_to_reg_mid(cf->can_id); | |
| 376 | if (cf->can_id & CAN_EFF_FLAG) | ||
| 377 | reg_mid = (cf->can_id & CAN_EFF_MASK) | AT91_MID_MIDE; | ||
| 378 | else | ||
| 379 | reg_mid = (cf->can_id & CAN_SFF_MASK) << 18; | ||
| 380 | |||
| 381 | reg_mcr = ((cf->can_id & CAN_RTR_FLAG) ? AT91_MCR_MRTR : 0) | | 403 | reg_mcr = ((cf->can_id & CAN_RTR_FLAG) ? AT91_MCR_MRTR : 0) | |
| 382 | (cf->can_dlc << 16) | AT91_MCR_MTCR; | 404 | (cf->can_dlc << 16) | AT91_MCR_MTCR; |
| 383 | 405 | ||
| @@ -539,27 +561,31 @@ static void at91_read_msg(struct net_device *dev, unsigned int mb) | |||
| 539 | * | 561 | * |
| 540 | * Theory of Operation: | 562 | * Theory of Operation: |
| 541 | * | 563 | * |
| 542 | * 12 of the 16 mailboxes on the chip are reserved for RX. we split | 564 | * 11 of the 16 mailboxes on the chip are reserved for RX. we split |
| 543 | * them into 2 groups. The lower group holds 8 and upper 4 mailboxes. | 565 | * them into 2 groups. The lower group holds 7 and upper 4 mailboxes. |
| 544 | * | 566 | * |
| 545 | * Like it or not, but the chip always saves a received CAN message | 567 | * Like it or not, but the chip always saves a received CAN message |
| 546 | * into the first free mailbox it finds (starting with the | 568 | * into the first free mailbox it finds (starting with the |
| 547 | * lowest). This makes it very difficult to read the messages in the | 569 | * lowest). This makes it very difficult to read the messages in the |
| 548 | * right order from the chip. This is how we work around that problem: | 570 | * right order from the chip. This is how we work around that problem: |
| 549 | * | 571 | * |
| 550 | * The first message goes into mb nr. 0 and issues an interrupt. All | 572 | * The first message goes into mb nr. 1 and issues an interrupt. All |
| 551 | * rx ints are disabled in the interrupt handler and a napi poll is | 573 | * rx ints are disabled in the interrupt handler and a napi poll is |
| 552 | * scheduled. We read the mailbox, but do _not_ reenable the mb (to | 574 | * scheduled. We read the mailbox, but do _not_ reenable the mb (to |
| 553 | * receive another message). | 575 | * receive another message). |
| 554 | * | 576 | * |
| 555 | * lower mbxs upper | 577 | * lower mbxs upper |
| 556 | * ______^______ __^__ | 578 | * ____^______ __^__ |
| 557 | * / \ / \ | 579 | * / \ / \ |
| 558 | * +-+-+-+-+-+-+-+-++-+-+-+-+ | 580 | * +-+-+-+-+-+-+-+-++-+-+-+-+ |
| 559 | * |x|x|x|x|x|x|x|x|| | | | | | 581 | * | |x|x|x|x|x|x|x|| | | | | |
| 560 | * +-+-+-+-+-+-+-+-++-+-+-+-+ | 582 | * +-+-+-+-+-+-+-+-++-+-+-+-+ |
| 561 | * 0 0 0 0 0 0 0 0 0 0 1 1 \ mail | 583 | * 0 0 0 0 0 0 0 0 0 0 1 1 \ mail |
| 562 | * 0 1 2 3 4 5 6 7 8 9 0 1 / box | 584 | * 0 1 2 3 4 5 6 7 8 9 0 1 / box |
| 585 | * ^ | ||
| 586 | * | | ||
| 587 | * \ | ||
| 588 | * unused, due to chip bug | ||
| 563 | * | 589 | * |
| 564 | * The variable priv->rx_next points to the next mailbox to read a | 590 | * The variable priv->rx_next points to the next mailbox to read a |
| 565 | * message from. As long we're in the lower mailboxes we just read the | 591 | * message from. As long we're in the lower mailboxes we just read the |
| @@ -590,10 +616,10 @@ static int at91_poll_rx(struct net_device *dev, int quota) | |||
| 590 | "order of incoming frames cannot be guaranteed\n"); | 616 | "order of incoming frames cannot be guaranteed\n"); |
| 591 | 617 | ||
| 592 | again: | 618 | again: |
| 593 | for (mb = find_next_bit(addr, AT91_MB_RX_NUM, priv->rx_next); | 619 | for (mb = find_next_bit(addr, AT91_MB_RX_LAST + 1, priv->rx_next); |
| 594 | mb < AT91_MB_RX_NUM && quota > 0; | 620 | mb < AT91_MB_RX_LAST + 1 && quota > 0; |
| 595 | reg_sr = at91_read(priv, AT91_SR), | 621 | reg_sr = at91_read(priv, AT91_SR), |
| 596 | mb = find_next_bit(addr, AT91_MB_RX_NUM, ++priv->rx_next)) { | 622 | mb = find_next_bit(addr, AT91_MB_RX_LAST + 1, ++priv->rx_next)) { |
| 597 | at91_read_msg(dev, mb); | 623 | at91_read_msg(dev, mb); |
| 598 | 624 | ||
| 599 | /* reactivate mailboxes */ | 625 | /* reactivate mailboxes */ |
| @@ -610,8 +636,8 @@ static int at91_poll_rx(struct net_device *dev, int quota) | |||
| 610 | 636 | ||
| 611 | /* upper group completed, look again in lower */ | 637 | /* upper group completed, look again in lower */ |
| 612 | if (priv->rx_next > AT91_MB_RX_LOW_LAST && | 638 | if (priv->rx_next > AT91_MB_RX_LOW_LAST && |
| 613 | quota > 0 && mb >= AT91_MB_RX_NUM) { | 639 | quota > 0 && mb > AT91_MB_RX_LAST) { |
| 614 | priv->rx_next = 0; | 640 | priv->rx_next = AT91_MB_RX_FIRST; |
| 615 | goto again; | 641 | goto again; |
| 616 | } | 642 | } |
| 617 | 643 | ||
| @@ -1037,6 +1063,64 @@ static const struct net_device_ops at91_netdev_ops = { | |||
| 1037 | .ndo_start_xmit = at91_start_xmit, | 1063 | .ndo_start_xmit = at91_start_xmit, |
| 1038 | }; | 1064 | }; |
| 1039 | 1065 | ||
| 1066 | static ssize_t at91_sysfs_show_mb0_id(struct device *dev, | ||
| 1067 | struct device_attribute *attr, char *buf) | ||
| 1068 | { | ||
| 1069 | struct at91_priv *priv = netdev_priv(to_net_dev(dev)); | ||
| 1070 | |||
| 1071 | if (priv->mb0_id & CAN_EFF_FLAG) | ||
| 1072 | return snprintf(buf, PAGE_SIZE, "0x%08x\n", priv->mb0_id); | ||
| 1073 | else | ||
| 1074 | return snprintf(buf, PAGE_SIZE, "0x%03x\n", priv->mb0_id); | ||
| 1075 | } | ||
| 1076 | |||
| 1077 | static ssize_t at91_sysfs_set_mb0_id(struct device *dev, | ||
| 1078 | struct device_attribute *attr, const char *buf, size_t count) | ||
| 1079 | { | ||
| 1080 | struct net_device *ndev = to_net_dev(dev); | ||
| 1081 | struct at91_priv *priv = netdev_priv(ndev); | ||
| 1082 | unsigned long can_id; | ||
| 1083 | ssize_t ret; | ||
| 1084 | int err; | ||
| 1085 | |||
| 1086 | rtnl_lock(); | ||
| 1087 | |||
| 1088 | if (ndev->flags & IFF_UP) { | ||
| 1089 | ret = -EBUSY; | ||
| 1090 | goto out; | ||
| 1091 | } | ||
| 1092 | |||
| 1093 | err = strict_strtoul(buf, 0, &can_id); | ||
| 1094 | if (err) { | ||
| 1095 | ret = err; | ||
| 1096 | goto out; | ||
| 1097 | } | ||
| 1098 | |||
| 1099 | if (can_id & CAN_EFF_FLAG) | ||
| 1100 | can_id &= CAN_EFF_MASK | CAN_EFF_FLAG; | ||
| 1101 | else | ||
| 1102 | can_id &= CAN_SFF_MASK; | ||
| 1103 | |||
| 1104 | priv->mb0_id = can_id; | ||
| 1105 | ret = count; | ||
| 1106 | |||
| 1107 | out: | ||
| 1108 | rtnl_unlock(); | ||
| 1109 | return ret; | ||
| 1110 | } | ||
| 1111 | |||
| 1112 | static DEVICE_ATTR(mb0_id, S_IWUSR | S_IRUGO, | ||
| 1113 | at91_sysfs_show_mb0_id, at91_sysfs_set_mb0_id); | ||
| 1114 | |||
| 1115 | static struct attribute *at91_sysfs_attrs[] = { | ||
| 1116 | &dev_attr_mb0_id.attr, | ||
| 1117 | NULL, | ||
| 1118 | }; | ||
| 1119 | |||
| 1120 | static struct attribute_group at91_sysfs_attr_group = { | ||
| 1121 | .attrs = at91_sysfs_attrs, | ||
| 1122 | }; | ||
| 1123 | |||
| 1040 | static int __devinit at91_can_probe(struct platform_device *pdev) | 1124 | static int __devinit at91_can_probe(struct platform_device *pdev) |
| 1041 | { | 1125 | { |
| 1042 | struct net_device *dev; | 1126 | struct net_device *dev; |
| @@ -1082,6 +1166,7 @@ static int __devinit at91_can_probe(struct platform_device *pdev) | |||
| 1082 | dev->netdev_ops = &at91_netdev_ops; | 1166 | dev->netdev_ops = &at91_netdev_ops; |
| 1083 | dev->irq = irq; | 1167 | dev->irq = irq; |
| 1084 | dev->flags |= IFF_ECHO; | 1168 | dev->flags |= IFF_ECHO; |
| 1169 | dev->sysfs_groups[0] = &at91_sysfs_attr_group; | ||
| 1085 | 1170 | ||
| 1086 | priv = netdev_priv(dev); | 1171 | priv = netdev_priv(dev); |
| 1087 | priv->can.clock.freq = clk_get_rate(clk); | 1172 | priv->can.clock.freq = clk_get_rate(clk); |
| @@ -1093,6 +1178,7 @@ static int __devinit at91_can_probe(struct platform_device *pdev) | |||
| 1093 | priv->dev = dev; | 1178 | priv->dev = dev; |
| 1094 | priv->clk = clk; | 1179 | priv->clk = clk; |
| 1095 | priv->pdata = pdev->dev.platform_data; | 1180 | priv->pdata = pdev->dev.platform_data; |
| 1181 | priv->mb0_id = 0x7ff; | ||
| 1096 | 1182 | ||
| 1097 | netif_napi_add(dev, &priv->napi, at91_poll, AT91_NAPI_WEIGHT); | 1183 | netif_napi_add(dev, &priv->napi, at91_poll, AT91_NAPI_WEIGHT); |
| 1098 | 1184 | ||
diff --git a/drivers/net/can/janz-ican3.c b/drivers/net/can/janz-ican3.c index b9a6d7a5a73..366f5cc050a 100644 --- a/drivers/net/can/janz-ican3.c +++ b/drivers/net/can/janz-ican3.c | |||
| @@ -1618,7 +1618,7 @@ static ssize_t ican3_sysfs_set_term(struct device *dev, | |||
| 1618 | return count; | 1618 | return count; |
| 1619 | } | 1619 | } |
| 1620 | 1620 | ||
| 1621 | static DEVICE_ATTR(termination, S_IWUGO | S_IRUGO, ican3_sysfs_show_term, | 1621 | static DEVICE_ATTR(termination, S_IWUSR | S_IRUGO, ican3_sysfs_show_term, |
| 1622 | ican3_sysfs_set_term); | 1622 | ican3_sysfs_set_term); |
| 1623 | 1623 | ||
| 1624 | static struct attribute *ican3_sysfs_attrs[] = { | 1624 | static struct attribute *ican3_sysfs_attrs[] = { |
diff --git a/drivers/net/can/softing/Kconfig b/drivers/net/can/softing/Kconfig new file mode 100644 index 00000000000..8ba81b3ddd9 --- /dev/null +++ b/drivers/net/can/softing/Kconfig | |||
| @@ -0,0 +1,30 @@ | |||
| 1 | config CAN_SOFTING | ||
| 2 | tristate "Softing Gmbh CAN generic support" | ||
| 3 | depends on CAN_DEV && HAS_IOMEM | ||
| 4 | ---help--- | ||
| 5 | Support for CAN cards from Softing Gmbh & some cards | ||
| 6 | from Vector Gmbh. | ||
| 7 | Softing Gmbh CAN cards come with 1 or 2 physical busses. | ||
| 8 | Those cards typically use Dual Port RAM to communicate | ||
| 9 | with the host CPU. The interface is then identical for PCI | ||
| 10 | and PCMCIA cards. This driver operates on a platform device, | ||
| 11 | which has been created by softing_cs or softing_pci driver. | ||
| 12 | Warning: | ||
| 13 | The API of the card does not allow fine control per bus, but | ||
| 14 | controls the 2 busses on the card together. | ||
| 15 | As such, some actions (start/stop/busoff recovery) on 1 bus | ||
| 16 | must bring down the other bus too temporarily. | ||
| 17 | |||
| 18 | config CAN_SOFTING_CS | ||
| 19 | tristate "Softing Gmbh CAN pcmcia cards" | ||
| 20 | depends on PCMCIA | ||
| 21 | select CAN_SOFTING | ||
| 22 | ---help--- | ||
| 23 | Support for PCMCIA cards from Softing Gmbh & some cards | ||
| 24 | from Vector Gmbh. | ||
| 25 | You need firmware for these, which you can get at | ||
| 26 | http://developer.berlios.de/projects/socketcan/ | ||
| 27 | This version of the driver is written against | ||
| 28 | firmware version 4.6 (softing-fw-4.6-binaries.tar.gz) | ||
| 29 | In order to use the card as CAN device, you need the Softing generic | ||
| 30 | support too. | ||
diff --git a/drivers/net/can/softing/Makefile b/drivers/net/can/softing/Makefile new file mode 100644 index 00000000000..c5e5016c742 --- /dev/null +++ b/drivers/net/can/softing/Makefile | |||
| @@ -0,0 +1,6 @@ | |||
| 1 | |||
| 2 | softing-y := softing_main.o softing_fw.o | ||
| 3 | obj-$(CONFIG_CAN_SOFTING) += softing.o | ||
| 4 | obj-$(CONFIG_CAN_SOFTING_CS) += softing_cs.o | ||
| 5 | |||
| 6 | ccflags-$(CONFIG_CAN_DEBUG_DEVICES) := -DDEBUG | ||
diff --git a/drivers/net/can/softing/softing.h b/drivers/net/can/softing/softing.h new file mode 100644 index 00000000000..7ec9f4db3d5 --- /dev/null +++ b/drivers/net/can/softing/softing.h | |||
| @@ -0,0 +1,167 @@ | |||
| 1 | /* | ||
| 2 | * softing common interfaces | ||
| 3 | * | ||
| 4 | * by Kurt Van Dijck, 2008-2010 | ||
| 5 | */ | ||
| 6 | |||
| 7 | #include <linux/atomic.h> | ||
| 8 | #include <linux/netdevice.h> | ||
| 9 | #include <linux/ktime.h> | ||
| 10 | #include <linux/mutex.h> | ||
| 11 | #include <linux/spinlock.h> | ||
| 12 | #include <linux/can.h> | ||
| 13 | #include <linux/can/dev.h> | ||
| 14 | |||
| 15 | #include "softing_platform.h" | ||
| 16 | |||
| 17 | struct softing; | ||
| 18 | |||
| 19 | struct softing_priv { | ||
| 20 | struct can_priv can; /* must be the first member! */ | ||
| 21 | struct net_device *netdev; | ||
| 22 | struct softing *card; | ||
| 23 | struct { | ||
| 24 | int pending; | ||
| 25 | /* variables wich hold the circular buffer */ | ||
| 26 | int echo_put; | ||
| 27 | int echo_get; | ||
| 28 | } tx; | ||
| 29 | struct can_bittiming_const btr_const; | ||
| 30 | int index; | ||
| 31 | uint8_t output; | ||
| 32 | uint16_t chip; | ||
| 33 | }; | ||
| 34 | #define netdev2softing(netdev) ((struct softing_priv *)netdev_priv(netdev)) | ||
| 35 | |||
| 36 | struct softing { | ||
| 37 | const struct softing_platform_data *pdat; | ||
| 38 | struct platform_device *pdev; | ||
| 39 | struct net_device *net[2]; | ||
| 40 | spinlock_t spin; /* protect this structure & DPRAM access */ | ||
| 41 | ktime_t ts_ref; | ||
| 42 | ktime_t ts_overflow; /* timestamp overflow value, in ktime */ | ||
| 43 | |||
| 44 | struct { | ||
| 45 | /* indication of firmware status */ | ||
| 46 | int up; | ||
| 47 | /* protection of the 'up' variable */ | ||
| 48 | struct mutex lock; | ||
| 49 | } fw; | ||
| 50 | struct { | ||
| 51 | int nr; | ||
| 52 | int requested; | ||
| 53 | int svc_count; | ||
| 54 | unsigned int dpram_position; | ||
| 55 | } irq; | ||
| 56 | struct { | ||
| 57 | int pending; | ||
| 58 | int last_bus; | ||
| 59 | /* | ||
| 60 | * keep the bus that last tx'd a message, | ||
| 61 | * in order to let every netdev queue resume | ||
| 62 | */ | ||
| 63 | } tx; | ||
| 64 | __iomem uint8_t *dpram; | ||
| 65 | unsigned long dpram_phys; | ||
| 66 | unsigned long dpram_size; | ||
| 67 | struct { | ||
| 68 | uint16_t fw_version, hw_version, license, serial; | ||
| 69 | uint16_t chip[2]; | ||
| 70 | unsigned int freq; /* remote cpu's operating frequency */ | ||
| 71 | } id; | ||
| 72 | }; | ||
| 73 | |||
| 74 | extern int softing_default_output(struct net_device *netdev); | ||
| 75 | |||
| 76 | extern ktime_t softing_raw2ktime(struct softing *card, u32 raw); | ||
| 77 | |||
| 78 | extern int softing_chip_poweron(struct softing *card); | ||
| 79 | |||
| 80 | extern int softing_bootloader_command(struct softing *card, int16_t cmd, | ||
| 81 | const char *msg); | ||
| 82 | |||
| 83 | /* Load firmware after reset */ | ||
| 84 | extern int softing_load_fw(const char *file, struct softing *card, | ||
| 85 | __iomem uint8_t *virt, unsigned int size, int offset); | ||
| 86 | |||
| 87 | /* Load final application firmware after bootloader */ | ||
| 88 | extern int softing_load_app_fw(const char *file, struct softing *card); | ||
| 89 | |||
| 90 | /* | ||
| 91 | * enable or disable irq | ||
| 92 | * only called with fw.lock locked | ||
| 93 | */ | ||
| 94 | extern int softing_enable_irq(struct softing *card, int enable); | ||
| 95 | |||
| 96 | /* start/stop 1 bus on card */ | ||
| 97 | extern int softing_startstop(struct net_device *netdev, int up); | ||
| 98 | |||
| 99 | /* netif_rx() */ | ||
| 100 | extern int softing_netdev_rx(struct net_device *netdev, | ||
| 101 | const struct can_frame *msg, ktime_t ktime); | ||
| 102 | |||
| 103 | /* SOFTING DPRAM mappings */ | ||
| 104 | #define DPRAM_RX 0x0000 | ||
| 105 | #define DPRAM_RX_SIZE 32 | ||
| 106 | #define DPRAM_RX_CNT 16 | ||
| 107 | #define DPRAM_RX_RD 0x0201 /* uint8_t */ | ||
| 108 | #define DPRAM_RX_WR 0x0205 /* uint8_t */ | ||
| 109 | #define DPRAM_RX_LOST 0x0207 /* uint8_t */ | ||
| 110 | |||
| 111 | #define DPRAM_FCT_PARAM 0x0300 /* int16_t [20] */ | ||
| 112 | #define DPRAM_FCT_RESULT 0x0328 /* int16_t */ | ||
| 113 | #define DPRAM_FCT_HOST 0x032b /* uint16_t */ | ||
| 114 | |||
| 115 | #define DPRAM_INFO_BUSSTATE 0x0331 /* uint16_t */ | ||
| 116 | #define DPRAM_INFO_BUSSTATE2 0x0335 /* uint16_t */ | ||
| 117 | #define DPRAM_INFO_ERRSTATE 0x0339 /* uint16_t */ | ||
| 118 | #define DPRAM_INFO_ERRSTATE2 0x033d /* uint16_t */ | ||
| 119 | #define DPRAM_RESET 0x0341 /* uint16_t */ | ||
| 120 | #define DPRAM_CLR_RECV_FIFO 0x0345 /* uint16_t */ | ||
| 121 | #define DPRAM_RESET_TIME 0x034d /* uint16_t */ | ||
| 122 | #define DPRAM_TIME 0x0350 /* uint64_t */ | ||
| 123 | #define DPRAM_WR_START 0x0358 /* uint8_t */ | ||
| 124 | #define DPRAM_WR_END 0x0359 /* uint8_t */ | ||
| 125 | #define DPRAM_RESET_RX_FIFO 0x0361 /* uint16_t */ | ||
| 126 | #define DPRAM_RESET_TX_FIFO 0x0364 /* uint8_t */ | ||
| 127 | #define DPRAM_READ_FIFO_LEVEL 0x0365 /* uint8_t */ | ||
| 128 | #define DPRAM_RX_FIFO_LEVEL 0x0366 /* uint16_t */ | ||
| 129 | #define DPRAM_TX_FIFO_LEVEL 0x0366 /* uint16_t */ | ||
| 130 | |||
| 131 | #define DPRAM_TX 0x0400 /* uint16_t */ | ||
| 132 | #define DPRAM_TX_SIZE 16 | ||
| 133 | #define DPRAM_TX_CNT 32 | ||
| 134 | #define DPRAM_TX_RD 0x0601 /* uint8_t */ | ||
| 135 | #define DPRAM_TX_WR 0x0605 /* uint8_t */ | ||
| 136 | |||
| 137 | #define DPRAM_COMMAND 0x07e0 /* uint16_t */ | ||
| 138 | #define DPRAM_RECEIPT 0x07f0 /* uint16_t */ | ||
| 139 | #define DPRAM_IRQ_TOHOST 0x07fe /* uint8_t */ | ||
| 140 | #define DPRAM_IRQ_TOCARD 0x07ff /* uint8_t */ | ||
| 141 | |||
| 142 | #define DPRAM_V2_RESET 0x0e00 /* uint8_t */ | ||
| 143 | #define DPRAM_V2_IRQ_TOHOST 0x0e02 /* uint8_t */ | ||
| 144 | |||
| 145 | #define TXMAX (DPRAM_TX_CNT - 1) | ||
| 146 | |||
| 147 | /* DPRAM return codes */ | ||
| 148 | #define RES_NONE 0 | ||
| 149 | #define RES_OK 1 | ||
| 150 | #define RES_NOK 2 | ||
| 151 | #define RES_UNKNOWN 3 | ||
| 152 | /* DPRAM flags */ | ||
| 153 | #define CMD_TX 0x01 | ||
| 154 | #define CMD_ACK 0x02 | ||
| 155 | #define CMD_XTD 0x04 | ||
| 156 | #define CMD_RTR 0x08 | ||
| 157 | #define CMD_ERR 0x10 | ||
| 158 | #define CMD_BUS2 0x80 | ||
| 159 | |||
| 160 | /* returned fifo entry bus state masks */ | ||
| 161 | #define SF_MASK_BUSOFF 0x80 | ||
| 162 | #define SF_MASK_EPASSIVE 0x60 | ||
| 163 | |||
| 164 | /* bus states */ | ||
| 165 | #define STATE_BUSOFF 2 | ||
| 166 | #define STATE_EPASSIVE 1 | ||
| 167 | #define STATE_EACTIVE 0 | ||
diff --git a/drivers/net/can/softing/softing_cs.c b/drivers/net/can/softing/softing_cs.c new file mode 100644 index 00000000000..300fe75dd1a --- /dev/null +++ b/drivers/net/can/softing/softing_cs.c | |||
| @@ -0,0 +1,359 @@ | |||
| 1 | /* | ||
| 2 | * Copyright (C) 2008-2010 | ||
| 3 | * | ||
| 4 | * - Kurt Van Dijck, EIA Electronics | ||
| 5 | * | ||
| 6 | * This program is free software; you can redistribute it and/or modify | ||
| 7 | * it under the terms of the version 2 of the GNU General Public License | ||
| 8 | * as published by the Free Software Foundation | ||
| 9 | * | ||
| 10 | * This program is distributed in the hope that it will be useful, | ||
| 11 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 12 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
| 13 | * GNU General Public License for more details. | ||
| 14 | * | ||
| 15 | * You should have received a copy of the GNU General Public License | ||
| 16 | * along with this program; if not, write to the Free Software | ||
| 17 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | ||
| 18 | */ | ||
| 19 | |||
| 20 | #include <linux/module.h> | ||
| 21 | #include <linux/kernel.h> | ||
| 22 | |||
| 23 | #include <pcmcia/cistpl.h> | ||
| 24 | #include <pcmcia/ds.h> | ||
| 25 | |||
| 26 | #include "softing_platform.h" | ||
| 27 | |||
| 28 | static int softingcs_index; | ||
| 29 | static spinlock_t softingcs_index_lock; | ||
| 30 | |||
| 31 | static int softingcs_reset(struct platform_device *pdev, int v); | ||
| 32 | static int softingcs_enable_irq(struct platform_device *pdev, int v); | ||
| 33 | |||
| 34 | /* | ||
| 35 | * platform_data descriptions | ||
| 36 | */ | ||
| 37 | #define MHZ (1000*1000) | ||
| 38 | static const struct softing_platform_data softingcs_platform_data[] = { | ||
| 39 | { | ||
| 40 | .name = "CANcard", | ||
| 41 | .manf = 0x0168, .prod = 0x001, | ||
| 42 | .generation = 1, | ||
| 43 | .nbus = 2, | ||
| 44 | .freq = 16 * MHZ, .max_brp = 32, .max_sjw = 4, | ||
| 45 | .dpram_size = 0x0800, | ||
| 46 | .boot = {0x0000, 0x000000, fw_dir "bcard.bin",}, | ||
| 47 | .load = {0x0120, 0x00f600, fw_dir "ldcard.bin",}, | ||
| 48 | .app = {0x0010, 0x0d0000, fw_dir "cancard.bin",}, | ||
| 49 | .reset = softingcs_reset, | ||
| 50 | .enable_irq = softingcs_enable_irq, | ||
| 51 | }, { | ||
| 52 | .name = "CANcard-NEC", | ||
| 53 | .manf = 0x0168, .prod = 0x002, | ||
| 54 | .generation = 1, | ||
| 55 | .nbus = 2, | ||
| 56 | .freq = 16 * MHZ, .max_brp = 32, .max_sjw = 4, | ||
| 57 | .dpram_size = 0x0800, | ||
| 58 | .boot = {0x0000, 0x000000, fw_dir "bcard.bin",}, | ||
| 59 | .load = {0x0120, 0x00f600, fw_dir "ldcard.bin",}, | ||
| 60 | .app = {0x0010, 0x0d0000, fw_dir "cancard.bin",}, | ||
| 61 | .reset = softingcs_reset, | ||
| 62 | .enable_irq = softingcs_enable_irq, | ||
| 63 | }, { | ||
| 64 | .name = "CANcard-SJA", | ||
| 65 | .manf = 0x0168, .prod = 0x004, | ||
| 66 | .generation = 1, | ||
| 67 | .nbus = 2, | ||
| 68 | .freq = 20 * MHZ, .max_brp = 32, .max_sjw = 4, | ||
| 69 | .dpram_size = 0x0800, | ||
| 70 | .boot = {0x0000, 0x000000, fw_dir "bcard.bin",}, | ||
| 71 | .load = {0x0120, 0x00f600, fw_dir "ldcard.bin",}, | ||
| 72 | .app = {0x0010, 0x0d0000, fw_dir "cansja.bin",}, | ||
| 73 | .reset = softingcs_reset, | ||
| 74 | .enable_irq = softingcs_enable_irq, | ||
| 75 | }, { | ||
| 76 | .name = "CANcard-2", | ||
| 77 | .manf = 0x0168, .prod = 0x005, | ||
| 78 | .generation = 2, | ||
| 79 | .nbus = 2, | ||
| 80 | .freq = 24 * MHZ, .max_brp = 64, .max_sjw = 4, | ||
| 81 | .dpram_size = 0x1000, | ||
| 82 | .boot = {0x0000, 0x000000, fw_dir "bcard2.bin",}, | ||
| 83 | .load = {0x0120, 0x00f600, fw_dir "ldcard2.bin",}, | ||
| 84 | .app = {0x0010, 0x0d0000, fw_dir "cancrd2.bin",}, | ||
| 85 | .reset = softingcs_reset, | ||
| 86 | .enable_irq = NULL, | ||
| 87 | }, { | ||
| 88 | .name = "Vector-CANcard", | ||
| 89 | .manf = 0x0168, .prod = 0x081, | ||
| 90 | .generation = 1, | ||
| 91 | .nbus = 2, | ||
| 92 | .freq = 16 * MHZ, .max_brp = 64, .max_sjw = 4, | ||
| 93 | .dpram_size = 0x0800, | ||
| 94 | .boot = {0x0000, 0x000000, fw_dir "bcard.bin",}, | ||
| 95 | .load = {0x0120, 0x00f600, fw_dir "ldcard.bin",}, | ||
| 96 | .app = {0x0010, 0x0d0000, fw_dir "cancard.bin",}, | ||
| 97 | .reset = softingcs_reset, | ||
| 98 | .enable_irq = softingcs_enable_irq, | ||
| 99 | }, { | ||
| 100 | .name = "Vector-CANcard-SJA", | ||
| 101 | .manf = 0x0168, .prod = 0x084, | ||
| 102 | .generation = 1, | ||
| 103 | .nbus = 2, | ||
| 104 | .freq = 20 * MHZ, .max_brp = 32, .max_sjw = 4, | ||
| 105 | .dpram_size = 0x0800, | ||
| 106 | .boot = {0x0000, 0x000000, fw_dir "bcard.bin",}, | ||
| 107 | .load = {0x0120, 0x00f600, fw_dir "ldcard.bin",}, | ||
| 108 | .app = {0x0010, 0x0d0000, fw_dir "cansja.bin",}, | ||
| 109 | .reset = softingcs_reset, | ||
| 110 | .enable_irq = softingcs_enable_irq, | ||
| 111 | }, { | ||
| 112 | .name = "Vector-CANcard-2", | ||
| 113 | .manf = 0x0168, .prod = 0x085, | ||
| 114 | .generation = 2, | ||
| 115 | .nbus = 2, | ||
| 116 | .freq = 24 * MHZ, .max_brp = 64, .max_sjw = 4, | ||
| 117 | .dpram_size = 0x1000, | ||
| 118 | .boot = {0x0000, 0x000000, fw_dir "bcard2.bin",}, | ||
| 119 | .load = {0x0120, 0x00f600, fw_dir "ldcard2.bin",}, | ||
| 120 | .app = {0x0010, 0x0d0000, fw_dir "cancrd2.bin",}, | ||
| 121 | .reset = softingcs_reset, | ||
| 122 | .enable_irq = NULL, | ||
| 123 | }, { | ||
| 124 | .name = "EDICcard-NEC", | ||
| 125 | .manf = 0x0168, .prod = 0x102, | ||
| 126 | .generation = 1, | ||
| 127 | .nbus = 2, | ||
| 128 | .freq = 16 * MHZ, .max_brp = 64, .max_sjw = 4, | ||
| 129 | .dpram_size = 0x0800, | ||
| 130 | .boot = {0x0000, 0x000000, fw_dir "bcard.bin",}, | ||
| 131 | .load = {0x0120, 0x00f600, fw_dir "ldcard.bin",}, | ||
| 132 | .app = {0x0010, 0x0d0000, fw_dir "cancard.bin",}, | ||
| 133 | .reset = softingcs_reset, | ||
| 134 | .enable_irq = softingcs_enable_irq, | ||
| 135 | }, { | ||
| 136 | .name = "EDICcard-2", | ||
| 137 | .manf = 0x0168, .prod = 0x105, | ||
| 138 | .generation = 2, | ||
| 139 | .nbus = 2, | ||
| 140 | .freq = 24 * MHZ, .max_brp = 64, .max_sjw = 4, | ||
| 141 | .dpram_size = 0x1000, | ||
| 142 | .boot = {0x0000, 0x000000, fw_dir "bcard2.bin",}, | ||
| 143 | .load = {0x0120, 0x00f600, fw_dir "ldcard2.bin",}, | ||
| 144 | .app = {0x0010, 0x0d0000, fw_dir "cancrd2.bin",}, | ||
| 145 | .reset = softingcs_reset, | ||
| 146 | .enable_irq = NULL, | ||
| 147 | }, { | ||
| 148 | 0, 0, | ||
| 149 | }, | ||
| 150 | }; | ||
| 151 | |||
| 152 | MODULE_FIRMWARE(fw_dir "bcard.bin"); | ||
| 153 | MODULE_FIRMWARE(fw_dir "ldcard.bin"); | ||
| 154 | MODULE_FIRMWARE(fw_dir "cancard.bin"); | ||
| 155 | MODULE_FIRMWARE(fw_dir "cansja.bin"); | ||
| 156 | |||
| 157 | MODULE_FIRMWARE(fw_dir "bcard2.bin"); | ||
| 158 | MODULE_FIRMWARE(fw_dir "ldcard2.bin"); | ||
| 159 | MODULE_FIRMWARE(fw_dir "cancrd2.bin"); | ||
| 160 | |||
| 161 | static __devinit const struct softing_platform_data | ||
| 162 | *softingcs_find_platform_data(unsigned int manf, unsigned int prod) | ||
| 163 | { | ||
| 164 | const struct softing_platform_data *lp; | ||
| 165 | |||
| 166 | for (lp = softingcs_platform_data; lp->manf; ++lp) { | ||
| 167 | if ((lp->manf == manf) && (lp->prod == prod)) | ||
| 168 | return lp; | ||
| 169 | } | ||
| 170 | return NULL; | ||
| 171 | } | ||
| 172 | |||
| 173 | /* | ||
| 174 | * platformdata callbacks | ||
| 175 | */ | ||
| 176 | static int softingcs_reset(struct platform_device *pdev, int v) | ||
| 177 | { | ||
| 178 | struct pcmcia_device *pcmcia = to_pcmcia_dev(pdev->dev.parent); | ||
| 179 | |||
| 180 | dev_dbg(&pdev->dev, "pcmcia config [2] %02x\n", v ? 0 : 0x20); | ||
| 181 | return pcmcia_write_config_byte(pcmcia, 2, v ? 0 : 0x20); | ||
| 182 | } | ||
| 183 | |||
| 184 | static int softingcs_enable_irq(struct platform_device *pdev, int v) | ||
| 185 | { | ||
| 186 | struct pcmcia_device *pcmcia = to_pcmcia_dev(pdev->dev.parent); | ||
| 187 | |||
| 188 | dev_dbg(&pdev->dev, "pcmcia config [0] %02x\n", v ? 0x60 : 0); | ||
| 189 | return pcmcia_write_config_byte(pcmcia, 0, v ? 0x60 : 0); | ||
| 190 | } | ||
| 191 | |||
| 192 | /* | ||
| 193 | * pcmcia check | ||
| 194 | */ | ||
| 195 | static __devinit int softingcs_probe_config(struct pcmcia_device *pcmcia, | ||
| 196 | void *priv_data) | ||
| 197 | { | ||
| 198 | struct softing_platform_data *pdat = priv_data; | ||
| 199 | struct resource *pres; | ||
| 200 | int memspeed = 0; | ||
| 201 | |||
| 202 | WARN_ON(!pdat); | ||
| 203 | pres = pcmcia->resource[PCMCIA_IOMEM_0]; | ||
| 204 | if (resource_size(pres) < 0x1000) | ||
| 205 | return -ERANGE; | ||
| 206 | |||
| 207 | pres->flags |= WIN_MEMORY_TYPE_CM | WIN_ENABLE; | ||
| 208 | if (pdat->generation < 2) { | ||
| 209 | pres->flags |= WIN_USE_WAIT | WIN_DATA_WIDTH_8; | ||
| 210 | memspeed = 3; | ||
| 211 | } else { | ||
| 212 | pres->flags |= WIN_DATA_WIDTH_16; | ||
| 213 | } | ||
| 214 | return pcmcia_request_window(pcmcia, pres, memspeed); | ||
| 215 | } | ||
| 216 | |||
| 217 | static __devexit void softingcs_remove(struct pcmcia_device *pcmcia) | ||
| 218 | { | ||
| 219 | struct platform_device *pdev = pcmcia->priv; | ||
| 220 | |||
| 221 | /* free bits */ | ||
| 222 | platform_device_unregister(pdev); | ||
| 223 | /* release pcmcia stuff */ | ||
| 224 | pcmcia_disable_device(pcmcia); | ||
| 225 | } | ||
| 226 | |||
| 227 | /* | ||
| 228 | * platform_device wrapper | ||
| 229 | * pdev->resource has 2 entries: io & irq | ||
| 230 | */ | ||
| 231 | static void softingcs_pdev_release(struct device *dev) | ||
| 232 | { | ||
| 233 | struct platform_device *pdev = to_platform_device(dev); | ||
| 234 | kfree(pdev); | ||
| 235 | } | ||
| 236 | |||
| 237 | static __devinit int softingcs_probe(struct pcmcia_device *pcmcia) | ||
| 238 | { | ||
| 239 | int ret; | ||
| 240 | struct platform_device *pdev; | ||
| 241 | const struct softing_platform_data *pdat; | ||
| 242 | struct resource *pres; | ||
| 243 | struct dev { | ||
| 244 | struct platform_device pdev; | ||
| 245 | struct resource res[2]; | ||
| 246 | } *dev; | ||
| 247 | |||
| 248 | /* find matching platform_data */ | ||
| 249 | pdat = softingcs_find_platform_data(pcmcia->manf_id, pcmcia->card_id); | ||
| 250 | if (!pdat) | ||
| 251 | return -ENOTTY; | ||
| 252 | |||
| 253 | /* setup pcmcia device */ | ||
| 254 | pcmcia->config_flags |= CONF_ENABLE_IRQ | CONF_AUTO_SET_IOMEM | | ||
| 255 | CONF_AUTO_SET_VPP | CONF_AUTO_CHECK_VCC; | ||
| 256 | ret = pcmcia_loop_config(pcmcia, softingcs_probe_config, (void *)pdat); | ||
| 257 | if (ret) | ||
| 258 | goto pcmcia_failed; | ||
| 259 | |||
| 260 | ret = pcmcia_enable_device(pcmcia); | ||
| 261 | if (ret < 0) | ||
| 262 | goto pcmcia_failed; | ||
| 263 | |||
| 264 | pres = pcmcia->resource[PCMCIA_IOMEM_0]; | ||
| 265 | if (!pres) { | ||
| 266 | ret = -EBADF; | ||
| 267 | goto pcmcia_bad; | ||
| 268 | } | ||
| 269 | |||
| 270 | /* create softing platform device */ | ||
| 271 | dev = kzalloc(sizeof(*dev), GFP_KERNEL); | ||
| 272 | if (!dev) { | ||
| 273 | ret = -ENOMEM; | ||
| 274 | goto mem_failed; | ||
| 275 | } | ||
| 276 | dev->pdev.resource = dev->res; | ||
| 277 | dev->pdev.num_resources = ARRAY_SIZE(dev->res); | ||
| 278 | dev->pdev.dev.release = softingcs_pdev_release; | ||
| 279 | |||
| 280 | pdev = &dev->pdev; | ||
| 281 | pdev->dev.platform_data = (void *)pdat; | ||
| 282 | pdev->dev.parent = &pcmcia->dev; | ||
| 283 | pcmcia->priv = pdev; | ||
| 284 | |||
| 285 | /* platform device resources */ | ||
| 286 | pdev->resource[0].flags = IORESOURCE_MEM; | ||
| 287 | pdev->resource[0].start = pres->start; | ||
| 288 | pdev->resource[0].end = pres->end; | ||
| 289 | |||
| 290 | pdev->resource[1].flags = IORESOURCE_IRQ; | ||
| 291 | pdev->resource[1].start = pcmcia->irq; | ||
| 292 | pdev->resource[1].end = pdev->resource[1].start; | ||
| 293 | |||
| 294 | /* platform device setup */ | ||
| 295 | spin_lock(&softingcs_index_lock); | ||
| 296 | pdev->id = softingcs_index++; | ||
| 297 | spin_unlock(&softingcs_index_lock); | ||
| 298 | pdev->name = "softing"; | ||
| 299 | dev_set_name(&pdev->dev, "softingcs.%i", pdev->id); | ||
| 300 | ret = platform_device_register(pdev); | ||
| 301 | if (ret < 0) | ||
| 302 | goto platform_failed; | ||
| 303 | |||
| 304 | dev_info(&pcmcia->dev, "created %s\n", dev_name(&pdev->dev)); | ||
| 305 | return 0; | ||
| 306 | |||
| 307 | platform_failed: | ||
| 308 | kfree(dev); | ||
| 309 | mem_failed: | ||
| 310 | pcmcia_bad: | ||
| 311 | pcmcia_failed: | ||
| 312 | pcmcia_disable_device(pcmcia); | ||
| 313 | pcmcia->priv = NULL; | ||
| 314 | return ret ?: -ENODEV; | ||
| 315 | } | ||
| 316 | |||
| 317 | static /*const*/ struct pcmcia_device_id softingcs_ids[] = { | ||
| 318 | /* softing */ | ||
| 319 | PCMCIA_DEVICE_MANF_CARD(0x0168, 0x0001), | ||
| 320 | PCMCIA_DEVICE_MANF_CARD(0x0168, 0x0002), | ||
| 321 | PCMCIA_DEVICE_MANF_CARD(0x0168, 0x0004), | ||
| 322 | PCMCIA_DEVICE_MANF_CARD(0x0168, 0x0005), | ||
| 323 | /* vector, manufacturer? */ | ||
| 324 | PCMCIA_DEVICE_MANF_CARD(0x0168, 0x0081), | ||
| 325 | PCMCIA_DEVICE_MANF_CARD(0x0168, 0x0084), | ||
| 326 | PCMCIA_DEVICE_MANF_CARD(0x0168, 0x0085), | ||
| 327 | /* EDIC */ | ||
| 328 | PCMCIA_DEVICE_MANF_CARD(0x0168, 0x0102), | ||
| 329 | PCMCIA_DEVICE_MANF_CARD(0x0168, 0x0105), | ||
| 330 | PCMCIA_DEVICE_NULL, | ||
| 331 | }; | ||
| 332 | |||
| 333 | MODULE_DEVICE_TABLE(pcmcia, softingcs_ids); | ||
| 334 | |||
| 335 | static struct pcmcia_driver softingcs_driver = { | ||
| 336 | .owner = THIS_MODULE, | ||
| 337 | .name = "softingcs", | ||
| 338 | .id_table = softingcs_ids, | ||
| 339 | .probe = softingcs_probe, | ||
| 340 | .remove = __devexit_p(softingcs_remove), | ||
| 341 | }; | ||
| 342 | |||
| 343 | static int __init softingcs_start(void) | ||
| 344 | { | ||
| 345 | spin_lock_init(&softingcs_index_lock); | ||
| 346 | return pcmcia_register_driver(&softingcs_driver); | ||
| 347 | } | ||
| 348 | |||
| 349 | static void __exit softingcs_stop(void) | ||
| 350 | { | ||
| 351 | pcmcia_unregister_driver(&softingcs_driver); | ||
| 352 | } | ||
| 353 | |||
| 354 | module_init(softingcs_start); | ||
| 355 | module_exit(softingcs_stop); | ||
| 356 | |||
| 357 | MODULE_DESCRIPTION("softing CANcard driver" | ||
| 358 | ", links PCMCIA card to softing driver"); | ||
| 359 | MODULE_LICENSE("GPL v2"); | ||
diff --git a/drivers/net/can/softing/softing_fw.c b/drivers/net/can/softing/softing_fw.c new file mode 100644 index 00000000000..b520784fb19 --- /dev/null +++ b/drivers/net/can/softing/softing_fw.c | |||
| @@ -0,0 +1,691 @@ | |||
| 1 | /* | ||
| 2 | * Copyright (C) 2008-2010 | ||
| 3 | * | ||
| 4 | * - Kurt Van Dijck, EIA Electronics | ||
| 5 | * | ||
| 6 | * This program is free software; you can redistribute it and/or modify | ||
| 7 | * it under the terms of the version 2 of the GNU General Public License | ||
| 8 | * as published by the Free Software Foundation | ||
| 9 | * | ||
| 10 | * This program is distributed in the hope that it will be useful, | ||
| 11 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 12 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
| 13 | * GNU General Public License for more details. | ||
| 14 | * | ||
| 15 | * You should have received a copy of the GNU General Public License | ||
| 16 | * along with this program; if not, write to the Free Software | ||
| 17 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | ||
| 18 | */ | ||
| 19 | |||
| 20 | #include <linux/firmware.h> | ||
| 21 | #include <linux/sched.h> | ||
| 22 | #include <asm/div64.h> | ||
| 23 | |||
| 24 | #include "softing.h" | ||
| 25 | |||
| 26 | /* | ||
| 27 | * low level DPRAM command. | ||
| 28 | * Make sure that card->dpram[DPRAM_FCT_HOST] is preset | ||
| 29 | */ | ||
| 30 | static int _softing_fct_cmd(struct softing *card, int16_t cmd, uint16_t vector, | ||
| 31 | const char *msg) | ||
| 32 | { | ||
| 33 | int ret; | ||
| 34 | unsigned long stamp; | ||
| 35 | |||
| 36 | iowrite16(cmd, &card->dpram[DPRAM_FCT_PARAM]); | ||
| 37 | iowrite8(vector >> 8, &card->dpram[DPRAM_FCT_HOST + 1]); | ||
| 38 | iowrite8(vector, &card->dpram[DPRAM_FCT_HOST]); | ||
| 39 | /* be sure to flush this to the card */ | ||
| 40 | wmb(); | ||
| 41 | stamp = jiffies + 1 * HZ; | ||
| 42 | /* wait for card */ | ||
| 43 | do { | ||
| 44 | /* DPRAM_FCT_HOST is _not_ aligned */ | ||
| 45 | ret = ioread8(&card->dpram[DPRAM_FCT_HOST]) + | ||
| 46 | (ioread8(&card->dpram[DPRAM_FCT_HOST + 1]) << 8); | ||
| 47 | /* don't have any cached variables */ | ||
| 48 | rmb(); | ||
| 49 | if (ret == RES_OK) | ||
| 50 | /* read return-value now */ | ||
| 51 | return ioread16(&card->dpram[DPRAM_FCT_RESULT]); | ||
| 52 | |||
| 53 | if ((ret != vector) || time_after(jiffies, stamp)) | ||
| 54 | break; | ||
| 55 | /* process context => relax */ | ||
| 56 | usleep_range(500, 10000); | ||
| 57 | } while (1); | ||
| 58 | |||
| 59 | ret = (ret == RES_NONE) ? -ETIMEDOUT : -ECANCELED; | ||
| 60 | dev_alert(&card->pdev->dev, "firmware %s failed (%i)\n", msg, ret); | ||
| 61 | return ret; | ||
| 62 | } | ||
| 63 | |||
| 64 | static int softing_fct_cmd(struct softing *card, int16_t cmd, const char *msg) | ||
| 65 | { | ||
| 66 | int ret; | ||
| 67 | |||
| 68 | ret = _softing_fct_cmd(card, cmd, 0, msg); | ||
| 69 | if (ret > 0) { | ||
| 70 | dev_alert(&card->pdev->dev, "%s returned %u\n", msg, ret); | ||
| 71 | ret = -EIO; | ||
| 72 | } | ||
| 73 | return ret; | ||
| 74 | } | ||
| 75 | |||
| 76 | int softing_bootloader_command(struct softing *card, int16_t cmd, | ||
| 77 | const char *msg) | ||
| 78 | { | ||
| 79 | int ret; | ||
| 80 | unsigned long stamp; | ||
| 81 | |||
| 82 | iowrite16(RES_NONE, &card->dpram[DPRAM_RECEIPT]); | ||
| 83 | iowrite16(cmd, &card->dpram[DPRAM_COMMAND]); | ||
| 84 | /* be sure to flush this to the card */ | ||
| 85 | wmb(); | ||
| 86 | stamp = jiffies + 3 * HZ; | ||
| 87 | /* wait for card */ | ||
| 88 | do { | ||
| 89 | ret = ioread16(&card->dpram[DPRAM_RECEIPT]); | ||
| 90 | /* don't have any cached variables */ | ||
| 91 | rmb(); | ||
| 92 | if (ret == RES_OK) | ||
| 93 | return 0; | ||
| 94 | if (time_after(jiffies, stamp)) | ||
| 95 | break; | ||
| 96 | /* process context => relax */ | ||
| 97 | usleep_range(500, 10000); | ||
| 98 | } while (!signal_pending(current)); | ||
| 99 | |||
| 100 | ret = (ret == RES_NONE) ? -ETIMEDOUT : -ECANCELED; | ||
| 101 | dev_alert(&card->pdev->dev, "bootloader %s failed (%i)\n", msg, ret); | ||
| 102 | return ret; | ||
| 103 | } | ||
| 104 | |||
| 105 | static int fw_parse(const uint8_t **pmem, uint16_t *ptype, uint32_t *paddr, | ||
| 106 | uint16_t *plen, const uint8_t **pdat) | ||
| 107 | { | ||
| 108 | uint16_t checksum[2]; | ||
| 109 | const uint8_t *mem; | ||
| 110 | const uint8_t *end; | ||
| 111 | |||
| 112 | /* | ||
| 113 | * firmware records are a binary, unaligned stream composed of: | ||
| 114 | * uint16_t type; | ||
| 115 | * uint32_t addr; | ||
| 116 | * uint16_t len; | ||
| 117 | * uint8_t dat[len]; | ||
| 118 | * uint16_t checksum; | ||
| 119 | * all values in little endian. | ||
| 120 | * We could define a struct for this, with __attribute__((packed)), | ||
| 121 | * but would that solve the alignment in _all_ cases (cfr. the | ||
| 122 | * struct itself may be an odd address)? | ||
| 123 | * | ||
| 124 | * I chose to use leXX_to_cpup() since this solves both | ||
| 125 | * endianness & alignment. | ||
| 126 | */ | ||
| 127 | mem = *pmem; | ||
| 128 | *ptype = le16_to_cpup((void *)&mem[0]); | ||
| 129 | *paddr = le32_to_cpup((void *)&mem[2]); | ||
| 130 | *plen = le16_to_cpup((void *)&mem[6]); | ||
| 131 | *pdat = &mem[8]; | ||
| 132 | /* verify checksum */ | ||
| 133 | end = &mem[8 + *plen]; | ||
| 134 | checksum[0] = le16_to_cpup((void *)end); | ||
| 135 | for (checksum[1] = 0; mem < end; ++mem) | ||
| 136 | checksum[1] += *mem; | ||
| 137 | if (checksum[0] != checksum[1]) | ||
| 138 | return -EINVAL; | ||
| 139 | /* increment */ | ||
| 140 | *pmem += 10 + *plen; | ||
| 141 | return 0; | ||
| 142 | } | ||
| 143 | |||
| 144 | int softing_load_fw(const char *file, struct softing *card, | ||
| 145 | __iomem uint8_t *dpram, unsigned int size, int offset) | ||
| 146 | { | ||
| 147 | const struct firmware *fw; | ||
| 148 | int ret; | ||
| 149 | const uint8_t *mem, *end, *dat; | ||
| 150 | uint16_t type, len; | ||
| 151 | uint32_t addr; | ||
| 152 | uint8_t *buf = NULL; | ||
| 153 | int buflen = 0; | ||
| 154 | int8_t type_end = 0; | ||
| 155 | |||
| 156 | ret = request_firmware(&fw, file, &card->pdev->dev); | ||
| 157 | if (ret < 0) | ||
| 158 | return ret; | ||
| 159 | dev_dbg(&card->pdev->dev, "%s, firmware(%s) got %u bytes" | ||
| 160 | ", offset %c0x%04x\n", | ||
| 161 | card->pdat->name, file, (unsigned int)fw->size, | ||
| 162 | (offset >= 0) ? '+' : '-', (unsigned int)abs(offset)); | ||
| 163 | /* parse the firmware */ | ||
| 164 | mem = fw->data; | ||
| 165 | end = &mem[fw->size]; | ||
| 166 | /* look for header record */ | ||
| 167 | ret = fw_parse(&mem, &type, &addr, &len, &dat); | ||
| 168 | if (ret < 0) | ||
| 169 | goto failed; | ||
| 170 | if (type != 0xffff) | ||
| 171 | goto failed; | ||
| 172 | if (strncmp("Structured Binary Format, Softing GmbH" , dat, len)) { | ||
| 173 | ret = -EINVAL; | ||
| 174 | goto failed; | ||
| 175 | } | ||
| 176 | /* ok, we had a header */ | ||
| 177 | while (mem < end) { | ||
| 178 | ret = fw_parse(&mem, &type, &addr, &len, &dat); | ||
| 179 | if (ret < 0) | ||
| 180 | goto failed; | ||
| 181 | if (type == 3) { | ||
| 182 | /* start address, not used here */ | ||
| 183 | continue; | ||
| 184 | } else if (type == 1) { | ||
| 185 | /* eof */ | ||
| 186 | type_end = 1; | ||
| 187 | break; | ||
| 188 | } else if (type != 0) { | ||
| 189 | ret = -EINVAL; | ||
| 190 | goto failed; | ||
| 191 | } | ||
| 192 | |||
| 193 | if ((addr + len + offset) > size) | ||
| 194 | goto failed; | ||
| 195 | memcpy_toio(&dpram[addr + offset], dat, len); | ||
| 196 | /* be sure to flush caches from IO space */ | ||
| 197 | mb(); | ||
| 198 | if (len > buflen) { | ||
| 199 | /* align buflen */ | ||
| 200 | buflen = (len + (1024-1)) & ~(1024-1); | ||
| 201 | buf = krealloc(buf, buflen, GFP_KERNEL); | ||
| 202 | if (!buf) { | ||
| 203 | ret = -ENOMEM; | ||
| 204 | goto failed; | ||
| 205 | } | ||
| 206 | } | ||
| 207 | /* verify record data */ | ||
| 208 | memcpy_fromio(buf, &dpram[addr + offset], len); | ||
| 209 | if (memcmp(buf, dat, len)) { | ||
| 210 | /* is not ok */ | ||
| 211 | dev_alert(&card->pdev->dev, "DPRAM readback failed\n"); | ||
| 212 | ret = -EIO; | ||
| 213 | goto failed; | ||
| 214 | } | ||
| 215 | } | ||
| 216 | if (!type_end) | ||
| 217 | /* no end record seen */ | ||
| 218 | goto failed; | ||
| 219 | ret = 0; | ||
| 220 | failed: | ||
| 221 | kfree(buf); | ||
| 222 | release_firmware(fw); | ||
| 223 | if (ret < 0) | ||
| 224 | dev_info(&card->pdev->dev, "firmware %s failed\n", file); | ||
| 225 | return ret; | ||
| 226 | } | ||
| 227 | |||
| 228 | int softing_load_app_fw(const char *file, struct softing *card) | ||
| 229 | { | ||
| 230 | const struct firmware *fw; | ||
| 231 | const uint8_t *mem, *end, *dat; | ||
| 232 | int ret, j; | ||
| 233 | uint16_t type, len; | ||
| 234 | uint32_t addr, start_addr = 0; | ||
| 235 | unsigned int sum, rx_sum; | ||
| 236 | int8_t type_end = 0, type_entrypoint = 0; | ||
| 237 | |||
| 238 | ret = request_firmware(&fw, file, &card->pdev->dev); | ||
| 239 | if (ret) { | ||
| 240 | dev_alert(&card->pdev->dev, "request_firmware(%s) got %i\n", | ||
| 241 | file, ret); | ||
| 242 | return ret; | ||
| 243 | } | ||
| 244 | dev_dbg(&card->pdev->dev, "firmware(%s) got %lu bytes\n", | ||
| 245 | file, (unsigned long)fw->size); | ||
| 246 | /* parse the firmware */ | ||
| 247 | mem = fw->data; | ||
| 248 | end = &mem[fw->size]; | ||
| 249 | /* look for header record */ | ||
| 250 | ret = fw_parse(&mem, &type, &addr, &len, &dat); | ||
| 251 | if (ret) | ||
| 252 | goto failed; | ||
| 253 | ret = -EINVAL; | ||
| 254 | if (type != 0xffff) { | ||
| 255 | dev_alert(&card->pdev->dev, "firmware starts with type 0x%x\n", | ||
| 256 | type); | ||
| 257 | goto failed; | ||
| 258 | } | ||
| 259 | if (strncmp("Structured Binary Format, Softing GmbH", dat, len)) { | ||
| 260 | dev_alert(&card->pdev->dev, "firmware string '%.*s' fault\n", | ||
| 261 | len, dat); | ||
| 262 | goto failed; | ||
| 263 | } | ||
| 264 | /* ok, we had a header */ | ||
| 265 | while (mem < end) { | ||
| 266 | ret = fw_parse(&mem, &type, &addr, &len, &dat); | ||
| 267 | if (ret) | ||
| 268 | goto failed; | ||
| 269 | |||
| 270 | if (type == 3) { | ||
| 271 | /* start address */ | ||
| 272 | start_addr = addr; | ||
| 273 | type_entrypoint = 1; | ||
| 274 | continue; | ||
| 275 | } else if (type == 1) { | ||
| 276 | /* eof */ | ||
| 277 | type_end = 1; | ||
| 278 | break; | ||
| 279 | } else if (type != 0) { | ||
| 280 | dev_alert(&card->pdev->dev, | ||
| 281 | "unknown record type 0x%04x\n", type); | ||
| 282 | ret = -EINVAL; | ||
| 283 | goto failed; | ||
| 284 | } | ||
| 285 | |||
| 286 | /* regualar data */ | ||
| 287 | for (sum = 0, j = 0; j < len; ++j) | ||
| 288 | sum += dat[j]; | ||
| 289 | /* work in 16bit (target) */ | ||
| 290 | sum &= 0xffff; | ||
| 291 | |||
| 292 | memcpy_toio(&card->dpram[card->pdat->app.offs], dat, len); | ||
| 293 | iowrite32(card->pdat->app.offs + card->pdat->app.addr, | ||
| 294 | &card->dpram[DPRAM_COMMAND + 2]); | ||
| 295 | iowrite32(addr, &card->dpram[DPRAM_COMMAND + 6]); | ||
| 296 | iowrite16(len, &card->dpram[DPRAM_COMMAND + 10]); | ||
| 297 | iowrite8(1, &card->dpram[DPRAM_COMMAND + 12]); | ||
| 298 | ret = softing_bootloader_command(card, 1, "loading app."); | ||
| 299 | if (ret < 0) | ||
| 300 | goto failed; | ||
| 301 | /* verify checksum */ | ||
| 302 | rx_sum = ioread16(&card->dpram[DPRAM_RECEIPT + 2]); | ||
| 303 | if (rx_sum != sum) { | ||
| 304 | dev_alert(&card->pdev->dev, "SRAM seems to be damaged" | ||
| 305 | ", wanted 0x%04x, got 0x%04x\n", sum, rx_sum); | ||
| 306 | ret = -EIO; | ||
| 307 | goto failed; | ||
| 308 | } | ||
| 309 | } | ||
| 310 | if (!type_end || !type_entrypoint) | ||
| 311 | goto failed; | ||
| 312 | /* start application in card */ | ||
| 313 | iowrite32(start_addr, &card->dpram[DPRAM_COMMAND + 2]); | ||
| 314 | iowrite8(1, &card->dpram[DPRAM_COMMAND + 6]); | ||
| 315 | ret = softing_bootloader_command(card, 3, "start app."); | ||
| 316 | if (ret < 0) | ||
| 317 | goto failed; | ||
| 318 | ret = 0; | ||
| 319 | failed: | ||
| 320 | release_firmware(fw); | ||
| 321 | if (ret < 0) | ||
| 322 | dev_info(&card->pdev->dev, "firmware %s failed\n", file); | ||
| 323 | return ret; | ||
| 324 | } | ||
| 325 | |||
| 326 | static int softing_reset_chip(struct softing *card) | ||
| 327 | { | ||
| 328 | int ret; | ||
| 329 | |||
| 330 | do { | ||
| 331 | /* reset chip */ | ||
| 332 | iowrite8(0, &card->dpram[DPRAM_RESET_RX_FIFO]); | ||
| 333 | iowrite8(0, &card->dpram[DPRAM_RESET_RX_FIFO+1]); | ||
| 334 | iowrite8(1, &card->dpram[DPRAM_RESET]); | ||
| 335 | iowrite8(0, &card->dpram[DPRAM_RESET+1]); | ||
| 336 | |||
| 337 | ret = softing_fct_cmd(card, 0, "reset_can"); | ||
| 338 | if (!ret) | ||
| 339 | break; | ||
| 340 | if (signal_pending(current)) | ||
| 341 | /* don't wait any longer */ | ||
| 342 | break; | ||
| 343 | } while (1); | ||
| 344 | card->tx.pending = 0; | ||
| 345 | return ret; | ||
| 346 | } | ||
| 347 | |||
| 348 | int softing_chip_poweron(struct softing *card) | ||
| 349 | { | ||
| 350 | int ret; | ||
| 351 | /* sync */ | ||
| 352 | ret = _softing_fct_cmd(card, 99, 0x55, "sync-a"); | ||
| 353 | if (ret < 0) | ||
| 354 | goto failed; | ||
| 355 | |||
| 356 | ret = _softing_fct_cmd(card, 99, 0xaa, "sync-b"); | ||
| 357 | if (ret < 0) | ||
| 358 | goto failed; | ||
| 359 | |||
| 360 | ret = softing_reset_chip(card); | ||
| 361 | if (ret < 0) | ||
| 362 | goto failed; | ||
| 363 | /* get_serial */ | ||
| 364 | ret = softing_fct_cmd(card, 43, "get_serial_number"); | ||
| 365 | if (ret < 0) | ||
| 366 | goto failed; | ||
| 367 | card->id.serial = ioread32(&card->dpram[DPRAM_FCT_PARAM]); | ||
| 368 | /* get_version */ | ||
| 369 | ret = softing_fct_cmd(card, 12, "get_version"); | ||
| 370 | if (ret < 0) | ||
| 371 | goto failed; | ||
| 372 | card->id.fw_version = ioread16(&card->dpram[DPRAM_FCT_PARAM + 2]); | ||
| 373 | card->id.hw_version = ioread16(&card->dpram[DPRAM_FCT_PARAM + 4]); | ||
| 374 | card->id.license = ioread16(&card->dpram[DPRAM_FCT_PARAM + 6]); | ||
| 375 | card->id.chip[0] = ioread16(&card->dpram[DPRAM_FCT_PARAM + 8]); | ||
| 376 | card->id.chip[1] = ioread16(&card->dpram[DPRAM_FCT_PARAM + 10]); | ||
| 377 | return 0; | ||
| 378 | failed: | ||
| 379 | return ret; | ||
| 380 | } | ||
| 381 | |||
| 382 | static void softing_initialize_timestamp(struct softing *card) | ||
| 383 | { | ||
| 384 | uint64_t ovf; | ||
| 385 | |||
| 386 | card->ts_ref = ktime_get(); | ||
| 387 | |||
| 388 | /* 16MHz is the reference */ | ||
| 389 | ovf = 0x100000000ULL * 16; | ||
| 390 | do_div(ovf, card->pdat->freq ?: 16); | ||
| 391 | |||
| 392 | card->ts_overflow = ktime_add_us(ktime_set(0, 0), ovf); | ||
| 393 | } | ||
| 394 | |||
| 395 | ktime_t softing_raw2ktime(struct softing *card, u32 raw) | ||
| 396 | { | ||
| 397 | uint64_t rawl; | ||
| 398 | ktime_t now, real_offset; | ||
| 399 | ktime_t target; | ||
| 400 | ktime_t tmp; | ||
| 401 | |||
| 402 | now = ktime_get(); | ||
| 403 | real_offset = ktime_sub(ktime_get_real(), now); | ||
| 404 | |||
| 405 | /* find nsec from card */ | ||
| 406 | rawl = raw * 16; | ||
| 407 | do_div(rawl, card->pdat->freq ?: 16); | ||
| 408 | target = ktime_add_us(card->ts_ref, rawl); | ||
| 409 | /* test for overflows */ | ||
| 410 | tmp = ktime_add(target, card->ts_overflow); | ||
| 411 | while (unlikely(ktime_to_ns(tmp) > ktime_to_ns(now))) { | ||
| 412 | card->ts_ref = ktime_add(card->ts_ref, card->ts_overflow); | ||
| 413 | target = tmp; | ||
| 414 | tmp = ktime_add(target, card->ts_overflow); | ||
| 415 | } | ||
| 416 | return ktime_add(target, real_offset); | ||
| 417 | } | ||
| 418 | |||
| 419 | static inline int softing_error_reporting(struct net_device *netdev) | ||
| 420 | { | ||
| 421 | struct softing_priv *priv = netdev_priv(netdev); | ||
| 422 | |||
| 423 | return (priv->can.ctrlmode & CAN_CTRLMODE_BERR_REPORTING) | ||
| 424 | ? 1 : 0; | ||
| 425 | } | ||
| 426 | |||
| 427 | int softing_startstop(struct net_device *dev, int up) | ||
| 428 | { | ||
| 429 | int ret; | ||
| 430 | struct softing *card; | ||
| 431 | struct softing_priv *priv; | ||
| 432 | struct net_device *netdev; | ||
| 433 | int bus_bitmask_start; | ||
| 434 | int j, error_reporting; | ||
| 435 | struct can_frame msg; | ||
| 436 | const struct can_bittiming *bt; | ||
| 437 | |||
| 438 | priv = netdev_priv(dev); | ||
| 439 | card = priv->card; | ||
| 440 | |||
| 441 | if (!card->fw.up) | ||
| 442 | return -EIO; | ||
| 443 | |||
| 444 | ret = mutex_lock_interruptible(&card->fw.lock); | ||
| 445 | if (ret) | ||
| 446 | return ret; | ||
| 447 | |||
| 448 | bus_bitmask_start = 0; | ||
| 449 | if (dev && up) | ||
| 450 | /* prepare to start this bus as well */ | ||
| 451 | bus_bitmask_start |= (1 << priv->index); | ||
| 452 | /* bring netdevs down */ | ||
| 453 | for (j = 0; j < ARRAY_SIZE(card->net); ++j) { | ||
| 454 | netdev = card->net[j]; | ||
| 455 | if (!netdev) | ||
| 456 | continue; | ||
| 457 | priv = netdev_priv(netdev); | ||
| 458 | |||
| 459 | if (dev != netdev) | ||
| 460 | netif_stop_queue(netdev); | ||
| 461 | |||
| 462 | if (netif_running(netdev)) { | ||
| 463 | if (dev != netdev) | ||
| 464 | bus_bitmask_start |= (1 << j); | ||
| 465 | priv->tx.pending = 0; | ||
| 466 | priv->tx.echo_put = 0; | ||
| 467 | priv->tx.echo_get = 0; | ||
| 468 | /* | ||
| 469 | * this bus' may just have called open_candev() | ||
| 470 | * which is rather stupid to call close_candev() | ||
| 471 | * already | ||
| 472 | * but we may come here from busoff recovery too | ||
| 473 | * in which case the echo_skb _needs_ flushing too. | ||
| 474 | * just be sure to call open_candev() again | ||
| 475 | */ | ||
| 476 | close_candev(netdev); | ||
| 477 | } | ||
| 478 | priv->can.state = CAN_STATE_STOPPED; | ||
| 479 | } | ||
| 480 | card->tx.pending = 0; | ||
| 481 | |||
| 482 | softing_enable_irq(card, 0); | ||
| 483 | ret = softing_reset_chip(card); | ||
| 484 | if (ret) | ||
| 485 | goto failed; | ||
| 486 | if (!bus_bitmask_start) | ||
| 487 | /* no busses to be brought up */ | ||
| 488 | goto card_done; | ||
| 489 | |||
| 490 | if ((bus_bitmask_start & 1) && (bus_bitmask_start & 2) | ||
| 491 | && (softing_error_reporting(card->net[0]) | ||
| 492 | != softing_error_reporting(card->net[1]))) { | ||
| 493 | dev_alert(&card->pdev->dev, | ||
| 494 | "err_reporting flag differs for busses\n"); | ||
| 495 | goto invalid; | ||
| 496 | } | ||
| 497 | error_reporting = 0; | ||
| 498 | if (bus_bitmask_start & 1) { | ||
| 499 | netdev = card->net[0]; | ||
| 500 | priv = netdev_priv(netdev); | ||
| 501 | error_reporting += softing_error_reporting(netdev); | ||
| 502 | /* init chip 1 */ | ||
| 503 | bt = &priv->can.bittiming; | ||
| 504 | iowrite16(bt->brp, &card->dpram[DPRAM_FCT_PARAM + 2]); | ||
| 505 | iowrite16(bt->sjw, &card->dpram[DPRAM_FCT_PARAM + 4]); | ||
| 506 | iowrite16(bt->phase_seg1 + bt->prop_seg, | ||
| 507 | &card->dpram[DPRAM_FCT_PARAM + 6]); | ||
| 508 | iowrite16(bt->phase_seg2, &card->dpram[DPRAM_FCT_PARAM + 8]); | ||
| 509 | iowrite16((priv->can.ctrlmode & CAN_CTRLMODE_3_SAMPLES) ? 1 : 0, | ||
| 510 | &card->dpram[DPRAM_FCT_PARAM + 10]); | ||
| 511 | ret = softing_fct_cmd(card, 1, "initialize_chip[0]"); | ||
| 512 | if (ret < 0) | ||
| 513 | goto failed; | ||
| 514 | /* set mode */ | ||
| 515 | iowrite16(0, &card->dpram[DPRAM_FCT_PARAM + 2]); | ||
| 516 | iowrite16(0, &card->dpram[DPRAM_FCT_PARAM + 4]); | ||
| 517 | ret = softing_fct_cmd(card, 3, "set_mode[0]"); | ||
| 518 | if (ret < 0) | ||
| 519 | goto failed; | ||
| 520 | /* set filter */ | ||
| 521 | /* 11bit id & mask */ | ||
| 522 | iowrite16(0x0000, &card->dpram[DPRAM_FCT_PARAM + 2]); | ||
| 523 | iowrite16(0x07ff, &card->dpram[DPRAM_FCT_PARAM + 4]); | ||
| 524 | /* 29bit id.lo & mask.lo & id.hi & mask.hi */ | ||
| 525 | iowrite16(0x0000, &card->dpram[DPRAM_FCT_PARAM + 6]); | ||
| 526 | iowrite16(0xffff, &card->dpram[DPRAM_FCT_PARAM + 8]); | ||
| 527 | iowrite16(0x0000, &card->dpram[DPRAM_FCT_PARAM + 10]); | ||
| 528 | iowrite16(0x1fff, &card->dpram[DPRAM_FCT_PARAM + 12]); | ||
| 529 | ret = softing_fct_cmd(card, 7, "set_filter[0]"); | ||
| 530 | if (ret < 0) | ||
| 531 | goto failed; | ||
| 532 | /* set output control */ | ||
| 533 | iowrite16(priv->output, &card->dpram[DPRAM_FCT_PARAM + 2]); | ||
| 534 | ret = softing_fct_cmd(card, 5, "set_output[0]"); | ||
| 535 | if (ret < 0) | ||
| 536 | goto failed; | ||
| 537 | } | ||
| 538 | if (bus_bitmask_start & 2) { | ||
| 539 | netdev = card->net[1]; | ||
| 540 | priv = netdev_priv(netdev); | ||
| 541 | error_reporting += softing_error_reporting(netdev); | ||
| 542 | /* init chip2 */ | ||
| 543 | bt = &priv->can.bittiming; | ||
| 544 | iowrite16(bt->brp, &card->dpram[DPRAM_FCT_PARAM + 2]); | ||
| 545 | iowrite16(bt->sjw, &card->dpram[DPRAM_FCT_PARAM + 4]); | ||
| 546 | iowrite16(bt->phase_seg1 + bt->prop_seg, | ||
| 547 | &card->dpram[DPRAM_FCT_PARAM + 6]); | ||
| 548 | iowrite16(bt->phase_seg2, &card->dpram[DPRAM_FCT_PARAM + 8]); | ||
| 549 | iowrite16((priv->can.ctrlmode & CAN_CTRLMODE_3_SAMPLES) ? 1 : 0, | ||
| 550 | &card->dpram[DPRAM_FCT_PARAM + 10]); | ||
| 551 | ret = softing_fct_cmd(card, 2, "initialize_chip[1]"); | ||
| 552 | if (ret < 0) | ||
| 553 | goto failed; | ||
| 554 | /* set mode2 */ | ||
| 555 | iowrite16(0, &card->dpram[DPRAM_FCT_PARAM + 2]); | ||
| 556 | iowrite16(0, &card->dpram[DPRAM_FCT_PARAM + 4]); | ||
| 557 | ret = softing_fct_cmd(card, 4, "set_mode[1]"); | ||
| 558 | if (ret < 0) | ||
| 559 | goto failed; | ||
| 560 | /* set filter2 */ | ||
| 561 | /* 11bit id & mask */ | ||
| 562 | iowrite16(0x0000, &card->dpram[DPRAM_FCT_PARAM + 2]); | ||
| 563 | iowrite16(0x07ff, &card->dpram[DPRAM_FCT_PARAM + 4]); | ||
| 564 | /* 29bit id.lo & mask.lo & id.hi & mask.hi */ | ||
| 565 | iowrite16(0x0000, &card->dpram[DPRAM_FCT_PARAM + 6]); | ||
| 566 | iowrite16(0xffff, &card->dpram[DPRAM_FCT_PARAM + 8]); | ||
| 567 | iowrite16(0x0000, &card->dpram[DPRAM_FCT_PARAM + 10]); | ||
| 568 | iowrite16(0x1fff, &card->dpram[DPRAM_FCT_PARAM + 12]); | ||
| 569 | ret = softing_fct_cmd(card, 8, "set_filter[1]"); | ||
| 570 | if (ret < 0) | ||
| 571 | goto failed; | ||
| 572 | /* set output control2 */ | ||
| 573 | iowrite16(priv->output, &card->dpram[DPRAM_FCT_PARAM + 2]); | ||
| 574 | ret = softing_fct_cmd(card, 6, "set_output[1]"); | ||
| 575 | if (ret < 0) | ||
| 576 | goto failed; | ||
| 577 | } | ||
| 578 | /* enable_error_frame */ | ||
| 579 | /* | ||
| 580 | * Error reporting is switched off at the moment since | ||
| 581 | * the receiving of them is not yet 100% verified | ||
| 582 | * This should be enabled sooner or later | ||
| 583 | * | ||
| 584 | if (error_reporting) { | ||
| 585 | ret = softing_fct_cmd(card, 51, "enable_error_frame"); | ||
| 586 | if (ret < 0) | ||
| 587 | goto failed; | ||
| 588 | } | ||
| 589 | */ | ||
| 590 | /* initialize interface */ | ||
| 591 | iowrite16(1, &card->dpram[DPRAM_FCT_PARAM + 2]); | ||
| 592 | iowrite16(1, &card->dpram[DPRAM_FCT_PARAM + 4]); | ||
| 593 | iowrite16(1, &card->dpram[DPRAM_FCT_PARAM + 6]); | ||
| 594 | iowrite16(1, &card->dpram[DPRAM_FCT_PARAM + 8]); | ||
| 595 | iowrite16(1, &card->dpram[DPRAM_FCT_PARAM + 10]); | ||
| 596 | iowrite16(1, &card->dpram[DPRAM_FCT_PARAM + 12]); | ||
| 597 | iowrite16(1, &card->dpram[DPRAM_FCT_PARAM + 14]); | ||
| 598 | iowrite16(1, &card->dpram[DPRAM_FCT_PARAM + 16]); | ||
| 599 | iowrite16(1, &card->dpram[DPRAM_FCT_PARAM + 18]); | ||
| 600 | iowrite16(1, &card->dpram[DPRAM_FCT_PARAM + 20]); | ||
| 601 | ret = softing_fct_cmd(card, 17, "initialize_interface"); | ||
| 602 | if (ret < 0) | ||
| 603 | goto failed; | ||
| 604 | /* enable_fifo */ | ||
| 605 | ret = softing_fct_cmd(card, 36, "enable_fifo"); | ||
| 606 | if (ret < 0) | ||
| 607 | goto failed; | ||
| 608 | /* enable fifo tx ack */ | ||
| 609 | ret = softing_fct_cmd(card, 13, "fifo_tx_ack[0]"); | ||
| 610 | if (ret < 0) | ||
| 611 | goto failed; | ||
| 612 | /* enable fifo tx ack2 */ | ||
| 613 | ret = softing_fct_cmd(card, 14, "fifo_tx_ack[1]"); | ||
| 614 | if (ret < 0) | ||
| 615 | goto failed; | ||
| 616 | /* start_chip */ | ||
| 617 | ret = softing_fct_cmd(card, 11, "start_chip"); | ||
| 618 | if (ret < 0) | ||
| 619 | goto failed; | ||
| 620 | iowrite8(0, &card->dpram[DPRAM_INFO_BUSSTATE]); | ||
| 621 | iowrite8(0, &card->dpram[DPRAM_INFO_BUSSTATE2]); | ||
| 622 | if (card->pdat->generation < 2) { | ||
| 623 | iowrite8(0, &card->dpram[DPRAM_V2_IRQ_TOHOST]); | ||
| 624 | /* flush the DPRAM caches */ | ||
| 625 | wmb(); | ||
| 626 | } | ||
| 627 | |||
| 628 | softing_initialize_timestamp(card); | ||
| 629 | |||
| 630 | /* | ||
| 631 | * do socketcan notifications/status changes | ||
| 632 | * from here, no errors should occur, or the failed: part | ||
| 633 | * must be reviewed | ||
| 634 | */ | ||
| 635 | memset(&msg, 0, sizeof(msg)); | ||
| 636 | msg.can_id = CAN_ERR_FLAG | CAN_ERR_RESTARTED; | ||
| 637 | msg.can_dlc = CAN_ERR_DLC; | ||
| 638 | for (j = 0; j < ARRAY_SIZE(card->net); ++j) { | ||
| 639 | if (!(bus_bitmask_start & (1 << j))) | ||
| 640 | continue; | ||
| 641 | netdev = card->net[j]; | ||
| 642 | if (!netdev) | ||
| 643 | continue; | ||
| 644 | priv = netdev_priv(netdev); | ||
| 645 | priv->can.state = CAN_STATE_ERROR_ACTIVE; | ||
| 646 | open_candev(netdev); | ||
| 647 | if (dev != netdev) { | ||
| 648 | /* notify other busses on the restart */ | ||
| 649 | softing_netdev_rx(netdev, &msg, ktime_set(0, 0)); | ||
| 650 | ++priv->can.can_stats.restarts; | ||
| 651 | } | ||
| 652 | netif_wake_queue(netdev); | ||
| 653 | } | ||
| 654 | |||
| 655 | /* enable interrupts */ | ||
| 656 | ret = softing_enable_irq(card, 1); | ||
| 657 | if (ret) | ||
| 658 | goto failed; | ||
| 659 | card_done: | ||
| 660 | mutex_unlock(&card->fw.lock); | ||
| 661 | return 0; | ||
| 662 | invalid: | ||
| 663 | ret = -EINVAL; | ||
| 664 | failed: | ||
| 665 | softing_enable_irq(card, 0); | ||
| 666 | softing_reset_chip(card); | ||
| 667 | mutex_unlock(&card->fw.lock); | ||
| 668 | /* bring all other interfaces down */ | ||
| 669 | for (j = 0; j < ARRAY_SIZE(card->net); ++j) { | ||
| 670 | netdev = card->net[j]; | ||
| 671 | if (!netdev) | ||
| 672 | continue; | ||
| 673 | dev_close(netdev); | ||
| 674 | } | ||
| 675 | return ret; | ||
| 676 | } | ||
| 677 | |||
| 678 | int softing_default_output(struct net_device *netdev) | ||
| 679 | { | ||
| 680 | struct softing_priv *priv = netdev_priv(netdev); | ||
| 681 | struct softing *card = priv->card; | ||
| 682 | |||
| 683 | switch (priv->chip) { | ||
| 684 | case 1000: | ||
| 685 | return (card->pdat->generation < 2) ? 0xfb : 0xfa; | ||
| 686 | case 5: | ||
| 687 | return 0x60; | ||
| 688 | default: | ||
| 689 | return 0x40; | ||
| 690 | } | ||
| 691 | } | ||
diff --git a/drivers/net/can/softing/softing_main.c b/drivers/net/can/softing/softing_main.c new file mode 100644 index 00000000000..5157e15e96e --- /dev/null +++ b/drivers/net/can/softing/softing_main.c | |||
| @@ -0,0 +1,893 @@ | |||
| 1 | /* | ||
| 2 | * Copyright (C) 2008-2010 | ||
| 3 | * | ||
| 4 | * - Kurt Van Dijck, EIA Electronics | ||
| 5 | * | ||
| 6 | * This program is free software; you can redistribute it and/or modify | ||
| 7 | * it under the terms of the version 2 of the GNU General Public License | ||
| 8 | * as published by the Free Software Foundation | ||
| 9 | * | ||
| 10 | * This program is distributed in the hope that it will be useful, | ||
| 11 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 12 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
| 13 | * GNU General Public License for more details. | ||
| 14 | * | ||
| 15 | * You should have received a copy of the GNU General Public License | ||
| 16 | * along with this program; if not, write to the Free Software | ||
| 17 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | ||
| 18 | */ | ||
| 19 | |||
| 20 | #include <linux/version.h> | ||
| 21 | #include <linux/module.h> | ||
| 22 | #include <linux/init.h> | ||
| 23 | #include <linux/interrupt.h> | ||
| 24 | |||
| 25 | #include "softing.h" | ||
| 26 | |||
| 27 | #define TX_ECHO_SKB_MAX (((TXMAX+1)/2)-1) | ||
| 28 | |||
| 29 | /* | ||
| 30 | * test is a specific CAN netdev | ||
| 31 | * is online (ie. up 'n running, not sleeping, not busoff | ||
| 32 | */ | ||
| 33 | static inline int canif_is_active(struct net_device *netdev) | ||
| 34 | { | ||
| 35 | struct can_priv *can = netdev_priv(netdev); | ||
| 36 | |||
| 37 | if (!netif_running(netdev)) | ||
| 38 | return 0; | ||
| 39 | return (can->state <= CAN_STATE_ERROR_PASSIVE); | ||
| 40 | } | ||
| 41 | |||
| 42 | /* reset DPRAM */ | ||
| 43 | static inline void softing_set_reset_dpram(struct softing *card) | ||
| 44 | { | ||
| 45 | if (card->pdat->generation >= 2) { | ||
| 46 | spin_lock_bh(&card->spin); | ||
| 47 | iowrite8(ioread8(&card->dpram[DPRAM_V2_RESET]) & ~1, | ||
| 48 | &card->dpram[DPRAM_V2_RESET]); | ||
| 49 | spin_unlock_bh(&card->spin); | ||
| 50 | } | ||
| 51 | } | ||
| 52 | |||
| 53 | static inline void softing_clr_reset_dpram(struct softing *card) | ||
| 54 | { | ||
| 55 | if (card->pdat->generation >= 2) { | ||
| 56 | spin_lock_bh(&card->spin); | ||
| 57 | iowrite8(ioread8(&card->dpram[DPRAM_V2_RESET]) | 1, | ||
| 58 | &card->dpram[DPRAM_V2_RESET]); | ||
| 59 | spin_unlock_bh(&card->spin); | ||
| 60 | } | ||
| 61 | } | ||
| 62 | |||
| 63 | /* trigger the tx queue-ing */ | ||
| 64 | static netdev_tx_t softing_netdev_start_xmit(struct sk_buff *skb, | ||
| 65 | struct net_device *dev) | ||
| 66 | { | ||
| 67 | struct softing_priv *priv = netdev_priv(dev); | ||
| 68 | struct softing *card = priv->card; | ||
| 69 | int ret; | ||
| 70 | uint8_t *ptr; | ||
| 71 | uint8_t fifo_wr, fifo_rd; | ||
| 72 | struct can_frame *cf = (struct can_frame *)skb->data; | ||
| 73 | uint8_t buf[DPRAM_TX_SIZE]; | ||
| 74 | |||
| 75 | if (can_dropped_invalid_skb(dev, skb)) | ||
| 76 | return NETDEV_TX_OK; | ||
| 77 | |||
| 78 | spin_lock(&card->spin); | ||
| 79 | |||
| 80 | ret = NETDEV_TX_BUSY; | ||
| 81 | if (!card->fw.up || | ||
| 82 | (card->tx.pending >= TXMAX) || | ||
| 83 | (priv->tx.pending >= TX_ECHO_SKB_MAX)) | ||
| 84 | goto xmit_done; | ||
| 85 | fifo_wr = ioread8(&card->dpram[DPRAM_TX_WR]); | ||
| 86 | fifo_rd = ioread8(&card->dpram[DPRAM_TX_RD]); | ||
| 87 | if (fifo_wr == fifo_rd) | ||
| 88 | /* fifo full */ | ||
| 89 | goto xmit_done; | ||
| 90 | memset(buf, 0, sizeof(buf)); | ||
| 91 | ptr = buf; | ||
| 92 | *ptr = CMD_TX; | ||
| 93 | if (cf->can_id & CAN_RTR_FLAG) | ||
| 94 | *ptr |= CMD_RTR; | ||
| 95 | if (cf->can_id & CAN_EFF_FLAG) | ||
| 96 | *ptr |= CMD_XTD; | ||
| 97 | if (priv->index) | ||
| 98 | *ptr |= CMD_BUS2; | ||
| 99 | ++ptr; | ||
| 100 | *ptr++ = cf->can_dlc; | ||
| 101 | *ptr++ = (cf->can_id >> 0); | ||
| 102 | *ptr++ = (cf->can_id >> 8); | ||
| 103 | if (cf->can_id & CAN_EFF_FLAG) { | ||
| 104 | *ptr++ = (cf->can_id >> 16); | ||
| 105 | *ptr++ = (cf->can_id >> 24); | ||
| 106 | } else { | ||
| 107 | /* increment 1, not 2 as you might think */ | ||
| 108 | ptr += 1; | ||
| 109 | } | ||
| 110 | if (!(cf->can_id & CAN_RTR_FLAG)) | ||
| 111 | memcpy(ptr, &cf->data[0], cf->can_dlc); | ||
| 112 | memcpy_toio(&card->dpram[DPRAM_TX + DPRAM_TX_SIZE * fifo_wr], | ||
| 113 | buf, DPRAM_TX_SIZE); | ||
| 114 | if (++fifo_wr >= DPRAM_TX_CNT) | ||
| 115 | fifo_wr = 0; | ||
| 116 | iowrite8(fifo_wr, &card->dpram[DPRAM_TX_WR]); | ||
| 117 | card->tx.last_bus = priv->index; | ||
| 118 | ++card->tx.pending; | ||
| 119 | ++priv->tx.pending; | ||
| 120 | can_put_echo_skb(skb, dev, priv->tx.echo_put); | ||
| 121 | ++priv->tx.echo_put; | ||
| 122 | if (priv->tx.echo_put >= TX_ECHO_SKB_MAX) | ||
| 123 | priv->tx.echo_put = 0; | ||
| 124 | /* can_put_echo_skb() saves the skb, safe to return TX_OK */ | ||
| 125 | ret = NETDEV_TX_OK; | ||
| 126 | xmit_done: | ||
| 127 | spin_unlock(&card->spin); | ||
| 128 | if (card->tx.pending >= TXMAX) { | ||
| 129 | int j; | ||
| 130 | for (j = 0; j < ARRAY_SIZE(card->net); ++j) { | ||
| 131 | if (card->net[j]) | ||
| 132 | netif_stop_queue(card->net[j]); | ||
| 133 | } | ||
| 134 | } | ||
| 135 | if (ret != NETDEV_TX_OK) | ||
| 136 | netif_stop_queue(dev); | ||
| 137 | |||
| 138 | return ret; | ||
| 139 | } | ||
| 140 | |||
| 141 | /* | ||
| 142 | * shortcut for skb delivery | ||
| 143 | */ | ||
| 144 | int softing_netdev_rx(struct net_device *netdev, const struct can_frame *msg, | ||
| 145 | ktime_t ktime) | ||
| 146 | { | ||
| 147 | struct sk_buff *skb; | ||
| 148 | struct can_frame *cf; | ||
| 149 | |||
| 150 | skb = alloc_can_skb(netdev, &cf); | ||
| 151 | if (!skb) | ||
| 152 | return -ENOMEM; | ||
| 153 | memcpy(cf, msg, sizeof(*msg)); | ||
| 154 | skb->tstamp = ktime; | ||
| 155 | return netif_rx(skb); | ||
| 156 | } | ||
| 157 | |||
| 158 | /* | ||
| 159 | * softing_handle_1 | ||
| 160 | * pop 1 entry from the DPRAM queue, and process | ||
| 161 | */ | ||
| 162 | static int softing_handle_1(struct softing *card) | ||
| 163 | { | ||
| 164 | struct net_device *netdev; | ||
| 165 | struct softing_priv *priv; | ||
| 166 | ktime_t ktime; | ||
| 167 | struct can_frame msg; | ||
| 168 | int cnt = 0, lost_msg; | ||
| 169 | uint8_t fifo_rd, fifo_wr, cmd; | ||
| 170 | uint8_t *ptr; | ||
| 171 | uint32_t tmp_u32; | ||
| 172 | uint8_t buf[DPRAM_RX_SIZE]; | ||
| 173 | |||
| 174 | memset(&msg, 0, sizeof(msg)); | ||
| 175 | /* test for lost msgs */ | ||
| 176 | lost_msg = ioread8(&card->dpram[DPRAM_RX_LOST]); | ||
| 177 | if (lost_msg) { | ||
| 178 | int j; | ||
| 179 | /* reset condition */ | ||
| 180 | iowrite8(0, &card->dpram[DPRAM_RX_LOST]); | ||
| 181 | /* prepare msg */ | ||
| 182 | msg.can_id = CAN_ERR_FLAG | CAN_ERR_CRTL; | ||
| 183 | msg.can_dlc = CAN_ERR_DLC; | ||
| 184 | msg.data[1] = CAN_ERR_CRTL_RX_OVERFLOW; | ||
| 185 | /* | ||
| 186 | * service to all busses, we don't know which it was applicable | ||
| 187 | * but only service busses that are online | ||
| 188 | */ | ||
| 189 | for (j = 0; j < ARRAY_SIZE(card->net); ++j) { | ||
| 190 | netdev = card->net[j]; | ||
| 191 | if (!netdev) | ||
| 192 | continue; | ||
| 193 | if (!canif_is_active(netdev)) | ||
| 194 | /* a dead bus has no overflows */ | ||
| 195 | continue; | ||
| 196 | ++netdev->stats.rx_over_errors; | ||
| 197 | softing_netdev_rx(netdev, &msg, ktime_set(0, 0)); | ||
| 198 | } | ||
| 199 | /* prepare for other use */ | ||
| 200 | memset(&msg, 0, sizeof(msg)); | ||
| 201 | ++cnt; | ||
| 202 | } | ||
| 203 | |||
| 204 | fifo_rd = ioread8(&card->dpram[DPRAM_RX_RD]); | ||
| 205 | fifo_wr = ioread8(&card->dpram[DPRAM_RX_WR]); | ||
| 206 | |||
| 207 | if (++fifo_rd >= DPRAM_RX_CNT) | ||
| 208 | fifo_rd = 0; | ||
| 209 | if (fifo_wr == fifo_rd) | ||
| 210 | return cnt; | ||
| 211 | |||
| 212 | memcpy_fromio(buf, &card->dpram[DPRAM_RX + DPRAM_RX_SIZE*fifo_rd], | ||
| 213 | DPRAM_RX_SIZE); | ||
| 214 | mb(); | ||
| 215 | /* trigger dual port RAM */ | ||
| 216 | iowrite8(fifo_rd, &card->dpram[DPRAM_RX_RD]); | ||
| 217 | |||
| 218 | ptr = buf; | ||
| 219 | cmd = *ptr++; | ||
| 220 | if (cmd == 0xff) | ||
| 221 | /* not quite usefull, probably the card has got out */ | ||
| 222 | return 0; | ||
| 223 | netdev = card->net[0]; | ||
| 224 | if (cmd & CMD_BUS2) | ||
| 225 | netdev = card->net[1]; | ||
| 226 | priv = netdev_priv(netdev); | ||
| 227 | |||
| 228 | if (cmd & CMD_ERR) { | ||
| 229 | uint8_t can_state, state; | ||
| 230 | |||
| 231 | state = *ptr++; | ||
| 232 | |||
| 233 | msg.can_id = CAN_ERR_FLAG; | ||
| 234 | msg.can_dlc = CAN_ERR_DLC; | ||
| 235 | |||
| 236 | if (state & SF_MASK_BUSOFF) { | ||
| 237 | can_state = CAN_STATE_BUS_OFF; | ||
| 238 | msg.can_id |= CAN_ERR_BUSOFF; | ||
| 239 | state = STATE_BUSOFF; | ||
| 240 | } else if (state & SF_MASK_EPASSIVE) { | ||
| 241 | can_state = CAN_STATE_ERROR_PASSIVE; | ||
| 242 | msg.can_id |= CAN_ERR_CRTL; | ||
| 243 | msg.data[1] = CAN_ERR_CRTL_TX_PASSIVE; | ||
| 244 | state = STATE_EPASSIVE; | ||
| 245 | } else { | ||
| 246 | can_state = CAN_STATE_ERROR_ACTIVE; | ||
| 247 | msg.can_id |= CAN_ERR_CRTL; | ||
| 248 | state = STATE_EACTIVE; | ||
| 249 | } | ||
| 250 | /* update DPRAM */ | ||
| 251 | iowrite8(state, &card->dpram[priv->index ? | ||
| 252 | DPRAM_INFO_BUSSTATE2 : DPRAM_INFO_BUSSTATE]); | ||
| 253 | /* timestamp */ | ||
| 254 | tmp_u32 = le32_to_cpup((void *)ptr); | ||
| 255 | ptr += 4; | ||
| 256 | ktime = softing_raw2ktime(card, tmp_u32); | ||
| 257 | |||
| 258 | ++netdev->stats.rx_errors; | ||
| 259 | /* update internal status */ | ||
| 260 | if (can_state != priv->can.state) { | ||
| 261 | priv->can.state = can_state; | ||
| 262 | if (can_state == CAN_STATE_ERROR_PASSIVE) | ||
| 263 | ++priv->can.can_stats.error_passive; | ||
| 264 | else if (can_state == CAN_STATE_BUS_OFF) { | ||
| 265 | /* this calls can_close_cleanup() */ | ||
| 266 | can_bus_off(netdev); | ||
| 267 | netif_stop_queue(netdev); | ||
| 268 | } | ||
| 269 | /* trigger socketcan */ | ||
| 270 | softing_netdev_rx(netdev, &msg, ktime); | ||
| 271 | } | ||
| 272 | |||
| 273 | } else { | ||
| 274 | if (cmd & CMD_RTR) | ||
| 275 | msg.can_id |= CAN_RTR_FLAG; | ||
| 276 | msg.can_dlc = get_can_dlc(*ptr++); | ||
| 277 | if (cmd & CMD_XTD) { | ||
| 278 | msg.can_id |= CAN_EFF_FLAG; | ||
| 279 | msg.can_id |= le32_to_cpup((void *)ptr); | ||
| 280 | ptr += 4; | ||
| 281 | } else { | ||
| 282 | msg.can_id |= le16_to_cpup((void *)ptr); | ||
| 283 | ptr += 2; | ||
| 284 | } | ||
| 285 | /* timestamp */ | ||
| 286 | tmp_u32 = le32_to_cpup((void *)ptr); | ||
| 287 | ptr += 4; | ||
| 288 | ktime = softing_raw2ktime(card, tmp_u32); | ||
| 289 | if (!(msg.can_id & CAN_RTR_FLAG)) | ||
| 290 | memcpy(&msg.data[0], ptr, 8); | ||
| 291 | ptr += 8; | ||
| 292 | /* update socket */ | ||
| 293 | if (cmd & CMD_ACK) { | ||
| 294 | /* acknowledge, was tx msg */ | ||
| 295 | struct sk_buff *skb; | ||
| 296 | skb = priv->can.echo_skb[priv->tx.echo_get]; | ||
| 297 | if (skb) | ||
| 298 | skb->tstamp = ktime; | ||
| 299 | can_get_echo_skb(netdev, priv->tx.echo_get); | ||
| 300 | ++priv->tx.echo_get; | ||
| 301 | if (priv->tx.echo_get >= TX_ECHO_SKB_MAX) | ||
| 302 | priv->tx.echo_get = 0; | ||
| 303 | if (priv->tx.pending) | ||
| 304 | --priv->tx.pending; | ||
| 305 | if (card->tx.pending) | ||
| 306 | --card->tx.pending; | ||
| 307 | ++netdev->stats.tx_packets; | ||
| 308 | if (!(msg.can_id & CAN_RTR_FLAG)) | ||
| 309 | netdev->stats.tx_bytes += msg.can_dlc; | ||
| 310 | } else { | ||
| 311 | int ret; | ||
| 312 | |||
| 313 | ret = softing_netdev_rx(netdev, &msg, ktime); | ||
| 314 | if (ret == NET_RX_SUCCESS) { | ||
| 315 | ++netdev->stats.rx_packets; | ||
| 316 | if (!(msg.can_id & CAN_RTR_FLAG)) | ||
| 317 | netdev->stats.rx_bytes += msg.can_dlc; | ||
| 318 | } else { | ||
| 319 | ++netdev->stats.rx_dropped; | ||
| 320 | } | ||
| 321 | } | ||
| 322 | } | ||
| 323 | ++cnt; | ||
| 324 | return cnt; | ||
| 325 | } | ||
| 326 | |||
| 327 | /* | ||
| 328 | * real interrupt handler | ||
| 329 | */ | ||
| 330 | static irqreturn_t softing_irq_thread(int irq, void *dev_id) | ||
| 331 | { | ||
| 332 | struct softing *card = (struct softing *)dev_id; | ||
| 333 | struct net_device *netdev; | ||
| 334 | struct softing_priv *priv; | ||
| 335 | int j, offset, work_done; | ||
| 336 | |||
| 337 | work_done = 0; | ||
| 338 | spin_lock_bh(&card->spin); | ||
| 339 | while (softing_handle_1(card) > 0) { | ||
| 340 | ++card->irq.svc_count; | ||
| 341 | ++work_done; | ||
| 342 | } | ||
| 343 | spin_unlock_bh(&card->spin); | ||
| 344 | /* resume tx queue's */ | ||
| 345 | offset = card->tx.last_bus; | ||
| 346 | for (j = 0; j < ARRAY_SIZE(card->net); ++j) { | ||
| 347 | if (card->tx.pending >= TXMAX) | ||
| 348 | break; | ||
| 349 | netdev = card->net[(j + offset + 1) % card->pdat->nbus]; | ||
| 350 | if (!netdev) | ||
| 351 | continue; | ||
| 352 | priv = netdev_priv(netdev); | ||
| 353 | if (!canif_is_active(netdev)) | ||
| 354 | /* it makes no sense to wake dead busses */ | ||
| 355 | continue; | ||
| 356 | if (priv->tx.pending >= TX_ECHO_SKB_MAX) | ||
| 357 | continue; | ||
| 358 | ++work_done; | ||
| 359 | netif_wake_queue(netdev); | ||
| 360 | } | ||
| 361 | return work_done ? IRQ_HANDLED : IRQ_NONE; | ||
| 362 | } | ||
| 363 | |||
| 364 | /* | ||
| 365 | * interrupt routines: | ||
| 366 | * schedule the 'real interrupt handler' | ||
| 367 | */ | ||
| 368 | static irqreturn_t softing_irq_v2(int irq, void *dev_id) | ||
| 369 | { | ||
| 370 | struct softing *card = (struct softing *)dev_id; | ||
| 371 | uint8_t ir; | ||
| 372 | |||
| 373 | ir = ioread8(&card->dpram[DPRAM_V2_IRQ_TOHOST]); | ||
| 374 | iowrite8(0, &card->dpram[DPRAM_V2_IRQ_TOHOST]); | ||
| 375 | return (1 == ir) ? IRQ_WAKE_THREAD : IRQ_NONE; | ||
| 376 | } | ||
| 377 | |||
| 378 | static irqreturn_t softing_irq_v1(int irq, void *dev_id) | ||
| 379 | { | ||
| 380 | struct softing *card = (struct softing *)dev_id; | ||
| 381 | uint8_t ir; | ||
| 382 | |||
| 383 | ir = ioread8(&card->dpram[DPRAM_IRQ_TOHOST]); | ||
| 384 | iowrite8(0, &card->dpram[DPRAM_IRQ_TOHOST]); | ||
| 385 | return ir ? IRQ_WAKE_THREAD : IRQ_NONE; | ||
| 386 | } | ||
| 387 | |||
| 388 | /* | ||
| 389 | * netdev/candev inter-operability | ||
| 390 | */ | ||
| 391 | static int softing_netdev_open(struct net_device *ndev) | ||
| 392 | { | ||
| 393 | int ret; | ||
| 394 | |||
| 395 | /* check or determine and set bittime */ | ||
| 396 | ret = open_candev(ndev); | ||
| 397 | if (!ret) | ||
| 398 | ret = softing_startstop(ndev, 1); | ||
| 399 | return ret; | ||
| 400 | } | ||
| 401 | |||
| 402 | static int softing_netdev_stop(struct net_device *ndev) | ||
| 403 | { | ||
| 404 | int ret; | ||
| 405 | |||
| 406 | netif_stop_queue(ndev); | ||
| 407 | |||
| 408 | /* softing cycle does close_candev() */ | ||
| 409 | ret = softing_startstop(ndev, 0); | ||
| 410 | return ret; | ||
| 411 | } | ||
| 412 | |||
| 413 | static int softing_candev_set_mode(struct net_device *ndev, enum can_mode mode) | ||
| 414 | { | ||
| 415 | int ret; | ||
| 416 | |||
| 417 | switch (mode) { | ||
| 418 | case CAN_MODE_START: | ||
| 419 | /* softing_startstop does close_candev() */ | ||
| 420 | ret = softing_startstop(ndev, 1); | ||
| 421 | return ret; | ||
| 422 | case CAN_MODE_STOP: | ||
| 423 | case CAN_MODE_SLEEP: | ||
| 424 | return -EOPNOTSUPP; | ||
| 425 | } | ||
| 426 | return 0; | ||
| 427 | } | ||
| 428 | |||
| 429 | /* | ||
| 430 | * Softing device management helpers | ||
| 431 | */ | ||
| 432 | int softing_enable_irq(struct softing *card, int enable) | ||
| 433 | { | ||
| 434 | int ret; | ||
| 435 | |||
| 436 | if (!card->irq.nr) { | ||
| 437 | return 0; | ||
| 438 | } else if (card->irq.requested && !enable) { | ||
| 439 | free_irq(card->irq.nr, card); | ||
| 440 | card->irq.requested = 0; | ||
| 441 | } else if (!card->irq.requested && enable) { | ||
| 442 | ret = request_threaded_irq(card->irq.nr, | ||
| 443 | (card->pdat->generation >= 2) ? | ||
| 444 | softing_irq_v2 : softing_irq_v1, | ||
| 445 | softing_irq_thread, IRQF_SHARED, | ||
| 446 | dev_name(&card->pdev->dev), card); | ||
| 447 | if (ret) { | ||
| 448 | dev_alert(&card->pdev->dev, | ||
| 449 | "request_threaded_irq(%u) failed\n", | ||
| 450 | card->irq.nr); | ||
| 451 | return ret; | ||
| 452 | } | ||
| 453 | card->irq.requested = 1; | ||
| 454 | } | ||
| 455 | return 0; | ||
| 456 | } | ||
| 457 | |||
| 458 | static void softing_card_shutdown(struct softing *card) | ||
| 459 | { | ||
| 460 | int fw_up = 0; | ||
| 461 | |||
| 462 | if (mutex_lock_interruptible(&card->fw.lock)) | ||
| 463 | /* return -ERESTARTSYS */; | ||
| 464 | fw_up = card->fw.up; | ||
| 465 | card->fw.up = 0; | ||
| 466 | |||
| 467 | if (card->irq.requested && card->irq.nr) { | ||
| 468 | free_irq(card->irq.nr, card); | ||
| 469 | card->irq.requested = 0; | ||
| 470 | } | ||
| 471 | if (fw_up) { | ||
| 472 | if (card->pdat->enable_irq) | ||
| 473 | card->pdat->enable_irq(card->pdev, 0); | ||
| 474 | softing_set_reset_dpram(card); | ||
| 475 | if (card->pdat->reset) | ||
| 476 | card->pdat->reset(card->pdev, 1); | ||
| 477 | } | ||
| 478 | mutex_unlock(&card->fw.lock); | ||
| 479 | } | ||
| 480 | |||
| 481 | static __devinit int softing_card_boot(struct softing *card) | ||
| 482 | { | ||
| 483 | int ret, j; | ||
| 484 | static const uint8_t stream[] = { | ||
| 485 | 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, }; | ||
| 486 | unsigned char back[sizeof(stream)]; | ||
| 487 | |||
| 488 | if (mutex_lock_interruptible(&card->fw.lock)) | ||
| 489 | return -ERESTARTSYS; | ||
| 490 | if (card->fw.up) { | ||
| 491 | mutex_unlock(&card->fw.lock); | ||
| 492 | return 0; | ||
| 493 | } | ||
| 494 | /* reset board */ | ||
| 495 | if (card->pdat->enable_irq) | ||
| 496 | card->pdat->enable_irq(card->pdev, 1); | ||
| 497 | /* boot card */ | ||
| 498 | softing_set_reset_dpram(card); | ||
| 499 | if (card->pdat->reset) | ||
| 500 | card->pdat->reset(card->pdev, 1); | ||
| 501 | for (j = 0; (j + sizeof(stream)) < card->dpram_size; | ||
| 502 | j += sizeof(stream)) { | ||
| 503 | |||
| 504 | memcpy_toio(&card->dpram[j], stream, sizeof(stream)); | ||
| 505 | /* flush IO cache */ | ||
| 506 | mb(); | ||
| 507 | memcpy_fromio(back, &card->dpram[j], sizeof(stream)); | ||
| 508 | |||
| 509 | if (!memcmp(back, stream, sizeof(stream))) | ||
| 510 | continue; | ||
| 511 | /* memory is not equal */ | ||
| 512 | dev_alert(&card->pdev->dev, "dpram failed at 0x%04x\n", j); | ||
| 513 | ret = -EIO; | ||
| 514 | goto failed; | ||
| 515 | } | ||
| 516 | wmb(); | ||
| 517 | /* load boot firmware */ | ||
| 518 | ret = softing_load_fw(card->pdat->boot.fw, card, card->dpram, | ||
| 519 | card->dpram_size, | ||
| 520 | card->pdat->boot.offs - card->pdat->boot.addr); | ||
| 521 | if (ret < 0) | ||
| 522 | goto failed; | ||
| 523 | /* load loader firmware */ | ||
| 524 | ret = softing_load_fw(card->pdat->load.fw, card, card->dpram, | ||
| 525 | card->dpram_size, | ||
| 526 | card->pdat->load.offs - card->pdat->load.addr); | ||
| 527 | if (ret < 0) | ||
| 528 | goto failed; | ||
| 529 | |||
| 530 | if (card->pdat->reset) | ||
| 531 | card->pdat->reset(card->pdev, 0); | ||
| 532 | softing_clr_reset_dpram(card); | ||
| 533 | ret = softing_bootloader_command(card, 0, "card boot"); | ||
| 534 | if (ret < 0) | ||
| 535 | goto failed; | ||
| 536 | ret = softing_load_app_fw(card->pdat->app.fw, card); | ||
| 537 | if (ret < 0) | ||
| 538 | goto failed; | ||
| 539 | |||
| 540 | ret = softing_chip_poweron(card); | ||
| 541 | if (ret < 0) | ||
| 542 | goto failed; | ||
| 543 | |||
| 544 | card->fw.up = 1; | ||
| 545 | mutex_unlock(&card->fw.lock); | ||
| 546 | return 0; | ||
| 547 | failed: | ||
| 548 | card->fw.up = 0; | ||
| 549 | if (card->pdat->enable_irq) | ||
| 550 | card->pdat->enable_irq(card->pdev, 0); | ||
| 551 | softing_set_reset_dpram(card); | ||
| 552 | if (card->pdat->reset) | ||
| 553 | card->pdat->reset(card->pdev, 1); | ||
| 554 | mutex_unlock(&card->fw.lock); | ||
| 555 | return ret; | ||
| 556 | } | ||
| 557 | |||
| 558 | /* | ||
| 559 | * netdev sysfs | ||
| 560 | */ | ||
| 561 | static ssize_t show_channel(struct device *dev, struct device_attribute *attr, | ||
| 562 | char *buf) | ||
| 563 | { | ||
| 564 | struct net_device *ndev = to_net_dev(dev); | ||
| 565 | struct softing_priv *priv = netdev2softing(ndev); | ||
| 566 | |||
| 567 | return sprintf(buf, "%i\n", priv->index); | ||
| 568 | } | ||
| 569 | |||
| 570 | static ssize_t show_chip(struct device *dev, struct device_attribute *attr, | ||
| 571 | char *buf) | ||
| 572 | { | ||
| 573 | struct net_device *ndev = to_net_dev(dev); | ||
| 574 | struct softing_priv *priv = netdev2softing(ndev); | ||
| 575 | |||
| 576 | return sprintf(buf, "%i\n", priv->chip); | ||
| 577 | } | ||
| 578 | |||
| 579 | static ssize_t show_output(struct device *dev, struct device_attribute *attr, | ||
| 580 | char *buf) | ||
| 581 | { | ||
| 582 | struct net_device *ndev = to_net_dev(dev); | ||
| 583 | struct softing_priv *priv = netdev2softing(ndev); | ||
| 584 | |||
| 585 | return sprintf(buf, "0x%02x\n", priv->output); | ||
| 586 | } | ||
| 587 | |||
| 588 | static ssize_t store_output(struct device *dev, struct device_attribute *attr, | ||
| 589 | const char *buf, size_t count) | ||
| 590 | { | ||
| 591 | struct net_device *ndev = to_net_dev(dev); | ||
| 592 | struct softing_priv *priv = netdev2softing(ndev); | ||
| 593 | struct softing *card = priv->card; | ||
| 594 | unsigned long val; | ||
| 595 | int ret; | ||
| 596 | |||
| 597 | ret = strict_strtoul(buf, 0, &val); | ||
| 598 | if (ret < 0) | ||
| 599 | return ret; | ||
| 600 | val &= 0xFF; | ||
| 601 | |||
| 602 | ret = mutex_lock_interruptible(&card->fw.lock); | ||
| 603 | if (ret) | ||
| 604 | return -ERESTARTSYS; | ||
| 605 | if (netif_running(ndev)) { | ||
| 606 | mutex_unlock(&card->fw.lock); | ||
| 607 | return -EBUSY; | ||
| 608 | } | ||
| 609 | priv->output = val; | ||
| 610 | mutex_unlock(&card->fw.lock); | ||
| 611 | return count; | ||
| 612 | } | ||
| 613 | |||
| 614 | static const DEVICE_ATTR(channel, S_IRUGO, show_channel, NULL); | ||
| 615 | static const DEVICE_ATTR(chip, S_IRUGO, show_chip, NULL); | ||
| 616 | static const DEVICE_ATTR(output, S_IRUGO | S_IWUSR, show_output, store_output); | ||
| 617 | |||
| 618 | static const struct attribute *const netdev_sysfs_attrs[] = { | ||
| 619 | &dev_attr_channel.attr, | ||
| 620 | &dev_attr_chip.attr, | ||
| 621 | &dev_attr_output.attr, | ||
| 622 | NULL, | ||
| 623 | }; | ||
| 624 | static const struct attribute_group netdev_sysfs_group = { | ||
| 625 | .name = NULL, | ||
| 626 | .attrs = (struct attribute **)netdev_sysfs_attrs, | ||
| 627 | }; | ||
| 628 | |||
| 629 | static const struct net_device_ops softing_netdev_ops = { | ||
| 630 | .ndo_open = softing_netdev_open, | ||
| 631 | .ndo_stop = softing_netdev_stop, | ||
| 632 | .ndo_start_xmit = softing_netdev_start_xmit, | ||
| 633 | }; | ||
| 634 | |||
| 635 | static const struct can_bittiming_const softing_btr_const = { | ||
| 636 | .tseg1_min = 1, | ||
| 637 | .tseg1_max = 16, | ||
| 638 | .tseg2_min = 1, | ||
| 639 | .tseg2_max = 8, | ||
| 640 | .sjw_max = 4, /* overruled */ | ||
| 641 | .brp_min = 1, | ||
| 642 | .brp_max = 32, /* overruled */ | ||
| 643 | .brp_inc = 1, | ||
| 644 | }; | ||
| 645 | |||
| 646 | |||
| 647 | static __devinit struct net_device *softing_netdev_create(struct softing *card, | ||
| 648 | uint16_t chip_id) | ||
| 649 | { | ||
| 650 | struct net_device *netdev; | ||
| 651 | struct softing_priv *priv; | ||
| 652 | |||
| 653 | netdev = alloc_candev(sizeof(*priv), TX_ECHO_SKB_MAX); | ||
| 654 | if (!netdev) { | ||
| 655 | dev_alert(&card->pdev->dev, "alloc_candev failed\n"); | ||
| 656 | return NULL; | ||
| 657 | } | ||
| 658 | priv = netdev_priv(netdev); | ||
| 659 | priv->netdev = netdev; | ||
| 660 | priv->card = card; | ||
| 661 | memcpy(&priv->btr_const, &softing_btr_const, sizeof(priv->btr_const)); | ||
| 662 | priv->btr_const.brp_max = card->pdat->max_brp; | ||
| 663 | priv->btr_const.sjw_max = card->pdat->max_sjw; | ||
| 664 | priv->can.bittiming_const = &priv->btr_const; | ||
| 665 | priv->can.clock.freq = 8000000; | ||
| 666 | priv->chip = chip_id; | ||
| 667 | priv->output = softing_default_output(netdev); | ||
| 668 | SET_NETDEV_DEV(netdev, &card->pdev->dev); | ||
| 669 | |||
| 670 | netdev->flags |= IFF_ECHO; | ||
| 671 | netdev->netdev_ops = &softing_netdev_ops; | ||
| 672 | priv->can.do_set_mode = softing_candev_set_mode; | ||
| 673 | priv->can.ctrlmode_supported = CAN_CTRLMODE_3_SAMPLES; | ||
| 674 | |||
| 675 | return netdev; | ||
| 676 | } | ||
| 677 | |||
| 678 | static __devinit int softing_netdev_register(struct net_device *netdev) | ||
| 679 | { | ||
| 680 | int ret; | ||
| 681 | |||
| 682 | netdev->sysfs_groups[0] = &netdev_sysfs_group; | ||
| 683 | ret = register_candev(netdev); | ||
| 684 | if (ret) { | ||
| 685 | dev_alert(&netdev->dev, "register failed\n"); | ||
| 686 | return ret; | ||
| 687 | } | ||
| 688 | return 0; | ||
| 689 | } | ||
| 690 | |||
| 691 | static void softing_netdev_cleanup(struct net_device *netdev) | ||
| 692 | { | ||
| 693 | unregister_candev(netdev); | ||
| 694 | free_candev(netdev); | ||
| 695 | } | ||
| 696 | |||
| 697 | /* | ||
| 698 | * sysfs for Platform device | ||
| 699 | */ | ||
| 700 | #define DEV_ATTR_RO(name, member) \ | ||
| 701 | static ssize_t show_##name(struct device *dev, \ | ||
| 702 | struct device_attribute *attr, char *buf) \ | ||
| 703 | { \ | ||
| 704 | struct softing *card = platform_get_drvdata(to_platform_device(dev)); \ | ||
| 705 | return sprintf(buf, "%u\n", card->member); \ | ||
| 706 | } \ | ||
| 707 | static DEVICE_ATTR(name, 0444, show_##name, NULL) | ||
| 708 | |||
| 709 | #define DEV_ATTR_RO_STR(name, member) \ | ||
| 710 | static ssize_t show_##name(struct device *dev, \ | ||
| 711 | struct device_attribute *attr, char *buf) \ | ||
| 712 | { \ | ||
| 713 | struct softing *card = platform_get_drvdata(to_platform_device(dev)); \ | ||
| 714 | return sprintf(buf, "%s\n", card->member); \ | ||
| 715 | } \ | ||
| 716 | static DEVICE_ATTR(name, 0444, show_##name, NULL) | ||
| 717 | |||
| 718 | DEV_ATTR_RO(serial, id.serial); | ||
| 719 | DEV_ATTR_RO_STR(firmware, pdat->app.fw); | ||
| 720 | DEV_ATTR_RO(firmware_version, id.fw_version); | ||
| 721 | DEV_ATTR_RO_STR(hardware, pdat->name); | ||
| 722 | DEV_ATTR_RO(hardware_version, id.hw_version); | ||
| 723 | DEV_ATTR_RO(license, id.license); | ||
| 724 | DEV_ATTR_RO(frequency, id.freq); | ||
| 725 | DEV_ATTR_RO(txpending, tx.pending); | ||
| 726 | |||
| 727 | static struct attribute *softing_pdev_attrs[] = { | ||
| 728 | &dev_attr_serial.attr, | ||
| 729 | &dev_attr_firmware.attr, | ||
| 730 | &dev_attr_firmware_version.attr, | ||
| 731 | &dev_attr_hardware.attr, | ||
| 732 | &dev_attr_hardware_version.attr, | ||
| 733 | &dev_attr_license.attr, | ||
| 734 | &dev_attr_frequency.attr, | ||
| 735 | &dev_attr_txpending.attr, | ||
| 736 | NULL, | ||
| 737 | }; | ||
| 738 | |||
| 739 | static const struct attribute_group softing_pdev_group = { | ||
| 740 | .name = NULL, | ||
| 741 | .attrs = softing_pdev_attrs, | ||
| 742 | }; | ||
| 743 | |||
| 744 | /* | ||
| 745 | * platform driver | ||
| 746 | */ | ||
| 747 | static __devexit int softing_pdev_remove(struct platform_device *pdev) | ||
| 748 | { | ||
| 749 | struct softing *card = platform_get_drvdata(pdev); | ||
| 750 | int j; | ||
| 751 | |||
| 752 | /* first, disable card*/ | ||
| 753 | softing_card_shutdown(card); | ||
| 754 | |||
| 755 | for (j = 0; j < ARRAY_SIZE(card->net); ++j) { | ||
| 756 | if (!card->net[j]) | ||
| 757 | continue; | ||
| 758 | softing_netdev_cleanup(card->net[j]); | ||
| 759 | card->net[j] = NULL; | ||
| 760 | } | ||
| 761 | sysfs_remove_group(&pdev->dev.kobj, &softing_pdev_group); | ||
| 762 | |||
| 763 | iounmap(card->dpram); | ||
| 764 | kfree(card); | ||
| 765 | return 0; | ||
| 766 | } | ||
| 767 | |||
| 768 | static __devinit int softing_pdev_probe(struct platform_device *pdev) | ||
| 769 | { | ||
| 770 | const struct softing_platform_data *pdat = pdev->dev.platform_data; | ||
| 771 | struct softing *card; | ||
| 772 | struct net_device *netdev; | ||
| 773 | struct softing_priv *priv; | ||
| 774 | struct resource *pres; | ||
| 775 | int ret; | ||
| 776 | int j; | ||
| 777 | |||
| 778 | if (!pdat) { | ||
| 779 | dev_warn(&pdev->dev, "no platform data\n"); | ||
| 780 | return -EINVAL; | ||
| 781 | } | ||
| 782 | if (pdat->nbus > ARRAY_SIZE(card->net)) { | ||
| 783 | dev_warn(&pdev->dev, "%u nets??\n", pdat->nbus); | ||
| 784 | return -EINVAL; | ||
| 785 | } | ||
| 786 | |||
| 787 | card = kzalloc(sizeof(*card), GFP_KERNEL); | ||
| 788 | if (!card) | ||
| 789 | return -ENOMEM; | ||
| 790 | card->pdat = pdat; | ||
| 791 | card->pdev = pdev; | ||
| 792 | platform_set_drvdata(pdev, card); | ||
| 793 | mutex_init(&card->fw.lock); | ||
| 794 | spin_lock_init(&card->spin); | ||
| 795 | |||
| 796 | ret = -EINVAL; | ||
| 797 | pres = platform_get_resource(pdev, IORESOURCE_MEM, 0); | ||
| 798 | if (!pres) | ||
| 799 | goto platform_resource_failed;; | ||
| 800 | card->dpram_phys = pres->start; | ||
| 801 | card->dpram_size = pres->end - pres->start + 1; | ||
| 802 | card->dpram = ioremap_nocache(card->dpram_phys, card->dpram_size); | ||
| 803 | if (!card->dpram) { | ||
| 804 | dev_alert(&card->pdev->dev, "dpram ioremap failed\n"); | ||
| 805 | goto ioremap_failed; | ||
| 806 | } | ||
| 807 | |||
| 808 | pres = platform_get_resource(pdev, IORESOURCE_IRQ, 0); | ||
| 809 | if (pres) | ||
| 810 | card->irq.nr = pres->start; | ||
| 811 | |||
| 812 | /* reset card */ | ||
| 813 | ret = softing_card_boot(card); | ||
| 814 | if (ret < 0) { | ||
| 815 | dev_alert(&pdev->dev, "failed to boot\n"); | ||
| 816 | goto boot_failed; | ||
| 817 | } | ||
| 818 | |||
| 819 | /* only now, the chip's are known */ | ||
| 820 | card->id.freq = card->pdat->freq; | ||
| 821 | |||
| 822 | ret = sysfs_create_group(&pdev->dev.kobj, &softing_pdev_group); | ||
| 823 | if (ret < 0) { | ||
| 824 | dev_alert(&card->pdev->dev, "sysfs failed\n"); | ||
| 825 | goto sysfs_failed; | ||
| 826 | } | ||
| 827 | |||
| 828 | ret = -ENOMEM; | ||
| 829 | for (j = 0; j < ARRAY_SIZE(card->net); ++j) { | ||
| 830 | card->net[j] = netdev = | ||
| 831 | softing_netdev_create(card, card->id.chip[j]); | ||
| 832 | if (!netdev) { | ||
| 833 | dev_alert(&pdev->dev, "failed to make can[%i]", j); | ||
| 834 | goto netdev_failed; | ||
| 835 | } | ||
| 836 | priv = netdev_priv(card->net[j]); | ||
| 837 | priv->index = j; | ||
| 838 | ret = softing_netdev_register(netdev); | ||
| 839 | if (ret) { | ||
| 840 | free_candev(netdev); | ||
| 841 | card->net[j] = NULL; | ||
| 842 | dev_alert(&card->pdev->dev, | ||
| 843 | "failed to register can[%i]\n", j); | ||
| 844 | goto netdev_failed; | ||
| 845 | } | ||
| 846 | } | ||
| 847 | dev_info(&card->pdev->dev, "%s ready.\n", card->pdat->name); | ||
| 848 | return 0; | ||
| 849 | |||
| 850 | netdev_failed: | ||
| 851 | for (j = 0; j < ARRAY_SIZE(card->net); ++j) { | ||
| 852 | if (!card->net[j]) | ||
| 853 | continue; | ||
| 854 | softing_netdev_cleanup(card->net[j]); | ||
| 855 | } | ||
| 856 | sysfs_remove_group(&pdev->dev.kobj, &softing_pdev_group); | ||
| 857 | sysfs_failed: | ||
| 858 | softing_card_shutdown(card); | ||
| 859 | boot_failed: | ||
| 860 | iounmap(card->dpram); | ||
| 861 | ioremap_failed: | ||
| 862 | platform_resource_failed: | ||
| 863 | kfree(card); | ||
| 864 | return ret; | ||
| 865 | } | ||
| 866 | |||
| 867 | static struct platform_driver softing_driver = { | ||
| 868 | .driver = { | ||
| 869 | .name = "softing", | ||
| 870 | .owner = THIS_MODULE, | ||
| 871 | }, | ||
| 872 | .probe = softing_pdev_probe, | ||
| 873 | .remove = __devexit_p(softing_pdev_remove), | ||
| 874 | }; | ||
| 875 | |||
| 876 | MODULE_ALIAS("platform:softing"); | ||
| 877 | |||
| 878 | static int __init softing_start(void) | ||
| 879 | { | ||
| 880 | return platform_driver_register(&softing_driver); | ||
| 881 | } | ||
| 882 | |||
| 883 | static void __exit softing_stop(void) | ||
| 884 | { | ||
| 885 | platform_driver_unregister(&softing_driver); | ||
| 886 | } | ||
| 887 | |||
| 888 | module_init(softing_start); | ||
| 889 | module_exit(softing_stop); | ||
| 890 | |||
| 891 | MODULE_DESCRIPTION("Softing DPRAM CAN driver"); | ||
| 892 | MODULE_AUTHOR("Kurt Van Dijck <kurt.van.dijck@eia.be>"); | ||
| 893 | MODULE_LICENSE("GPL v2"); | ||
diff --git a/drivers/net/can/softing/softing_platform.h b/drivers/net/can/softing/softing_platform.h new file mode 100644 index 00000000000..ebbf6981562 --- /dev/null +++ b/drivers/net/can/softing/softing_platform.h | |||
| @@ -0,0 +1,40 @@ | |||
| 1 | |||
| 2 | #include <linux/platform_device.h> | ||
| 3 | |||
| 4 | #ifndef _SOFTING_DEVICE_H_ | ||
| 5 | #define _SOFTING_DEVICE_H_ | ||
| 6 | |||
| 7 | /* softing firmware directory prefix */ | ||
| 8 | #define fw_dir "softing-4.6/" | ||
| 9 | |||
| 10 | struct softing_platform_data { | ||
| 11 | unsigned int manf; | ||
| 12 | unsigned int prod; | ||
| 13 | /* | ||
| 14 | * generation | ||
| 15 | * 1st with NEC or SJA1000 | ||
| 16 | * 8bit, exclusive interrupt, ... | ||
| 17 | * 2nd only SJA1000 | ||
| 18 | * 16bit, shared interrupt | ||
| 19 | */ | ||
| 20 | int generation; | ||
| 21 | int nbus; /* # busses on device */ | ||
| 22 | unsigned int freq; /* operating frequency in Hz */ | ||
| 23 | unsigned int max_brp; | ||
| 24 | unsigned int max_sjw; | ||
| 25 | unsigned long dpram_size; | ||
| 26 | const char *name; | ||
| 27 | struct { | ||
| 28 | unsigned long offs; | ||
| 29 | unsigned long addr; | ||
| 30 | const char *fw; | ||
| 31 | } boot, load, app; | ||
| 32 | /* | ||
| 33 | * reset() function | ||
| 34 | * bring pdev in or out of reset, depending on value | ||
| 35 | */ | ||
| 36 | int (*reset)(struct platform_device *pdev, int value); | ||
| 37 | int (*enable_irq)(struct platform_device *pdev, int value); | ||
| 38 | }; | ||
| 39 | |||
| 40 | #endif | ||
diff --git a/drivers/net/cnic.c b/drivers/net/cnic.c index 263a2944566..7ff170cbc7d 100644 --- a/drivers/net/cnic.c +++ b/drivers/net/cnic.c | |||
| @@ -699,13 +699,13 @@ static void cnic_free_dma(struct cnic_dev *dev, struct cnic_dma *dma) | |||
| 699 | static void cnic_setup_page_tbl(struct cnic_dev *dev, struct cnic_dma *dma) | 699 | static void cnic_setup_page_tbl(struct cnic_dev *dev, struct cnic_dma *dma) |
| 700 | { | 700 | { |
| 701 | int i; | 701 | int i; |
| 702 | u32 *page_table = dma->pgtbl; | 702 | __le32 *page_table = (__le32 *) dma->pgtbl; |
| 703 | 703 | ||
| 704 | for (i = 0; i < dma->num_pages; i++) { | 704 | for (i = 0; i < dma->num_pages; i++) { |
| 705 | /* Each entry needs to be in big endian format. */ | 705 | /* Each entry needs to be in big endian format. */ |
| 706 | *page_table = (u32) ((u64) dma->pg_map_arr[i] >> 32); | 706 | *page_table = cpu_to_le32((u64) dma->pg_map_arr[i] >> 32); |
| 707 | page_table++; | 707 | page_table++; |
| 708 | *page_table = (u32) dma->pg_map_arr[i]; | 708 | *page_table = cpu_to_le32(dma->pg_map_arr[i] & 0xffffffff); |
| 709 | page_table++; | 709 | page_table++; |
| 710 | } | 710 | } |
| 711 | } | 711 | } |
| @@ -713,13 +713,13 @@ static void cnic_setup_page_tbl(struct cnic_dev *dev, struct cnic_dma *dma) | |||
| 713 | static void cnic_setup_page_tbl_le(struct cnic_dev *dev, struct cnic_dma *dma) | 713 | static void cnic_setup_page_tbl_le(struct cnic_dev *dev, struct cnic_dma *dma) |
| 714 | { | 714 | { |
| 715 | int i; | 715 | int i; |
| 716 | u32 *page_table = dma->pgtbl; | 716 | __le32 *page_table = (__le32 *) dma->pgtbl; |
| 717 | 717 | ||
| 718 | for (i = 0; i < dma->num_pages; i++) { | 718 | for (i = 0; i < dma->num_pages; i++) { |
| 719 | /* Each entry needs to be in little endian format. */ | 719 | /* Each entry needs to be in little endian format. */ |
| 720 | *page_table = dma->pg_map_arr[i] & 0xffffffff; | 720 | *page_table = cpu_to_le32(dma->pg_map_arr[i] & 0xffffffff); |
| 721 | page_table++; | 721 | page_table++; |
| 722 | *page_table = (u32) ((u64) dma->pg_map_arr[i] >> 32); | 722 | *page_table = cpu_to_le32((u64) dma->pg_map_arr[i] >> 32); |
| 723 | page_table++; | 723 | page_table++; |
| 724 | } | 724 | } |
| 725 | } | 725 | } |
diff --git a/drivers/net/cxgb4/cxgb4_main.c b/drivers/net/cxgb4/cxgb4_main.c index 059c1eec8c3..ec35d458102 100644 --- a/drivers/net/cxgb4/cxgb4_main.c +++ b/drivers/net/cxgb4/cxgb4_main.c | |||
| @@ -2710,6 +2710,8 @@ static int cxgb_open(struct net_device *dev) | |||
| 2710 | struct port_info *pi = netdev_priv(dev); | 2710 | struct port_info *pi = netdev_priv(dev); |
| 2711 | struct adapter *adapter = pi->adapter; | 2711 | struct adapter *adapter = pi->adapter; |
| 2712 | 2712 | ||
| 2713 | netif_carrier_off(dev); | ||
| 2714 | |||
| 2713 | if (!(adapter->flags & FULL_INIT_DONE)) { | 2715 | if (!(adapter->flags & FULL_INIT_DONE)) { |
| 2714 | err = cxgb_up(adapter); | 2716 | err = cxgb_up(adapter); |
| 2715 | if (err < 0) | 2717 | if (err < 0) |
| @@ -3661,7 +3663,6 @@ static int __devinit init_one(struct pci_dev *pdev, | |||
| 3661 | pi->xact_addr_filt = -1; | 3663 | pi->xact_addr_filt = -1; |
| 3662 | pi->rx_offload = RX_CSO; | 3664 | pi->rx_offload = RX_CSO; |
| 3663 | pi->port_id = i; | 3665 | pi->port_id = i; |
| 3664 | netif_carrier_off(netdev); | ||
| 3665 | netdev->irq = pdev->irq; | 3666 | netdev->irq = pdev->irq; |
| 3666 | 3667 | ||
| 3667 | netdev->features |= NETIF_F_SG | TSO_FLAGS; | 3668 | netdev->features |= NETIF_F_SG | TSO_FLAGS; |
diff --git a/drivers/net/depca.c b/drivers/net/depca.c index 1b48b68ad4f..8b0084d17c8 100644 --- a/drivers/net/depca.c +++ b/drivers/net/depca.c | |||
| @@ -1094,7 +1094,7 @@ static int depca_rx(struct net_device *dev) | |||
| 1094 | } | 1094 | } |
| 1095 | } | 1095 | } |
| 1096 | /* Change buffer ownership for this last frame, back to the adapter */ | 1096 | /* Change buffer ownership for this last frame, back to the adapter */ |
| 1097 | for (; lp->rx_old != entry; lp->rx_old = (++lp->rx_old) & lp->rxRingMask) { | 1097 | for (; lp->rx_old != entry; lp->rx_old = (lp->rx_old + 1) & lp->rxRingMask) { |
| 1098 | writel(readl(&lp->rx_ring[lp->rx_old].base) | R_OWN, &lp->rx_ring[lp->rx_old].base); | 1098 | writel(readl(&lp->rx_ring[lp->rx_old].base) | R_OWN, &lp->rx_ring[lp->rx_old].base); |
| 1099 | } | 1099 | } |
| 1100 | writel(readl(&lp->rx_ring[entry].base) | R_OWN, &lp->rx_ring[entry].base); | 1100 | writel(readl(&lp->rx_ring[entry].base) | R_OWN, &lp->rx_ring[entry].base); |
| @@ -1103,7 +1103,7 @@ static int depca_rx(struct net_device *dev) | |||
| 1103 | /* | 1103 | /* |
| 1104 | ** Update entry information | 1104 | ** Update entry information |
| 1105 | */ | 1105 | */ |
| 1106 | lp->rx_new = (++lp->rx_new) & lp->rxRingMask; | 1106 | lp->rx_new = (lp->rx_new + 1) & lp->rxRingMask; |
| 1107 | } | 1107 | } |
| 1108 | 1108 | ||
| 1109 | return 0; | 1109 | return 0; |
| @@ -1148,7 +1148,7 @@ static int depca_tx(struct net_device *dev) | |||
| 1148 | } | 1148 | } |
| 1149 | 1149 | ||
| 1150 | /* Update all the pointers */ | 1150 | /* Update all the pointers */ |
| 1151 | lp->tx_old = (++lp->tx_old) & lp->txRingMask; | 1151 | lp->tx_old = (lp->tx_old + 1) & lp->txRingMask; |
| 1152 | } | 1152 | } |
| 1153 | 1153 | ||
| 1154 | return 0; | 1154 | return 0; |
diff --git a/drivers/net/dl2k.c b/drivers/net/dl2k.c index e1a8216ff69..c05db604605 100644 --- a/drivers/net/dl2k.c +++ b/drivers/net/dl2k.c | |||
| @@ -1753,8 +1753,6 @@ rio_close (struct net_device *dev) | |||
| 1753 | 1753 | ||
| 1754 | /* Free all the skbuffs in the queue. */ | 1754 | /* Free all the skbuffs in the queue. */ |
| 1755 | for (i = 0; i < RX_RING_SIZE; i++) { | 1755 | for (i = 0; i < RX_RING_SIZE; i++) { |
| 1756 | np->rx_ring[i].status = 0; | ||
| 1757 | np->rx_ring[i].fraginfo = 0; | ||
| 1758 | skb = np->rx_skbuff[i]; | 1756 | skb = np->rx_skbuff[i]; |
| 1759 | if (skb) { | 1757 | if (skb) { |
| 1760 | pci_unmap_single(np->pdev, | 1758 | pci_unmap_single(np->pdev, |
| @@ -1763,6 +1761,8 @@ rio_close (struct net_device *dev) | |||
| 1763 | dev_kfree_skb (skb); | 1761 | dev_kfree_skb (skb); |
| 1764 | np->rx_skbuff[i] = NULL; | 1762 | np->rx_skbuff[i] = NULL; |
| 1765 | } | 1763 | } |
| 1764 | np->rx_ring[i].status = 0; | ||
| 1765 | np->rx_ring[i].fraginfo = 0; | ||
| 1766 | } | 1766 | } |
| 1767 | for (i = 0; i < TX_RING_SIZE; i++) { | 1767 | for (i = 0; i < TX_RING_SIZE; i++) { |
| 1768 | skb = np->tx_skbuff[i]; | 1768 | skb = np->tx_skbuff[i]; |
diff --git a/drivers/net/enc28j60.c b/drivers/net/enc28j60.c index 112c5aa9af7..907b05a1c65 100644 --- a/drivers/net/enc28j60.c +++ b/drivers/net/enc28j60.c | |||
| @@ -812,7 +812,7 @@ static void enc28j60_read_tsv(struct enc28j60_net *priv, u8 tsv[TSV_SIZE]) | |||
| 812 | if (netif_msg_hw(priv)) | 812 | if (netif_msg_hw(priv)) |
| 813 | printk(KERN_DEBUG DRV_NAME ": reading TSV at addr:0x%04x\n", | 813 | printk(KERN_DEBUG DRV_NAME ": reading TSV at addr:0x%04x\n", |
| 814 | endptr + 1); | 814 | endptr + 1); |
| 815 | enc28j60_mem_read(priv, endptr + 1, sizeof(tsv), tsv); | 815 | enc28j60_mem_read(priv, endptr + 1, TSV_SIZE, tsv); |
| 816 | } | 816 | } |
| 817 | 817 | ||
| 818 | static void enc28j60_dump_tsv(struct enc28j60_net *priv, const char *msg, | 818 | static void enc28j60_dump_tsv(struct enc28j60_net *priv, const char *msg, |
diff --git a/drivers/net/mlx4/main.c b/drivers/net/mlx4/main.c index 4ffdc18fcb8..2765a3ce9c2 100644 --- a/drivers/net/mlx4/main.c +++ b/drivers/net/mlx4/main.c | |||
| @@ -1286,6 +1286,21 @@ static DEFINE_PCI_DEVICE_TABLE(mlx4_pci_table) = { | |||
| 1286 | { PCI_VDEVICE(MELLANOX, 0x6764) }, /* MT26468 ConnectX EN 10GigE PCIe gen2*/ | 1286 | { PCI_VDEVICE(MELLANOX, 0x6764) }, /* MT26468 ConnectX EN 10GigE PCIe gen2*/ |
| 1287 | { PCI_VDEVICE(MELLANOX, 0x6746) }, /* MT26438 ConnectX EN 40GigE PCIe gen2 5GT/s */ | 1287 | { PCI_VDEVICE(MELLANOX, 0x6746) }, /* MT26438 ConnectX EN 40GigE PCIe gen2 5GT/s */ |
| 1288 | { PCI_VDEVICE(MELLANOX, 0x676e) }, /* MT26478 ConnectX2 40GigE PCIe gen2 */ | 1288 | { PCI_VDEVICE(MELLANOX, 0x676e) }, /* MT26478 ConnectX2 40GigE PCIe gen2 */ |
| 1289 | { PCI_VDEVICE(MELLANOX, 0x1002) }, /* MT25400 Family [ConnectX-2 Virtual Function] */ | ||
| 1290 | { PCI_VDEVICE(MELLANOX, 0x1003) }, /* MT27500 Family [ConnectX-3] */ | ||
| 1291 | { PCI_VDEVICE(MELLANOX, 0x1004) }, /* MT27500 Family [ConnectX-3 Virtual Function] */ | ||
| 1292 | { PCI_VDEVICE(MELLANOX, 0x1005) }, /* MT27510 Family */ | ||
| 1293 | { PCI_VDEVICE(MELLANOX, 0x1006) }, /* MT27511 Family */ | ||
| 1294 | { PCI_VDEVICE(MELLANOX, 0x1007) }, /* MT27520 Family */ | ||
| 1295 | { PCI_VDEVICE(MELLANOX, 0x1008) }, /* MT27521 Family */ | ||
| 1296 | { PCI_VDEVICE(MELLANOX, 0x1009) }, /* MT27530 Family */ | ||
| 1297 | { PCI_VDEVICE(MELLANOX, 0x100a) }, /* MT27531 Family */ | ||
| 1298 | { PCI_VDEVICE(MELLANOX, 0x100b) }, /* MT27540 Family */ | ||
| 1299 | { PCI_VDEVICE(MELLANOX, 0x100c) }, /* MT27541 Family */ | ||
| 1300 | { PCI_VDEVICE(MELLANOX, 0x100d) }, /* MT27550 Family */ | ||
| 1301 | { PCI_VDEVICE(MELLANOX, 0x100e) }, /* MT27551 Family */ | ||
| 1302 | { PCI_VDEVICE(MELLANOX, 0x100f) }, /* MT27560 Family */ | ||
| 1303 | { PCI_VDEVICE(MELLANOX, 0x1010) }, /* MT27561 Family */ | ||
| 1289 | { 0, } | 1304 | { 0, } |
| 1290 | }; | 1305 | }; |
| 1291 | 1306 | ||
diff --git a/drivers/net/niu.c b/drivers/net/niu.c index 2541321bad8..9fb59d3f9c9 100644 --- a/drivers/net/niu.c +++ b/drivers/net/niu.c | |||
| @@ -4489,6 +4489,9 @@ static int niu_alloc_channels(struct niu *np) | |||
| 4489 | { | 4489 | { |
| 4490 | struct niu_parent *parent = np->parent; | 4490 | struct niu_parent *parent = np->parent; |
| 4491 | int first_rx_channel, first_tx_channel; | 4491 | int first_rx_channel, first_tx_channel; |
| 4492 | int num_rx_rings, num_tx_rings; | ||
| 4493 | struct rx_ring_info *rx_rings; | ||
| 4494 | struct tx_ring_info *tx_rings; | ||
| 4492 | int i, port, err; | 4495 | int i, port, err; |
| 4493 | 4496 | ||
| 4494 | port = np->port; | 4497 | port = np->port; |
| @@ -4498,18 +4501,21 @@ static int niu_alloc_channels(struct niu *np) | |||
| 4498 | first_tx_channel += parent->txchan_per_port[i]; | 4501 | first_tx_channel += parent->txchan_per_port[i]; |
| 4499 | } | 4502 | } |
| 4500 | 4503 | ||
| 4501 | np->num_rx_rings = parent->rxchan_per_port[port]; | 4504 | num_rx_rings = parent->rxchan_per_port[port]; |
| 4502 | np->num_tx_rings = parent->txchan_per_port[port]; | 4505 | num_tx_rings = parent->txchan_per_port[port]; |
| 4503 | 4506 | ||
| 4504 | netif_set_real_num_rx_queues(np->dev, np->num_rx_rings); | 4507 | rx_rings = kcalloc(num_rx_rings, sizeof(struct rx_ring_info), |
| 4505 | netif_set_real_num_tx_queues(np->dev, np->num_tx_rings); | 4508 | GFP_KERNEL); |
| 4506 | |||
| 4507 | np->rx_rings = kcalloc(np->num_rx_rings, sizeof(struct rx_ring_info), | ||
| 4508 | GFP_KERNEL); | ||
| 4509 | err = -ENOMEM; | 4509 | err = -ENOMEM; |
| 4510 | if (!np->rx_rings) | 4510 | if (!rx_rings) |
| 4511 | goto out_err; | 4511 | goto out_err; |
| 4512 | 4512 | ||
| 4513 | np->num_rx_rings = num_rx_rings; | ||
| 4514 | smp_wmb(); | ||
| 4515 | np->rx_rings = rx_rings; | ||
| 4516 | |||
| 4517 | netif_set_real_num_rx_queues(np->dev, num_rx_rings); | ||
| 4518 | |||
| 4513 | for (i = 0; i < np->num_rx_rings; i++) { | 4519 | for (i = 0; i < np->num_rx_rings; i++) { |
| 4514 | struct rx_ring_info *rp = &np->rx_rings[i]; | 4520 | struct rx_ring_info *rp = &np->rx_rings[i]; |
| 4515 | 4521 | ||
| @@ -4538,12 +4544,18 @@ static int niu_alloc_channels(struct niu *np) | |||
| 4538 | return err; | 4544 | return err; |
| 4539 | } | 4545 | } |
| 4540 | 4546 | ||
| 4541 | np->tx_rings = kcalloc(np->num_tx_rings, sizeof(struct tx_ring_info), | 4547 | tx_rings = kcalloc(num_tx_rings, sizeof(struct tx_ring_info), |
| 4542 | GFP_KERNEL); | 4548 | GFP_KERNEL); |
| 4543 | err = -ENOMEM; | 4549 | err = -ENOMEM; |
| 4544 | if (!np->tx_rings) | 4550 | if (!tx_rings) |
| 4545 | goto out_err; | 4551 | goto out_err; |
| 4546 | 4552 | ||
| 4553 | np->num_tx_rings = num_tx_rings; | ||
| 4554 | smp_wmb(); | ||
| 4555 | np->tx_rings = tx_rings; | ||
| 4556 | |||
| 4557 | netif_set_real_num_tx_queues(np->dev, num_tx_rings); | ||
| 4558 | |||
| 4547 | for (i = 0; i < np->num_tx_rings; i++) { | 4559 | for (i = 0; i < np->num_tx_rings; i++) { |
| 4548 | struct tx_ring_info *rp = &np->tx_rings[i]; | 4560 | struct tx_ring_info *rp = &np->tx_rings[i]; |
| 4549 | 4561 | ||
| @@ -6246,11 +6258,17 @@ static void niu_sync_mac_stats(struct niu *np) | |||
| 6246 | static void niu_get_rx_stats(struct niu *np) | 6258 | static void niu_get_rx_stats(struct niu *np) |
| 6247 | { | 6259 | { |
| 6248 | unsigned long pkts, dropped, errors, bytes; | 6260 | unsigned long pkts, dropped, errors, bytes; |
| 6261 | struct rx_ring_info *rx_rings; | ||
| 6249 | int i; | 6262 | int i; |
| 6250 | 6263 | ||
| 6251 | pkts = dropped = errors = bytes = 0; | 6264 | pkts = dropped = errors = bytes = 0; |
| 6265 | |||
| 6266 | rx_rings = ACCESS_ONCE(np->rx_rings); | ||
| 6267 | if (!rx_rings) | ||
| 6268 | goto no_rings; | ||
| 6269 | |||
| 6252 | for (i = 0; i < np->num_rx_rings; i++) { | 6270 | for (i = 0; i < np->num_rx_rings; i++) { |
| 6253 | struct rx_ring_info *rp = &np->rx_rings[i]; | 6271 | struct rx_ring_info *rp = &rx_rings[i]; |
| 6254 | 6272 | ||
| 6255 | niu_sync_rx_discard_stats(np, rp, 0); | 6273 | niu_sync_rx_discard_stats(np, rp, 0); |
| 6256 | 6274 | ||
| @@ -6259,6 +6277,8 @@ static void niu_get_rx_stats(struct niu *np) | |||
| 6259 | dropped += rp->rx_dropped; | 6277 | dropped += rp->rx_dropped; |
| 6260 | errors += rp->rx_errors; | 6278 | errors += rp->rx_errors; |
| 6261 | } | 6279 | } |
| 6280 | |||
| 6281 | no_rings: | ||
| 6262 | np->dev->stats.rx_packets = pkts; | 6282 | np->dev->stats.rx_packets = pkts; |
| 6263 | np->dev->stats.rx_bytes = bytes; | 6283 | np->dev->stats.rx_bytes = bytes; |
| 6264 | np->dev->stats.rx_dropped = dropped; | 6284 | np->dev->stats.rx_dropped = dropped; |
| @@ -6268,16 +6288,24 @@ static void niu_get_rx_stats(struct niu *np) | |||
| 6268 | static void niu_get_tx_stats(struct niu *np) | 6288 | static void niu_get_tx_stats(struct niu *np) |
| 6269 | { | 6289 | { |
| 6270 | unsigned long pkts, errors, bytes; | 6290 | unsigned long pkts, errors, bytes; |
| 6291 | struct tx_ring_info *tx_rings; | ||
| 6271 | int i; | 6292 | int i; |
| 6272 | 6293 | ||
| 6273 | pkts = errors = bytes = 0; | 6294 | pkts = errors = bytes = 0; |
| 6295 | |||
| 6296 | tx_rings = ACCESS_ONCE(np->tx_rings); | ||
| 6297 | if (!tx_rings) | ||
| 6298 | goto no_rings; | ||
| 6299 | |||
| 6274 | for (i = 0; i < np->num_tx_rings; i++) { | 6300 | for (i = 0; i < np->num_tx_rings; i++) { |
| 6275 | struct tx_ring_info *rp = &np->tx_rings[i]; | 6301 | struct tx_ring_info *rp = &tx_rings[i]; |
| 6276 | 6302 | ||
| 6277 | pkts += rp->tx_packets; | 6303 | pkts += rp->tx_packets; |
| 6278 | bytes += rp->tx_bytes; | 6304 | bytes += rp->tx_bytes; |
| 6279 | errors += rp->tx_errors; | 6305 | errors += rp->tx_errors; |
| 6280 | } | 6306 | } |
| 6307 | |||
| 6308 | no_rings: | ||
| 6281 | np->dev->stats.tx_packets = pkts; | 6309 | np->dev->stats.tx_packets = pkts; |
| 6282 | np->dev->stats.tx_bytes = bytes; | 6310 | np->dev->stats.tx_bytes = bytes; |
| 6283 | np->dev->stats.tx_errors = errors; | 6311 | np->dev->stats.tx_errors = errors; |
| @@ -6287,9 +6315,10 @@ static struct net_device_stats *niu_get_stats(struct net_device *dev) | |||
| 6287 | { | 6315 | { |
| 6288 | struct niu *np = netdev_priv(dev); | 6316 | struct niu *np = netdev_priv(dev); |
| 6289 | 6317 | ||
| 6290 | niu_get_rx_stats(np); | 6318 | if (netif_running(dev)) { |
| 6291 | niu_get_tx_stats(np); | 6319 | niu_get_rx_stats(np); |
| 6292 | 6320 | niu_get_tx_stats(np); | |
| 6321 | } | ||
| 6293 | return &dev->stats; | 6322 | return &dev->stats; |
| 6294 | } | 6323 | } |
| 6295 | 6324 | ||
diff --git a/drivers/net/pch_gbe/pch_gbe_main.c b/drivers/net/pch_gbe/pch_gbe_main.c index d7355306a73..1bf12339441 100644 --- a/drivers/net/pch_gbe/pch_gbe_main.c +++ b/drivers/net/pch_gbe/pch_gbe_main.c | |||
| @@ -2247,7 +2247,7 @@ static void pch_gbe_remove(struct pci_dev *pdev) | |||
| 2247 | struct net_device *netdev = pci_get_drvdata(pdev); | 2247 | struct net_device *netdev = pci_get_drvdata(pdev); |
| 2248 | struct pch_gbe_adapter *adapter = netdev_priv(netdev); | 2248 | struct pch_gbe_adapter *adapter = netdev_priv(netdev); |
| 2249 | 2249 | ||
| 2250 | flush_scheduled_work(); | 2250 | cancel_work_sync(&adapter->reset_task); |
| 2251 | unregister_netdev(netdev); | 2251 | unregister_netdev(netdev); |
| 2252 | 2252 | ||
| 2253 | pch_gbe_hal_phy_hw_reset(&adapter->hw); | 2253 | pch_gbe_hal_phy_hw_reset(&adapter->hw); |
diff --git a/drivers/net/pcmcia/axnet_cs.c b/drivers/net/pcmcia/axnet_cs.c index 1f42f6ac855..d3cb7720586 100644 --- a/drivers/net/pcmcia/axnet_cs.c +++ b/drivers/net/pcmcia/axnet_cs.c | |||
| @@ -1488,12 +1488,10 @@ static void ei_rx_overrun(struct net_device *dev) | |||
| 1488 | 1488 | ||
| 1489 | /* | 1489 | /* |
| 1490 | * Wait a full Tx time (1.2ms) + some guard time, NS says 1.6ms total. | 1490 | * Wait a full Tx time (1.2ms) + some guard time, NS says 1.6ms total. |
| 1491 | * Early datasheets said to poll the reset bit, but now they say that | 1491 | * We wait at least 2ms. |
| 1492 | * it "is not a reliable indicator and subsequently should be ignored." | ||
| 1493 | * We wait at least 10ms. | ||
| 1494 | */ | 1492 | */ |
| 1495 | 1493 | ||
| 1496 | mdelay(10); | 1494 | mdelay(2); |
| 1497 | 1495 | ||
| 1498 | /* | 1496 | /* |
| 1499 | * Reset RBCR[01] back to zero as per magic incantation. | 1497 | * Reset RBCR[01] back to zero as per magic incantation. |
diff --git a/drivers/net/r8169.c b/drivers/net/r8169.c index bde7d61f193..59ccf0c5c61 100644 --- a/drivers/net/r8169.c +++ b/drivers/net/r8169.c | |||
| @@ -973,7 +973,8 @@ static void __rtl8169_check_link_status(struct net_device *dev, | |||
| 973 | if (pm) | 973 | if (pm) |
| 974 | pm_request_resume(&tp->pci_dev->dev); | 974 | pm_request_resume(&tp->pci_dev->dev); |
| 975 | netif_carrier_on(dev); | 975 | netif_carrier_on(dev); |
| 976 | netif_info(tp, ifup, dev, "link up\n"); | 976 | if (net_ratelimit()) |
| 977 | netif_info(tp, ifup, dev, "link up\n"); | ||
| 977 | } else { | 978 | } else { |
| 978 | netif_carrier_off(dev); | 979 | netif_carrier_off(dev); |
| 979 | netif_info(tp, ifdown, dev, "link down\n"); | 980 | netif_info(tp, ifdown, dev, "link down\n"); |
| @@ -3757,7 +3758,8 @@ static void rtl_hw_start_8168(struct net_device *dev) | |||
| 3757 | RTL_W16(IntrMitigate, 0x5151); | 3758 | RTL_W16(IntrMitigate, 0x5151); |
| 3758 | 3759 | ||
| 3759 | /* Work around for RxFIFO overflow. */ | 3760 | /* Work around for RxFIFO overflow. */ |
| 3760 | if (tp->mac_version == RTL_GIGA_MAC_VER_11) { | 3761 | if (tp->mac_version == RTL_GIGA_MAC_VER_11 || |
| 3762 | tp->mac_version == RTL_GIGA_MAC_VER_22) { | ||
| 3761 | tp->intr_event |= RxFIFOOver | PCSTimeout; | 3763 | tp->intr_event |= RxFIFOOver | PCSTimeout; |
| 3762 | tp->intr_event &= ~RxOverflow; | 3764 | tp->intr_event &= ~RxOverflow; |
| 3763 | } | 3765 | } |
| @@ -4639,12 +4641,33 @@ static irqreturn_t rtl8169_interrupt(int irq, void *dev_instance) | |||
| 4639 | break; | 4641 | break; |
| 4640 | } | 4642 | } |
| 4641 | 4643 | ||
| 4642 | /* Work around for rx fifo overflow */ | 4644 | if (unlikely(status & RxFIFOOver)) { |
| 4643 | if (unlikely(status & RxFIFOOver) && | 4645 | switch (tp->mac_version) { |
| 4644 | (tp->mac_version == RTL_GIGA_MAC_VER_11)) { | 4646 | /* Work around for rx fifo overflow */ |
| 4645 | netif_stop_queue(dev); | 4647 | case RTL_GIGA_MAC_VER_11: |
| 4646 | rtl8169_tx_timeout(dev); | 4648 | case RTL_GIGA_MAC_VER_22: |
| 4647 | break; | 4649 | case RTL_GIGA_MAC_VER_26: |
| 4650 | netif_stop_queue(dev); | ||
| 4651 | rtl8169_tx_timeout(dev); | ||
| 4652 | goto done; | ||
| 4653 | /* Testers needed. */ | ||
| 4654 | case RTL_GIGA_MAC_VER_17: | ||
| 4655 | case RTL_GIGA_MAC_VER_19: | ||
| 4656 | case RTL_GIGA_MAC_VER_20: | ||
| 4657 | case RTL_GIGA_MAC_VER_21: | ||
| 4658 | case RTL_GIGA_MAC_VER_23: | ||
| 4659 | case RTL_GIGA_MAC_VER_24: | ||
| 4660 | case RTL_GIGA_MAC_VER_27: | ||
| 4661 | case RTL_GIGA_MAC_VER_28: | ||
| 4662 | /* Experimental science. Pktgen proof. */ | ||
| 4663 | case RTL_GIGA_MAC_VER_12: | ||
| 4664 | case RTL_GIGA_MAC_VER_25: | ||
| 4665 | if (status == RxFIFOOver) | ||
| 4666 | goto done; | ||
| 4667 | break; | ||
| 4668 | default: | ||
| 4669 | break; | ||
| 4670 | } | ||
| 4648 | } | 4671 | } |
| 4649 | 4672 | ||
| 4650 | if (unlikely(status & SYSErr)) { | 4673 | if (unlikely(status & SYSErr)) { |
| @@ -4680,7 +4703,7 @@ static irqreturn_t rtl8169_interrupt(int irq, void *dev_instance) | |||
| 4680 | (status & RxFIFOOver) ? (status | RxOverflow) : status); | 4703 | (status & RxFIFOOver) ? (status | RxOverflow) : status); |
| 4681 | status = RTL_R16(IntrStatus); | 4704 | status = RTL_R16(IntrStatus); |
| 4682 | } | 4705 | } |
| 4683 | 4706 | done: | |
| 4684 | return IRQ_RETVAL(handled); | 4707 | return IRQ_RETVAL(handled); |
| 4685 | } | 4708 | } |
| 4686 | 4709 | ||
diff --git a/drivers/net/tg3.c b/drivers/net/tg3.c index 7841a8f6999..93b32d36661 100644 --- a/drivers/net/tg3.c +++ b/drivers/net/tg3.c | |||
| @@ -60,12 +60,6 @@ | |||
| 60 | #define BAR_0 0 | 60 | #define BAR_0 0 |
| 61 | #define BAR_2 2 | 61 | #define BAR_2 2 |
| 62 | 62 | ||
| 63 | #if defined(CONFIG_VLAN_8021Q) || defined(CONFIG_VLAN_8021Q_MODULE) | ||
| 64 | #define TG3_VLAN_TAG_USED 1 | ||
| 65 | #else | ||
| 66 | #define TG3_VLAN_TAG_USED 0 | ||
| 67 | #endif | ||
| 68 | |||
| 69 | #include "tg3.h" | 63 | #include "tg3.h" |
| 70 | 64 | ||
| 71 | #define DRV_MODULE_NAME "tg3" | 65 | #define DRV_MODULE_NAME "tg3" |
| @@ -134,9 +128,6 @@ | |||
| 134 | TG3_TX_RING_SIZE) | 128 | TG3_TX_RING_SIZE) |
| 135 | #define NEXT_TX(N) (((N) + 1) & (TG3_TX_RING_SIZE - 1)) | 129 | #define NEXT_TX(N) (((N) + 1) & (TG3_TX_RING_SIZE - 1)) |
| 136 | 130 | ||
| 137 | #define TG3_RX_DMA_ALIGN 16 | ||
| 138 | #define TG3_RX_HEADROOM ALIGN(VLAN_HLEN, TG3_RX_DMA_ALIGN) | ||
| 139 | |||
| 140 | #define TG3_DMA_BYTE_ENAB 64 | 131 | #define TG3_DMA_BYTE_ENAB 64 |
| 141 | 132 | ||
| 142 | #define TG3_RX_STD_DMA_SZ 1536 | 133 | #define TG3_RX_STD_DMA_SZ 1536 |
| @@ -4722,8 +4713,6 @@ static int tg3_rx(struct tg3_napi *tnapi, int budget) | |||
| 4722 | struct sk_buff *skb; | 4713 | struct sk_buff *skb; |
| 4723 | dma_addr_t dma_addr; | 4714 | dma_addr_t dma_addr; |
| 4724 | u32 opaque_key, desc_idx, *post_ptr; | 4715 | u32 opaque_key, desc_idx, *post_ptr; |
| 4725 | bool hw_vlan __maybe_unused = false; | ||
| 4726 | u16 vtag __maybe_unused = 0; | ||
| 4727 | 4716 | ||
| 4728 | desc_idx = desc->opaque & RXD_OPAQUE_INDEX_MASK; | 4717 | desc_idx = desc->opaque & RXD_OPAQUE_INDEX_MASK; |
| 4729 | opaque_key = desc->opaque & RXD_OPAQUE_RING_MASK; | 4718 | opaque_key = desc->opaque & RXD_OPAQUE_RING_MASK; |
| @@ -4782,12 +4771,12 @@ static int tg3_rx(struct tg3_napi *tnapi, int budget) | |||
| 4782 | tg3_recycle_rx(tnapi, tpr, opaque_key, | 4771 | tg3_recycle_rx(tnapi, tpr, opaque_key, |
| 4783 | desc_idx, *post_ptr); | 4772 | desc_idx, *post_ptr); |
| 4784 | 4773 | ||
| 4785 | copy_skb = netdev_alloc_skb(tp->dev, len + VLAN_HLEN + | 4774 | copy_skb = netdev_alloc_skb(tp->dev, len + |
| 4786 | TG3_RAW_IP_ALIGN); | 4775 | TG3_RAW_IP_ALIGN); |
| 4787 | if (copy_skb == NULL) | 4776 | if (copy_skb == NULL) |
| 4788 | goto drop_it_no_recycle; | 4777 | goto drop_it_no_recycle; |
| 4789 | 4778 | ||
| 4790 | skb_reserve(copy_skb, TG3_RAW_IP_ALIGN + VLAN_HLEN); | 4779 | skb_reserve(copy_skb, TG3_RAW_IP_ALIGN); |
| 4791 | skb_put(copy_skb, len); | 4780 | skb_put(copy_skb, len); |
| 4792 | pci_dma_sync_single_for_cpu(tp->pdev, dma_addr, len, PCI_DMA_FROMDEVICE); | 4781 | pci_dma_sync_single_for_cpu(tp->pdev, dma_addr, len, PCI_DMA_FROMDEVICE); |
| 4793 | skb_copy_from_linear_data(skb, copy_skb->data, len); | 4782 | skb_copy_from_linear_data(skb, copy_skb->data, len); |
| @@ -4814,30 +4803,11 @@ static int tg3_rx(struct tg3_napi *tnapi, int budget) | |||
| 4814 | } | 4803 | } |
| 4815 | 4804 | ||
| 4816 | if (desc->type_flags & RXD_FLAG_VLAN && | 4805 | if (desc->type_flags & RXD_FLAG_VLAN && |
| 4817 | !(tp->rx_mode & RX_MODE_KEEP_VLAN_TAG)) { | 4806 | !(tp->rx_mode & RX_MODE_KEEP_VLAN_TAG)) |
| 4818 | vtag = desc->err_vlan & RXD_VLAN_MASK; | 4807 | __vlan_hwaccel_put_tag(skb, |
| 4819 | #if TG3_VLAN_TAG_USED | 4808 | desc->err_vlan & RXD_VLAN_MASK); |
| 4820 | if (tp->vlgrp) | ||
| 4821 | hw_vlan = true; | ||
| 4822 | else | ||
| 4823 | #endif | ||
| 4824 | { | ||
| 4825 | struct vlan_ethhdr *ve = (struct vlan_ethhdr *) | ||
| 4826 | __skb_push(skb, VLAN_HLEN); | ||
| 4827 | |||
| 4828 | memmove(ve, skb->data + VLAN_HLEN, | ||
| 4829 | ETH_ALEN * 2); | ||
| 4830 | ve->h_vlan_proto = htons(ETH_P_8021Q); | ||
| 4831 | ve->h_vlan_TCI = htons(vtag); | ||
| 4832 | } | ||
| 4833 | } | ||
| 4834 | 4809 | ||
| 4835 | #if TG3_VLAN_TAG_USED | 4810 | napi_gro_receive(&tnapi->napi, skb); |
| 4836 | if (hw_vlan) | ||
| 4837 | vlan_gro_receive(&tnapi->napi, tp->vlgrp, vtag, skb); | ||
| 4838 | else | ||
| 4839 | #endif | ||
| 4840 | napi_gro_receive(&tnapi->napi, skb); | ||
| 4841 | 4811 | ||
| 4842 | received++; | 4812 | received++; |
| 4843 | budget--; | 4813 | budget--; |
| @@ -5740,11 +5710,9 @@ static netdev_tx_t tg3_start_xmit(struct sk_buff *skb, | |||
| 5740 | base_flags |= TXD_FLAG_TCPUDP_CSUM; | 5710 | base_flags |= TXD_FLAG_TCPUDP_CSUM; |
| 5741 | } | 5711 | } |
| 5742 | 5712 | ||
| 5743 | #if TG3_VLAN_TAG_USED | ||
| 5744 | if (vlan_tx_tag_present(skb)) | 5713 | if (vlan_tx_tag_present(skb)) |
| 5745 | base_flags |= (TXD_FLAG_VLAN | | 5714 | base_flags |= (TXD_FLAG_VLAN | |
| 5746 | (vlan_tx_tag_get(skb) << 16)); | 5715 | (vlan_tx_tag_get(skb) << 16)); |
| 5747 | #endif | ||
| 5748 | 5716 | ||
| 5749 | len = skb_headlen(skb); | 5717 | len = skb_headlen(skb); |
| 5750 | 5718 | ||
| @@ -5986,11 +5954,10 @@ static netdev_tx_t tg3_start_xmit_dma_bug(struct sk_buff *skb, | |||
| 5986 | } | 5954 | } |
| 5987 | } | 5955 | } |
| 5988 | } | 5956 | } |
| 5989 | #if TG3_VLAN_TAG_USED | 5957 | |
| 5990 | if (vlan_tx_tag_present(skb)) | 5958 | if (vlan_tx_tag_present(skb)) |
| 5991 | base_flags |= (TXD_FLAG_VLAN | | 5959 | base_flags |= (TXD_FLAG_VLAN | |
| 5992 | (vlan_tx_tag_get(skb) << 16)); | 5960 | (vlan_tx_tag_get(skb) << 16)); |
| 5993 | #endif | ||
| 5994 | 5961 | ||
| 5995 | if ((tp->tg3_flags3 & TG3_FLG3_USE_JUMBO_BDFLAG) && | 5962 | if ((tp->tg3_flags3 & TG3_FLG3_USE_JUMBO_BDFLAG) && |
| 5996 | !mss && skb->len > VLAN_ETH_FRAME_LEN) | 5963 | !mss && skb->len > VLAN_ETH_FRAME_LEN) |
| @@ -9532,17 +9499,10 @@ static void __tg3_set_rx_mode(struct net_device *dev) | |||
| 9532 | rx_mode = tp->rx_mode & ~(RX_MODE_PROMISC | | 9499 | rx_mode = tp->rx_mode & ~(RX_MODE_PROMISC | |
| 9533 | RX_MODE_KEEP_VLAN_TAG); | 9500 | RX_MODE_KEEP_VLAN_TAG); |
| 9534 | 9501 | ||
| 9502 | #if !defined(CONFIG_VLAN_8021Q) && !defined(CONFIG_VLAN_8021Q_MODULE) | ||
| 9535 | /* When ASF is in use, we always keep the RX_MODE_KEEP_VLAN_TAG | 9503 | /* When ASF is in use, we always keep the RX_MODE_KEEP_VLAN_TAG |
| 9536 | * flag clear. | 9504 | * flag clear. |
| 9537 | */ | 9505 | */ |
| 9538 | #if TG3_VLAN_TAG_USED | ||
| 9539 | if (!tp->vlgrp && | ||
| 9540 | !(tp->tg3_flags & TG3_FLAG_ENABLE_ASF)) | ||
| 9541 | rx_mode |= RX_MODE_KEEP_VLAN_TAG; | ||
| 9542 | #else | ||
| 9543 | /* By definition, VLAN is disabled always in this | ||
| 9544 | * case. | ||
| 9545 | */ | ||
| 9546 | if (!(tp->tg3_flags & TG3_FLAG_ENABLE_ASF)) | 9506 | if (!(tp->tg3_flags & TG3_FLAG_ENABLE_ASF)) |
| 9547 | rx_mode |= RX_MODE_KEEP_VLAN_TAG; | 9507 | rx_mode |= RX_MODE_KEEP_VLAN_TAG; |
| 9548 | #endif | 9508 | #endif |
| @@ -11230,31 +11190,6 @@ static int tg3_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd) | |||
| 11230 | return -EOPNOTSUPP; | 11190 | return -EOPNOTSUPP; |
| 11231 | } | 11191 | } |
| 11232 | 11192 | ||
| 11233 | #if TG3_VLAN_TAG_USED | ||
| 11234 | static void tg3_vlan_rx_register(struct net_device *dev, struct vlan_group *grp) | ||
| 11235 | { | ||
| 11236 | struct tg3 *tp = netdev_priv(dev); | ||
| 11237 | |||
| 11238 | if (!netif_running(dev)) { | ||
| 11239 | tp->vlgrp = grp; | ||
| 11240 | return; | ||
| 11241 | } | ||
| 11242 | |||
| 11243 | tg3_netif_stop(tp); | ||
| 11244 | |||
| 11245 | tg3_full_lock(tp, 0); | ||
| 11246 | |||
| 11247 | tp->vlgrp = grp; | ||
| 11248 | |||
| 11249 | /* Update RX_MODE_KEEP_VLAN_TAG bit in RX_MODE register. */ | ||
| 11250 | __tg3_set_rx_mode(dev); | ||
| 11251 | |||
| 11252 | tg3_netif_start(tp); | ||
| 11253 | |||
| 11254 | tg3_full_unlock(tp); | ||
| 11255 | } | ||
| 11256 | #endif | ||
| 11257 | |||
| 11258 | static int tg3_get_coalesce(struct net_device *dev, struct ethtool_coalesce *ec) | 11193 | static int tg3_get_coalesce(struct net_device *dev, struct ethtool_coalesce *ec) |
| 11259 | { | 11194 | { |
| 11260 | struct tg3 *tp = netdev_priv(dev); | 11195 | struct tg3 *tp = netdev_priv(dev); |
| @@ -13066,9 +13001,7 @@ static struct pci_dev * __devinit tg3_find_peer(struct tg3 *); | |||
| 13066 | 13001 | ||
| 13067 | static void inline vlan_features_add(struct net_device *dev, unsigned long flags) | 13002 | static void inline vlan_features_add(struct net_device *dev, unsigned long flags) |
| 13068 | { | 13003 | { |
| 13069 | #if TG3_VLAN_TAG_USED | ||
| 13070 | dev->vlan_features |= flags; | 13004 | dev->vlan_features |= flags; |
| 13071 | #endif | ||
| 13072 | } | 13005 | } |
| 13073 | 13006 | ||
| 13074 | static inline u32 tg3_rx_ret_ring_size(struct tg3 *tp) | 13007 | static inline u32 tg3_rx_ret_ring_size(struct tg3 *tp) |
| @@ -13861,11 +13794,11 @@ static int __devinit tg3_get_invariants(struct tg3 *tp) | |||
| 13861 | else | 13794 | else |
| 13862 | tp->tg3_flags &= ~TG3_FLAG_POLL_SERDES; | 13795 | tp->tg3_flags &= ~TG3_FLAG_POLL_SERDES; |
| 13863 | 13796 | ||
| 13864 | tp->rx_offset = NET_IP_ALIGN + TG3_RX_HEADROOM; | 13797 | tp->rx_offset = NET_IP_ALIGN; |
| 13865 | tp->rx_copy_thresh = TG3_RX_COPY_THRESHOLD; | 13798 | tp->rx_copy_thresh = TG3_RX_COPY_THRESHOLD; |
| 13866 | if (GET_ASIC_REV(tp->pci_chip_rev_id) == ASIC_REV_5701 && | 13799 | if (GET_ASIC_REV(tp->pci_chip_rev_id) == ASIC_REV_5701 && |
| 13867 | (tp->tg3_flags & TG3_FLAG_PCIX_MODE) != 0) { | 13800 | (tp->tg3_flags & TG3_FLAG_PCIX_MODE) != 0) { |
| 13868 | tp->rx_offset -= NET_IP_ALIGN; | 13801 | tp->rx_offset = 0; |
| 13869 | #ifndef CONFIG_HAVE_EFFICIENT_UNALIGNED_ACCESS | 13802 | #ifndef CONFIG_HAVE_EFFICIENT_UNALIGNED_ACCESS |
| 13870 | tp->rx_copy_thresh = ~(u16)0; | 13803 | tp->rx_copy_thresh = ~(u16)0; |
| 13871 | #endif | 13804 | #endif |
| @@ -14629,9 +14562,6 @@ static const struct net_device_ops tg3_netdev_ops = { | |||
| 14629 | .ndo_do_ioctl = tg3_ioctl, | 14562 | .ndo_do_ioctl = tg3_ioctl, |
| 14630 | .ndo_tx_timeout = tg3_tx_timeout, | 14563 | .ndo_tx_timeout = tg3_tx_timeout, |
| 14631 | .ndo_change_mtu = tg3_change_mtu, | 14564 | .ndo_change_mtu = tg3_change_mtu, |
| 14632 | #if TG3_VLAN_TAG_USED | ||
| 14633 | .ndo_vlan_rx_register = tg3_vlan_rx_register, | ||
| 14634 | #endif | ||
| 14635 | #ifdef CONFIG_NET_POLL_CONTROLLER | 14565 | #ifdef CONFIG_NET_POLL_CONTROLLER |
| 14636 | .ndo_poll_controller = tg3_poll_controller, | 14566 | .ndo_poll_controller = tg3_poll_controller, |
| 14637 | #endif | 14567 | #endif |
| @@ -14648,9 +14578,6 @@ static const struct net_device_ops tg3_netdev_ops_dma_bug = { | |||
| 14648 | .ndo_do_ioctl = tg3_ioctl, | 14578 | .ndo_do_ioctl = tg3_ioctl, |
| 14649 | .ndo_tx_timeout = tg3_tx_timeout, | 14579 | .ndo_tx_timeout = tg3_tx_timeout, |
| 14650 | .ndo_change_mtu = tg3_change_mtu, | 14580 | .ndo_change_mtu = tg3_change_mtu, |
| 14651 | #if TG3_VLAN_TAG_USED | ||
| 14652 | .ndo_vlan_rx_register = tg3_vlan_rx_register, | ||
| 14653 | #endif | ||
| 14654 | #ifdef CONFIG_NET_POLL_CONTROLLER | 14581 | #ifdef CONFIG_NET_POLL_CONTROLLER |
| 14655 | .ndo_poll_controller = tg3_poll_controller, | 14582 | .ndo_poll_controller = tg3_poll_controller, |
| 14656 | #endif | 14583 | #endif |
| @@ -14700,9 +14627,7 @@ static int __devinit tg3_init_one(struct pci_dev *pdev, | |||
| 14700 | 14627 | ||
| 14701 | SET_NETDEV_DEV(dev, &pdev->dev); | 14628 | SET_NETDEV_DEV(dev, &pdev->dev); |
| 14702 | 14629 | ||
| 14703 | #if TG3_VLAN_TAG_USED | ||
| 14704 | dev->features |= NETIF_F_HW_VLAN_TX | NETIF_F_HW_VLAN_RX; | 14630 | dev->features |= NETIF_F_HW_VLAN_TX | NETIF_F_HW_VLAN_RX; |
| 14705 | #endif | ||
| 14706 | 14631 | ||
| 14707 | tp = netdev_priv(dev); | 14632 | tp = netdev_priv(dev); |
| 14708 | tp->pdev = pdev; | 14633 | tp->pdev = pdev; |
diff --git a/drivers/net/tg3.h b/drivers/net/tg3.h index d62c8d937c8..f528243e1a4 100644 --- a/drivers/net/tg3.h +++ b/drivers/net/tg3.h | |||
| @@ -2808,9 +2808,6 @@ struct tg3 { | |||
| 2808 | u32 rx_std_max_post; | 2808 | u32 rx_std_max_post; |
| 2809 | u32 rx_offset; | 2809 | u32 rx_offset; |
| 2810 | u32 rx_pkt_map_sz; | 2810 | u32 rx_pkt_map_sz; |
| 2811 | #if TG3_VLAN_TAG_USED | ||
| 2812 | struct vlan_group *vlgrp; | ||
| 2813 | #endif | ||
| 2814 | 2811 | ||
| 2815 | 2812 | ||
| 2816 | /* begin "everything else" cacheline(s) section */ | 2813 | /* begin "everything else" cacheline(s) section */ |
diff --git a/drivers/net/usb/kaweth.c b/drivers/net/usb/kaweth.c index 5e98643a4a2..7dc84971f26 100644 --- a/drivers/net/usb/kaweth.c +++ b/drivers/net/usb/kaweth.c | |||
| @@ -406,6 +406,7 @@ static int kaweth_download_firmware(struct kaweth_device *kaweth, | |||
| 406 | 406 | ||
| 407 | if (fw->size > KAWETH_FIRMWARE_BUF_SIZE) { | 407 | if (fw->size > KAWETH_FIRMWARE_BUF_SIZE) { |
| 408 | err("Firmware too big: %zu", fw->size); | 408 | err("Firmware too big: %zu", fw->size); |
| 409 | release_firmware(fw); | ||
| 409 | return -ENOSPC; | 410 | return -ENOSPC; |
| 410 | } | 411 | } |
| 411 | data_len = fw->size; | 412 | data_len = fw->size; |
diff --git a/drivers/net/vxge/vxge-config.c b/drivers/net/vxge/vxge-config.c index 01c05f53e2f..228d4f7a58a 100644 --- a/drivers/net/vxge/vxge-config.c +++ b/drivers/net/vxge/vxge-config.c | |||
| @@ -3690,7 +3690,7 @@ __vxge_hw_vpath_rts_table_get(struct __vxge_hw_vpath_handle *vp, | |||
| 3690 | if (status != VXGE_HW_OK) | 3690 | if (status != VXGE_HW_OK) |
| 3691 | goto exit; | 3691 | goto exit; |
| 3692 | 3692 | ||
| 3693 | if ((rts_table != VXGE_HW_RTS_ACCESS_STEER_CTRL_DATA_STRUCT_SEL_DA) || | 3693 | if ((rts_table != VXGE_HW_RTS_ACCESS_STEER_CTRL_DATA_STRUCT_SEL_DA) && |
| 3694 | (rts_table != | 3694 | (rts_table != |
| 3695 | VXGE_HW_RTS_ACS_STEER_CTRL_DATA_STRUCT_SEL_RTH_MULTI_IT)) | 3695 | VXGE_HW_RTS_ACS_STEER_CTRL_DATA_STRUCT_SEL_RTH_MULTI_IT)) |
| 3696 | *data1 = 0; | 3696 | *data1 = 0; |
diff --git a/drivers/net/wireless/ath/ath5k/dma.c b/drivers/net/wireless/ath/ath5k/dma.c index 0064be7ce5c..21091c26a9a 100644 --- a/drivers/net/wireless/ath/ath5k/dma.c +++ b/drivers/net/wireless/ath/ath5k/dma.c | |||
| @@ -838,9 +838,9 @@ int ath5k_hw_dma_stop(struct ath5k_hw *ah) | |||
| 838 | for (i = 0; i < qmax; i++) { | 838 | for (i = 0; i < qmax; i++) { |
| 839 | err = ath5k_hw_stop_tx_dma(ah, i); | 839 | err = ath5k_hw_stop_tx_dma(ah, i); |
| 840 | /* -EINVAL -> queue inactive */ | 840 | /* -EINVAL -> queue inactive */ |
| 841 | if (err != -EINVAL) | 841 | if (err && err != -EINVAL) |
| 842 | return err; | 842 | return err; |
| 843 | } | 843 | } |
| 844 | 844 | ||
| 845 | return err; | 845 | return 0; |
| 846 | } | 846 | } |
diff --git a/drivers/net/wireless/ath/ath5k/pcu.c b/drivers/net/wireless/ath/ath5k/pcu.c index e5f2b96a4c6..a702817daf7 100644 --- a/drivers/net/wireless/ath/ath5k/pcu.c +++ b/drivers/net/wireless/ath/ath5k/pcu.c | |||
| @@ -86,7 +86,7 @@ int ath5k_hw_get_frame_duration(struct ath5k_hw *ah, | |||
| 86 | if (!ah->ah_bwmode) { | 86 | if (!ah->ah_bwmode) { |
| 87 | dur = ieee80211_generic_frame_duration(sc->hw, | 87 | dur = ieee80211_generic_frame_duration(sc->hw, |
| 88 | NULL, len, rate); | 88 | NULL, len, rate); |
| 89 | return dur; | 89 | return le16_to_cpu(dur); |
| 90 | } | 90 | } |
| 91 | 91 | ||
| 92 | bitrate = rate->bitrate; | 92 | bitrate = rate->bitrate; |
| @@ -265,8 +265,6 @@ static inline void ath5k_hw_write_rate_duration(struct ath5k_hw *ah) | |||
| 265 | * what rate we should choose to TX ACKs. */ | 265 | * what rate we should choose to TX ACKs. */ |
| 266 | tx_time = ath5k_hw_get_frame_duration(ah, 10, rate); | 266 | tx_time = ath5k_hw_get_frame_duration(ah, 10, rate); |
| 267 | 267 | ||
| 268 | tx_time = le16_to_cpu(tx_time); | ||
| 269 | |||
| 270 | ath5k_hw_reg_write(ah, tx_time, reg); | 268 | ath5k_hw_reg_write(ah, tx_time, reg); |
| 271 | 269 | ||
| 272 | if (!(rate->flags & IEEE80211_RATE_SHORT_PREAMBLE)) | 270 | if (!(rate->flags & IEEE80211_RATE_SHORT_PREAMBLE)) |
diff --git a/drivers/net/wireless/ath/ath9k/ar9002_hw.c b/drivers/net/wireless/ath/ath9k/ar9002_hw.c index f8a7771faee..f44c84ab5dc 100644 --- a/drivers/net/wireless/ath/ath9k/ar9002_hw.c +++ b/drivers/net/wireless/ath/ath9k/ar9002_hw.c | |||
| @@ -426,9 +426,8 @@ static void ar9002_hw_configpcipowersave(struct ath_hw *ah, | |||
| 426 | } | 426 | } |
| 427 | 427 | ||
| 428 | /* WAR for ASPM system hang */ | 428 | /* WAR for ASPM system hang */ |
| 429 | if (AR_SREV_9280(ah) || AR_SREV_9285(ah) || AR_SREV_9287(ah)) { | 429 | if (AR_SREV_9285(ah) || AR_SREV_9287(ah)) |
| 430 | val |= (AR_WA_BIT6 | AR_WA_BIT7); | 430 | val |= (AR_WA_BIT6 | AR_WA_BIT7); |
| 431 | } | ||
| 432 | 431 | ||
| 433 | if (AR_SREV_9285E_20(ah)) | 432 | if (AR_SREV_9285E_20(ah)) |
| 434 | val |= AR_WA_BIT23; | 433 | val |= AR_WA_BIT23; |
diff --git a/drivers/net/wireless/ath/ath9k/htc_drv_init.c b/drivers/net/wireless/ath/ath9k/htc_drv_init.c index 38433f9bfe5..0352f0994ca 100644 --- a/drivers/net/wireless/ath/ath9k/htc_drv_init.c +++ b/drivers/net/wireless/ath/ath9k/htc_drv_init.c | |||
| @@ -142,9 +142,6 @@ static void ath9k_deinit_priv(struct ath9k_htc_priv *priv) | |||
| 142 | { | 142 | { |
| 143 | ath9k_htc_exit_debug(priv->ah); | 143 | ath9k_htc_exit_debug(priv->ah); |
| 144 | ath9k_hw_deinit(priv->ah); | 144 | ath9k_hw_deinit(priv->ah); |
| 145 | tasklet_kill(&priv->swba_tasklet); | ||
| 146 | tasklet_kill(&priv->rx_tasklet); | ||
| 147 | tasklet_kill(&priv->tx_tasklet); | ||
| 148 | kfree(priv->ah); | 145 | kfree(priv->ah); |
| 149 | priv->ah = NULL; | 146 | priv->ah = NULL; |
| 150 | } | 147 | } |
diff --git a/drivers/net/wireless/ath/ath9k/htc_drv_main.c b/drivers/net/wireless/ath/ath9k/htc_drv_main.c index f4d576bc3cc..6bb59958f71 100644 --- a/drivers/net/wireless/ath/ath9k/htc_drv_main.c +++ b/drivers/net/wireless/ath/ath9k/htc_drv_main.c | |||
| @@ -1025,12 +1025,6 @@ static void ath9k_htc_stop(struct ieee80211_hw *hw) | |||
| 1025 | int ret = 0; | 1025 | int ret = 0; |
| 1026 | u8 cmd_rsp; | 1026 | u8 cmd_rsp; |
| 1027 | 1027 | ||
| 1028 | /* Cancel all the running timers/work .. */ | ||
| 1029 | cancel_work_sync(&priv->fatal_work); | ||
| 1030 | cancel_work_sync(&priv->ps_work); | ||
| 1031 | cancel_delayed_work_sync(&priv->ath9k_led_blink_work); | ||
| 1032 | ath9k_led_stop_brightness(priv); | ||
| 1033 | |||
| 1034 | mutex_lock(&priv->mutex); | 1028 | mutex_lock(&priv->mutex); |
| 1035 | 1029 | ||
| 1036 | if (priv->op_flags & OP_INVALID) { | 1030 | if (priv->op_flags & OP_INVALID) { |
| @@ -1044,8 +1038,23 @@ static void ath9k_htc_stop(struct ieee80211_hw *hw) | |||
| 1044 | WMI_CMD(WMI_DISABLE_INTR_CMDID); | 1038 | WMI_CMD(WMI_DISABLE_INTR_CMDID); |
| 1045 | WMI_CMD(WMI_DRAIN_TXQ_ALL_CMDID); | 1039 | WMI_CMD(WMI_DRAIN_TXQ_ALL_CMDID); |
| 1046 | WMI_CMD(WMI_STOP_RECV_CMDID); | 1040 | WMI_CMD(WMI_STOP_RECV_CMDID); |
| 1041 | |||
| 1042 | tasklet_kill(&priv->swba_tasklet); | ||
| 1043 | tasklet_kill(&priv->rx_tasklet); | ||
| 1044 | tasklet_kill(&priv->tx_tasklet); | ||
| 1045 | |||
| 1047 | skb_queue_purge(&priv->tx_queue); | 1046 | skb_queue_purge(&priv->tx_queue); |
| 1048 | 1047 | ||
| 1048 | mutex_unlock(&priv->mutex); | ||
| 1049 | |||
| 1050 | /* Cancel all the running timers/work .. */ | ||
| 1051 | cancel_work_sync(&priv->fatal_work); | ||
| 1052 | cancel_work_sync(&priv->ps_work); | ||
| 1053 | cancel_delayed_work_sync(&priv->ath9k_led_blink_work); | ||
| 1054 | ath9k_led_stop_brightness(priv); | ||
| 1055 | |||
| 1056 | mutex_lock(&priv->mutex); | ||
| 1057 | |||
| 1049 | /* Remove monitor interface here */ | 1058 | /* Remove monitor interface here */ |
| 1050 | if (ah->opmode == NL80211_IFTYPE_MONITOR) { | 1059 | if (ah->opmode == NL80211_IFTYPE_MONITOR) { |
| 1051 | if (ath9k_htc_remove_monitor_interface(priv)) | 1060 | if (ath9k_htc_remove_monitor_interface(priv)) |
diff --git a/drivers/net/wireless/ath/ath9k/hw.c b/drivers/net/wireless/ath/ath9k/hw.c index 1afb8bb8575..9f01e50d5cd 100644 --- a/drivers/net/wireless/ath/ath9k/hw.c +++ b/drivers/net/wireless/ath/ath9k/hw.c | |||
| @@ -369,6 +369,9 @@ static void ath9k_hw_init_config(struct ath_hw *ah) | |||
| 369 | else | 369 | else |
| 370 | ah->config.ht_enable = 0; | 370 | ah->config.ht_enable = 0; |
| 371 | 371 | ||
| 372 | /* PAPRD needs some more work to be enabled */ | ||
| 373 | ah->config.paprd_disable = 1; | ||
| 374 | |||
| 372 | ah->config.rx_intr_mitigation = true; | 375 | ah->config.rx_intr_mitigation = true; |
| 373 | ah->config.pcieSerDesWrite = true; | 376 | ah->config.pcieSerDesWrite = true; |
| 374 | 377 | ||
| @@ -1933,7 +1936,8 @@ int ath9k_hw_fill_cap_info(struct ath_hw *ah) | |||
| 1933 | pCap->rx_status_len = sizeof(struct ar9003_rxs); | 1936 | pCap->rx_status_len = sizeof(struct ar9003_rxs); |
| 1934 | pCap->tx_desc_len = sizeof(struct ar9003_txc); | 1937 | pCap->tx_desc_len = sizeof(struct ar9003_txc); |
| 1935 | pCap->txs_len = sizeof(struct ar9003_txs); | 1938 | pCap->txs_len = sizeof(struct ar9003_txs); |
| 1936 | if (ah->eep_ops->get_eeprom(ah, EEP_PAPRD)) | 1939 | if (!ah->config.paprd_disable && |
| 1940 | ah->eep_ops->get_eeprom(ah, EEP_PAPRD)) | ||
| 1937 | pCap->hw_caps |= ATH9K_HW_CAP_PAPRD; | 1941 | pCap->hw_caps |= ATH9K_HW_CAP_PAPRD; |
| 1938 | } else { | 1942 | } else { |
| 1939 | pCap->tx_desc_len = sizeof(struct ath_desc); | 1943 | pCap->tx_desc_len = sizeof(struct ath_desc); |
diff --git a/drivers/net/wireless/ath/ath9k/hw.h b/drivers/net/wireless/ath/ath9k/hw.h index 5a3dfec45e9..ea9fde67064 100644 --- a/drivers/net/wireless/ath/ath9k/hw.h +++ b/drivers/net/wireless/ath/ath9k/hw.h | |||
| @@ -225,6 +225,7 @@ struct ath9k_ops_config { | |||
| 225 | u32 pcie_waen; | 225 | u32 pcie_waen; |
| 226 | u8 analog_shiftreg; | 226 | u8 analog_shiftreg; |
| 227 | u8 ht_enable; | 227 | u8 ht_enable; |
| 228 | u8 paprd_disable; | ||
| 228 | u32 ofdm_trig_low; | 229 | u32 ofdm_trig_low; |
| 229 | u32 ofdm_trig_high; | 230 | u32 ofdm_trig_high; |
| 230 | u32 cck_trig_high; | 231 | u32 cck_trig_high; |
diff --git a/drivers/net/wireless/ath/ath9k/init.c b/drivers/net/wireless/ath/ath9k/init.c index 767d8b86f1e..087a6a95edd 100644 --- a/drivers/net/wireless/ath/ath9k/init.c +++ b/drivers/net/wireless/ath/ath9k/init.c | |||
| @@ -598,8 +598,6 @@ err_btcoex: | |||
| 598 | err_queues: | 598 | err_queues: |
| 599 | ath9k_hw_deinit(ah); | 599 | ath9k_hw_deinit(ah); |
| 600 | err_hw: | 600 | err_hw: |
| 601 | tasklet_kill(&sc->intr_tq); | ||
| 602 | tasklet_kill(&sc->bcon_tasklet); | ||
| 603 | 601 | ||
| 604 | kfree(ah); | 602 | kfree(ah); |
| 605 | sc->sc_ah = NULL; | 603 | sc->sc_ah = NULL; |
| @@ -807,9 +805,6 @@ static void ath9k_deinit_softc(struct ath_softc *sc) | |||
| 807 | 805 | ||
| 808 | ath9k_hw_deinit(sc->sc_ah); | 806 | ath9k_hw_deinit(sc->sc_ah); |
| 809 | 807 | ||
| 810 | tasklet_kill(&sc->intr_tq); | ||
| 811 | tasklet_kill(&sc->bcon_tasklet); | ||
| 812 | |||
| 813 | kfree(sc->sc_ah); | 808 | kfree(sc->sc_ah); |
| 814 | sc->sc_ah = NULL; | 809 | sc->sc_ah = NULL; |
| 815 | } | 810 | } |
| @@ -824,6 +819,8 @@ void ath9k_deinit_device(struct ath_softc *sc) | |||
| 824 | wiphy_rfkill_stop_polling(sc->hw->wiphy); | 819 | wiphy_rfkill_stop_polling(sc->hw->wiphy); |
| 825 | ath_deinit_leds(sc); | 820 | ath_deinit_leds(sc); |
| 826 | 821 | ||
| 822 | ath9k_ps_restore(sc); | ||
| 823 | |||
| 827 | for (i = 0; i < sc->num_sec_wiphy; i++) { | 824 | for (i = 0; i < sc->num_sec_wiphy; i++) { |
| 828 | struct ath_wiphy *aphy = sc->sec_wiphy[i]; | 825 | struct ath_wiphy *aphy = sc->sec_wiphy[i]; |
| 829 | if (aphy == NULL) | 826 | if (aphy == NULL) |
diff --git a/drivers/net/wireless/ath/ath9k/main.c b/drivers/net/wireless/ath/ath9k/main.c index f90a6ca94a7..9040c2ff190 100644 --- a/drivers/net/wireless/ath/ath9k/main.c +++ b/drivers/net/wireless/ath/ath9k/main.c | |||
| @@ -325,6 +325,8 @@ static bool ath_paprd_send_frame(struct ath_softc *sc, struct sk_buff *skb, int | |||
| 325 | { | 325 | { |
| 326 | struct ieee80211_hw *hw = sc->hw; | 326 | struct ieee80211_hw *hw = sc->hw; |
| 327 | struct ieee80211_tx_info *tx_info = IEEE80211_SKB_CB(skb); | 327 | struct ieee80211_tx_info *tx_info = IEEE80211_SKB_CB(skb); |
| 328 | struct ath_hw *ah = sc->sc_ah; | ||
| 329 | struct ath_common *common = ath9k_hw_common(ah); | ||
| 328 | struct ath_tx_control txctl; | 330 | struct ath_tx_control txctl; |
| 329 | int time_left; | 331 | int time_left; |
| 330 | 332 | ||
| @@ -342,8 +344,12 @@ static bool ath_paprd_send_frame(struct ath_softc *sc, struct sk_buff *skb, int | |||
| 342 | init_completion(&sc->paprd_complete); | 344 | init_completion(&sc->paprd_complete); |
| 343 | sc->paprd_pending = true; | 345 | sc->paprd_pending = true; |
| 344 | txctl.paprd = BIT(chain); | 346 | txctl.paprd = BIT(chain); |
| 345 | if (ath_tx_start(hw, skb, &txctl) != 0) | 347 | |
| 348 | if (ath_tx_start(hw, skb, &txctl) != 0) { | ||
| 349 | ath_dbg(common, ATH_DBG_XMIT, "PAPRD TX failed\n"); | ||
| 350 | dev_kfree_skb_any(skb); | ||
| 346 | return false; | 351 | return false; |
| 352 | } | ||
| 347 | 353 | ||
| 348 | time_left = wait_for_completion_timeout(&sc->paprd_complete, | 354 | time_left = wait_for_completion_timeout(&sc->paprd_complete, |
| 349 | msecs_to_jiffies(ATH_PAPRD_TIMEOUT)); | 355 | msecs_to_jiffies(ATH_PAPRD_TIMEOUT)); |
| @@ -592,14 +598,12 @@ void ath9k_tasklet(unsigned long data) | |||
| 592 | u32 status = sc->intrstatus; | 598 | u32 status = sc->intrstatus; |
| 593 | u32 rxmask; | 599 | u32 rxmask; |
| 594 | 600 | ||
| 595 | ath9k_ps_wakeup(sc); | ||
| 596 | |||
| 597 | if (status & ATH9K_INT_FATAL) { | 601 | if (status & ATH9K_INT_FATAL) { |
| 598 | ath_reset(sc, true); | 602 | ath_reset(sc, true); |
| 599 | ath9k_ps_restore(sc); | ||
| 600 | return; | 603 | return; |
| 601 | } | 604 | } |
| 602 | 605 | ||
| 606 | ath9k_ps_wakeup(sc); | ||
| 603 | spin_lock(&sc->sc_pcu_lock); | 607 | spin_lock(&sc->sc_pcu_lock); |
| 604 | 608 | ||
| 605 | if (!ath9k_hw_check_alive(ah)) | 609 | if (!ath9k_hw_check_alive(ah)) |
| @@ -955,8 +959,6 @@ void ath_radio_disable(struct ath_softc *sc, struct ieee80211_hw *hw) | |||
| 955 | 959 | ||
| 956 | spin_unlock_bh(&sc->sc_pcu_lock); | 960 | spin_unlock_bh(&sc->sc_pcu_lock); |
| 957 | ath9k_ps_restore(sc); | 961 | ath9k_ps_restore(sc); |
| 958 | |||
| 959 | ath9k_setpower(sc, ATH9K_PM_FULL_SLEEP); | ||
| 960 | } | 962 | } |
| 961 | 963 | ||
| 962 | int ath_reset(struct ath_softc *sc, bool retry_tx) | 964 | int ath_reset(struct ath_softc *sc, bool retry_tx) |
| @@ -969,6 +971,7 @@ int ath_reset(struct ath_softc *sc, bool retry_tx) | |||
| 969 | /* Stop ANI */ | 971 | /* Stop ANI */ |
| 970 | del_timer_sync(&common->ani.timer); | 972 | del_timer_sync(&common->ani.timer); |
| 971 | 973 | ||
| 974 | ath9k_ps_wakeup(sc); | ||
| 972 | spin_lock_bh(&sc->sc_pcu_lock); | 975 | spin_lock_bh(&sc->sc_pcu_lock); |
| 973 | 976 | ||
| 974 | ieee80211_stop_queues(hw); | 977 | ieee80211_stop_queues(hw); |
| @@ -1015,6 +1018,7 @@ int ath_reset(struct ath_softc *sc, bool retry_tx) | |||
| 1015 | 1018 | ||
| 1016 | /* Start ANI */ | 1019 | /* Start ANI */ |
| 1017 | ath_start_ani(common); | 1020 | ath_start_ani(common); |
| 1021 | ath9k_ps_restore(sc); | ||
| 1018 | 1022 | ||
| 1019 | return r; | 1023 | return r; |
| 1020 | } | 1024 | } |
| @@ -1309,6 +1313,9 @@ static void ath9k_stop(struct ieee80211_hw *hw) | |||
| 1309 | 1313 | ||
| 1310 | spin_lock_bh(&sc->sc_pcu_lock); | 1314 | spin_lock_bh(&sc->sc_pcu_lock); |
| 1311 | 1315 | ||
| 1316 | /* prevent tasklets to enable interrupts once we disable them */ | ||
| 1317 | ah->imask &= ~ATH9K_INT_GLOBAL; | ||
| 1318 | |||
| 1312 | /* make sure h/w will not generate any interrupt | 1319 | /* make sure h/w will not generate any interrupt |
| 1313 | * before setting the invalid flag. */ | 1320 | * before setting the invalid flag. */ |
| 1314 | ath9k_hw_disable_interrupts(ah); | 1321 | ath9k_hw_disable_interrupts(ah); |
| @@ -1326,6 +1333,12 @@ static void ath9k_stop(struct ieee80211_hw *hw) | |||
| 1326 | 1333 | ||
| 1327 | spin_unlock_bh(&sc->sc_pcu_lock); | 1334 | spin_unlock_bh(&sc->sc_pcu_lock); |
| 1328 | 1335 | ||
| 1336 | /* we can now sync irq and kill any running tasklets, since we already | ||
| 1337 | * disabled interrupts and not holding a spin lock */ | ||
| 1338 | synchronize_irq(sc->irq); | ||
| 1339 | tasklet_kill(&sc->intr_tq); | ||
| 1340 | tasklet_kill(&sc->bcon_tasklet); | ||
| 1341 | |||
| 1329 | ath9k_ps_restore(sc); | 1342 | ath9k_ps_restore(sc); |
| 1330 | 1343 | ||
| 1331 | sc->ps_idle = true; | 1344 | sc->ps_idle = true; |
| @@ -1701,7 +1714,9 @@ static int ath9k_config(struct ieee80211_hw *hw, u32 changed) | |||
| 1701 | skip_chan_change: | 1714 | skip_chan_change: |
| 1702 | if (changed & IEEE80211_CONF_CHANGE_POWER) { | 1715 | if (changed & IEEE80211_CONF_CHANGE_POWER) { |
| 1703 | sc->config.txpowlimit = 2 * conf->power_level; | 1716 | sc->config.txpowlimit = 2 * conf->power_level; |
| 1717 | ath9k_ps_wakeup(sc); | ||
| 1704 | ath_update_txpow(sc); | 1718 | ath_update_txpow(sc); |
| 1719 | ath9k_ps_restore(sc); | ||
| 1705 | } | 1720 | } |
| 1706 | 1721 | ||
| 1707 | spin_lock_bh(&sc->wiphy_lock); | 1722 | spin_lock_bh(&sc->wiphy_lock); |
diff --git a/drivers/net/wireless/ath/ath9k/xmit.c b/drivers/net/wireless/ath/ath9k/xmit.c index 332d1feb5c1..33a37edbaf7 100644 --- a/drivers/net/wireless/ath/ath9k/xmit.c +++ b/drivers/net/wireless/ath/ath9k/xmit.c | |||
| @@ -2113,9 +2113,7 @@ static void ath_tx_complete_poll_work(struct work_struct *work) | |||
| 2113 | if (needreset) { | 2113 | if (needreset) { |
| 2114 | ath_dbg(ath9k_hw_common(sc->sc_ah), ATH_DBG_RESET, | 2114 | ath_dbg(ath9k_hw_common(sc->sc_ah), ATH_DBG_RESET, |
| 2115 | "tx hung, resetting the chip\n"); | 2115 | "tx hung, resetting the chip\n"); |
| 2116 | ath9k_ps_wakeup(sc); | ||
| 2117 | ath_reset(sc, true); | 2116 | ath_reset(sc, true); |
| 2118 | ath9k_ps_restore(sc); | ||
| 2119 | } | 2117 | } |
| 2120 | 2118 | ||
| 2121 | ieee80211_queue_delayed_work(sc->hw, &sc->tx_complete_work, | 2119 | ieee80211_queue_delayed_work(sc->hw, &sc->tx_complete_work, |
diff --git a/drivers/net/wireless/iwlwifi/iwl-4965.c b/drivers/net/wireless/iwlwifi/iwl-4965.c index 3f1e5f1bf84..91a9f525346 100644 --- a/drivers/net/wireless/iwlwifi/iwl-4965.c +++ b/drivers/net/wireless/iwlwifi/iwl-4965.c | |||
| @@ -2624,6 +2624,7 @@ struct iwl_cfg iwl4965_agn_cfg = { | |||
| 2624 | .fw_name_pre = IWL4965_FW_PRE, | 2624 | .fw_name_pre = IWL4965_FW_PRE, |
| 2625 | .ucode_api_max = IWL4965_UCODE_API_MAX, | 2625 | .ucode_api_max = IWL4965_UCODE_API_MAX, |
| 2626 | .ucode_api_min = IWL4965_UCODE_API_MIN, | 2626 | .ucode_api_min = IWL4965_UCODE_API_MIN, |
| 2627 | .sku = IWL_SKU_A|IWL_SKU_G|IWL_SKU_N, | ||
| 2627 | .valid_tx_ant = ANT_AB, | 2628 | .valid_tx_ant = ANT_AB, |
| 2628 | .valid_rx_ant = ANT_ABC, | 2629 | .valid_rx_ant = ANT_ABC, |
| 2629 | .eeprom_ver = EEPROM_4965_EEPROM_VERSION, | 2630 | .eeprom_ver = EEPROM_4965_EEPROM_VERSION, |
diff --git a/drivers/net/wireless/iwlwifi/iwl-agn-eeprom.c b/drivers/net/wireless/iwlwifi/iwl-agn-eeprom.c index 14ceb4df72f..27b5a3eec9d 100644 --- a/drivers/net/wireless/iwlwifi/iwl-agn-eeprom.c +++ b/drivers/net/wireless/iwlwifi/iwl-agn-eeprom.c | |||
| @@ -152,11 +152,14 @@ int iwl_eeprom_check_sku(struct iwl_priv *priv) | |||
| 152 | 152 | ||
| 153 | eeprom_sku = iwl_eeprom_query16(priv, EEPROM_SKU_CAP); | 153 | eeprom_sku = iwl_eeprom_query16(priv, EEPROM_SKU_CAP); |
| 154 | 154 | ||
| 155 | priv->cfg->sku = ((eeprom_sku & EEPROM_SKU_CAP_BAND_SELECTION) >> | 155 | if (!priv->cfg->sku) { |
| 156 | /* not using sku overwrite */ | ||
| 157 | priv->cfg->sku = | ||
| 158 | ((eeprom_sku & EEPROM_SKU_CAP_BAND_SELECTION) >> | ||
| 156 | EEPROM_SKU_CAP_BAND_POS); | 159 | EEPROM_SKU_CAP_BAND_POS); |
| 157 | if (eeprom_sku & EEPROM_SKU_CAP_11N_ENABLE) | 160 | if (eeprom_sku & EEPROM_SKU_CAP_11N_ENABLE) |
| 158 | priv->cfg->sku |= IWL_SKU_N; | 161 | priv->cfg->sku |= IWL_SKU_N; |
| 159 | 162 | } | |
| 160 | if (!priv->cfg->sku) { | 163 | if (!priv->cfg->sku) { |
| 161 | IWL_ERR(priv, "Invalid device sku\n"); | 164 | IWL_ERR(priv, "Invalid device sku\n"); |
| 162 | return -EINVAL; | 165 | return -EINVAL; |
diff --git a/drivers/net/wireless/rt2x00/rt73usb.c b/drivers/net/wireless/rt2x00/rt73usb.c index 0b4e8590cbb..029be3c6c03 100644 --- a/drivers/net/wireless/rt2x00/rt73usb.c +++ b/drivers/net/wireless/rt2x00/rt73usb.c | |||
| @@ -2446,6 +2446,7 @@ static struct usb_device_id rt73usb_device_table[] = { | |||
| 2446 | { USB_DEVICE(0x04bb, 0x093d), USB_DEVICE_DATA(&rt73usb_ops) }, | 2446 | { USB_DEVICE(0x04bb, 0x093d), USB_DEVICE_DATA(&rt73usb_ops) }, |
| 2447 | { USB_DEVICE(0x148f, 0x2573), USB_DEVICE_DATA(&rt73usb_ops) }, | 2447 | { USB_DEVICE(0x148f, 0x2573), USB_DEVICE_DATA(&rt73usb_ops) }, |
| 2448 | { USB_DEVICE(0x148f, 0x2671), USB_DEVICE_DATA(&rt73usb_ops) }, | 2448 | { USB_DEVICE(0x148f, 0x2671), USB_DEVICE_DATA(&rt73usb_ops) }, |
| 2449 | { USB_DEVICE(0x0812, 0x3101), USB_DEVICE_DATA(&rt73usb_ops) }, | ||
| 2449 | /* Qcom */ | 2450 | /* Qcom */ |
| 2450 | { USB_DEVICE(0x18e8, 0x6196), USB_DEVICE_DATA(&rt73usb_ops) }, | 2451 | { USB_DEVICE(0x18e8, 0x6196), USB_DEVICE_DATA(&rt73usb_ops) }, |
| 2451 | { USB_DEVICE(0x18e8, 0x6229), USB_DEVICE_DATA(&rt73usb_ops) }, | 2452 | { USB_DEVICE(0x18e8, 0x6229), USB_DEVICE_DATA(&rt73usb_ops) }, |
diff --git a/drivers/net/wireless/rtlwifi/efuse.c b/drivers/net/wireless/rtlwifi/efuse.c index b8433f3a9bc..62876cd5c41 100644 --- a/drivers/net/wireless/rtlwifi/efuse.c +++ b/drivers/net/wireless/rtlwifi/efuse.c | |||
| @@ -726,9 +726,9 @@ static int efuse_pg_packet_read(struct ieee80211_hw *hw, u8 offset, u8 *data) | |||
| 726 | } | 726 | } |
| 727 | 727 | ||
| 728 | static void efuse_write_data_case1(struct ieee80211_hw *hw, u16 *efuse_addr, | 728 | static void efuse_write_data_case1(struct ieee80211_hw *hw, u16 *efuse_addr, |
| 729 | u8 efuse_data, u8 offset, int *bcontinual, | 729 | u8 efuse_data, u8 offset, int *bcontinual, |
| 730 | u8 *write_state, struct pgpkt_struct target_pkt, | 730 | u8 *write_state, struct pgpkt_struct *target_pkt, |
| 731 | int *repeat_times, int *bresult, u8 word_en) | 731 | int *repeat_times, int *bresult, u8 word_en) |
| 732 | { | 732 | { |
| 733 | struct rtl_priv *rtlpriv = rtl_priv(hw); | 733 | struct rtl_priv *rtlpriv = rtl_priv(hw); |
| 734 | struct pgpkt_struct tmp_pkt; | 734 | struct pgpkt_struct tmp_pkt; |
| @@ -744,8 +744,8 @@ static void efuse_write_data_case1(struct ieee80211_hw *hw, u16 *efuse_addr, | |||
| 744 | tmp_pkt.word_en = tmp_header & 0x0F; | 744 | tmp_pkt.word_en = tmp_header & 0x0F; |
| 745 | tmp_word_cnts = efuse_calculate_word_cnts(tmp_pkt.word_en); | 745 | tmp_word_cnts = efuse_calculate_word_cnts(tmp_pkt.word_en); |
| 746 | 746 | ||
| 747 | if (tmp_pkt.offset != target_pkt.offset) { | 747 | if (tmp_pkt.offset != target_pkt->offset) { |
| 748 | efuse_addr = efuse_addr + (tmp_word_cnts * 2) + 1; | 748 | *efuse_addr = *efuse_addr + (tmp_word_cnts * 2) + 1; |
| 749 | *write_state = PG_STATE_HEADER; | 749 | *write_state = PG_STATE_HEADER; |
| 750 | } else { | 750 | } else { |
| 751 | for (tmpindex = 0; tmpindex < (tmp_word_cnts * 2); tmpindex++) { | 751 | for (tmpindex = 0; tmpindex < (tmp_word_cnts * 2); tmpindex++) { |
| @@ -756,23 +756,23 @@ static void efuse_write_data_case1(struct ieee80211_hw *hw, u16 *efuse_addr, | |||
| 756 | } | 756 | } |
| 757 | 757 | ||
| 758 | if (bdataempty == false) { | 758 | if (bdataempty == false) { |
| 759 | efuse_addr = efuse_addr + (tmp_word_cnts * 2) + 1; | 759 | *efuse_addr = *efuse_addr + (tmp_word_cnts * 2) + 1; |
| 760 | *write_state = PG_STATE_HEADER; | 760 | *write_state = PG_STATE_HEADER; |
| 761 | } else { | 761 | } else { |
| 762 | match_word_en = 0x0F; | 762 | match_word_en = 0x0F; |
| 763 | if (!((target_pkt.word_en & BIT(0)) | | 763 | if (!((target_pkt->word_en & BIT(0)) | |
| 764 | (tmp_pkt.word_en & BIT(0)))) | 764 | (tmp_pkt.word_en & BIT(0)))) |
| 765 | match_word_en &= (~BIT(0)); | 765 | match_word_en &= (~BIT(0)); |
| 766 | 766 | ||
| 767 | if (!((target_pkt.word_en & BIT(1)) | | 767 | if (!((target_pkt->word_en & BIT(1)) | |
| 768 | (tmp_pkt.word_en & BIT(1)))) | 768 | (tmp_pkt.word_en & BIT(1)))) |
| 769 | match_word_en &= (~BIT(1)); | 769 | match_word_en &= (~BIT(1)); |
| 770 | 770 | ||
| 771 | if (!((target_pkt.word_en & BIT(2)) | | 771 | if (!((target_pkt->word_en & BIT(2)) | |
| 772 | (tmp_pkt.word_en & BIT(2)))) | 772 | (tmp_pkt.word_en & BIT(2)))) |
| 773 | match_word_en &= (~BIT(2)); | 773 | match_word_en &= (~BIT(2)); |
| 774 | 774 | ||
| 775 | if (!((target_pkt.word_en & BIT(3)) | | 775 | if (!((target_pkt->word_en & BIT(3)) | |
| 776 | (tmp_pkt.word_en & BIT(3)))) | 776 | (tmp_pkt.word_en & BIT(3)))) |
| 777 | match_word_en &= (~BIT(3)); | 777 | match_word_en &= (~BIT(3)); |
| 778 | 778 | ||
| @@ -780,7 +780,7 @@ static void efuse_write_data_case1(struct ieee80211_hw *hw, u16 *efuse_addr, | |||
| 780 | badworden = efuse_word_enable_data_write( | 780 | badworden = efuse_word_enable_data_write( |
| 781 | hw, *efuse_addr + 1, | 781 | hw, *efuse_addr + 1, |
| 782 | tmp_pkt.word_en, | 782 | tmp_pkt.word_en, |
| 783 | target_pkt.data); | 783 | target_pkt->data); |
| 784 | 784 | ||
| 785 | if (0x0F != (badworden & 0x0F)) { | 785 | if (0x0F != (badworden & 0x0F)) { |
| 786 | u8 reorg_offset = offset; | 786 | u8 reorg_offset = offset; |
| @@ -791,26 +791,26 @@ static void efuse_write_data_case1(struct ieee80211_hw *hw, u16 *efuse_addr, | |||
| 791 | } | 791 | } |
| 792 | 792 | ||
| 793 | tmp_word_en = 0x0F; | 793 | tmp_word_en = 0x0F; |
| 794 | if ((target_pkt.word_en & BIT(0)) ^ | 794 | if ((target_pkt->word_en & BIT(0)) ^ |
| 795 | (match_word_en & BIT(0))) | 795 | (match_word_en & BIT(0))) |
| 796 | tmp_word_en &= (~BIT(0)); | 796 | tmp_word_en &= (~BIT(0)); |
| 797 | 797 | ||
| 798 | if ((target_pkt.word_en & BIT(1)) ^ | 798 | if ((target_pkt->word_en & BIT(1)) ^ |
| 799 | (match_word_en & BIT(1))) | 799 | (match_word_en & BIT(1))) |
| 800 | tmp_word_en &= (~BIT(1)); | 800 | tmp_word_en &= (~BIT(1)); |
| 801 | 801 | ||
| 802 | if ((target_pkt.word_en & BIT(2)) ^ | 802 | if ((target_pkt->word_en & BIT(2)) ^ |
| 803 | (match_word_en & BIT(2))) | 803 | (match_word_en & BIT(2))) |
| 804 | tmp_word_en &= (~BIT(2)); | 804 | tmp_word_en &= (~BIT(2)); |
| 805 | 805 | ||
| 806 | if ((target_pkt.word_en & BIT(3)) ^ | 806 | if ((target_pkt->word_en & BIT(3)) ^ |
| 807 | (match_word_en & BIT(3))) | 807 | (match_word_en & BIT(3))) |
| 808 | tmp_word_en &= (~BIT(3)); | 808 | tmp_word_en &= (~BIT(3)); |
| 809 | 809 | ||
| 810 | if ((tmp_word_en & 0x0F) != 0x0F) { | 810 | if ((tmp_word_en & 0x0F) != 0x0F) { |
| 811 | *efuse_addr = efuse_get_current_size(hw); | 811 | *efuse_addr = efuse_get_current_size(hw); |
| 812 | target_pkt.offset = offset; | 812 | target_pkt->offset = offset; |
| 813 | target_pkt.word_en = tmp_word_en; | 813 | target_pkt->word_en = tmp_word_en; |
| 814 | } else | 814 | } else |
| 815 | *bcontinual = false; | 815 | *bcontinual = false; |
| 816 | *write_state = PG_STATE_HEADER; | 816 | *write_state = PG_STATE_HEADER; |
| @@ -821,8 +821,8 @@ static void efuse_write_data_case1(struct ieee80211_hw *hw, u16 *efuse_addr, | |||
| 821 | } | 821 | } |
| 822 | } else { | 822 | } else { |
| 823 | *efuse_addr += (2 * tmp_word_cnts) + 1; | 823 | *efuse_addr += (2 * tmp_word_cnts) + 1; |
| 824 | target_pkt.offset = offset; | 824 | target_pkt->offset = offset; |
| 825 | target_pkt.word_en = word_en; | 825 | target_pkt->word_en = word_en; |
| 826 | *write_state = PG_STATE_HEADER; | 826 | *write_state = PG_STATE_HEADER; |
| 827 | } | 827 | } |
| 828 | } | 828 | } |
| @@ -938,7 +938,7 @@ static int efuse_pg_packet_write(struct ieee80211_hw *hw, | |||
| 938 | efuse_write_data_case1(hw, &efuse_addr, | 938 | efuse_write_data_case1(hw, &efuse_addr, |
| 939 | efuse_data, offset, | 939 | efuse_data, offset, |
| 940 | &bcontinual, | 940 | &bcontinual, |
| 941 | &write_state, target_pkt, | 941 | &write_state, &target_pkt, |
| 942 | &repeat_times, &bresult, | 942 | &repeat_times, &bresult, |
| 943 | word_en); | 943 | word_en); |
| 944 | else | 944 | else |
diff --git a/drivers/net/wireless/rtlwifi/pci.c b/drivers/net/wireless/rtlwifi/pci.c index 0fa36aa6701..1758d446324 100644 --- a/drivers/net/wireless/rtlwifi/pci.c +++ b/drivers/net/wireless/rtlwifi/pci.c | |||
| @@ -619,6 +619,13 @@ static void _rtl_pci_rx_interrupt(struct ieee80211_hw *hw) | |||
| 619 | struct sk_buff *uskb = NULL; | 619 | struct sk_buff *uskb = NULL; |
| 620 | u8 *pdata; | 620 | u8 *pdata; |
| 621 | uskb = dev_alloc_skb(skb->len + 128); | 621 | uskb = dev_alloc_skb(skb->len + 128); |
| 622 | if (!uskb) { | ||
| 623 | RT_TRACE(rtlpriv, | ||
| 624 | (COMP_INTR | COMP_RECV), | ||
| 625 | DBG_EMERG, | ||
| 626 | ("can't alloc rx skb\n")); | ||
| 627 | goto done; | ||
| 628 | } | ||
| 622 | memcpy(IEEE80211_SKB_RXCB(uskb), | 629 | memcpy(IEEE80211_SKB_RXCB(uskb), |
| 623 | &rx_status, | 630 | &rx_status, |
| 624 | sizeof(rx_status)); | 631 | sizeof(rx_status)); |
| @@ -641,7 +648,7 @@ static void _rtl_pci_rx_interrupt(struct ieee80211_hw *hw) | |||
| 641 | new_skb = dev_alloc_skb(rtlpci->rxbuffersize); | 648 | new_skb = dev_alloc_skb(rtlpci->rxbuffersize); |
| 642 | if (unlikely(!new_skb)) { | 649 | if (unlikely(!new_skb)) { |
| 643 | RT_TRACE(rtlpriv, (COMP_INTR | COMP_RECV), | 650 | RT_TRACE(rtlpriv, (COMP_INTR | COMP_RECV), |
| 644 | DBG_DMESG, | 651 | DBG_EMERG, |
| 645 | ("can't alloc skb for rx\n")); | 652 | ("can't alloc skb for rx\n")); |
| 646 | goto done; | 653 | goto done; |
| 647 | } | 654 | } |
| @@ -1066,9 +1073,9 @@ static int _rtl_pci_init_rx_ring(struct ieee80211_hw *hw) | |||
| 1066 | struct sk_buff *skb = | 1073 | struct sk_buff *skb = |
| 1067 | dev_alloc_skb(rtlpci->rxbuffersize); | 1074 | dev_alloc_skb(rtlpci->rxbuffersize); |
| 1068 | u32 bufferaddress; | 1075 | u32 bufferaddress; |
| 1069 | entry = &rtlpci->rx_ring[rx_queue_idx].desc[i]; | ||
| 1070 | if (!skb) | 1076 | if (!skb) |
| 1071 | return 0; | 1077 | return 0; |
| 1078 | entry = &rtlpci->rx_ring[rx_queue_idx].desc[i]; | ||
| 1072 | 1079 | ||
| 1073 | /*skb->dev = dev; */ | 1080 | /*skb->dev = dev; */ |
| 1074 | 1081 | ||
diff --git a/drivers/net/wireless/wl12xx/spi.c b/drivers/net/wireless/wl12xx/spi.c index 46714910f98..7145ea54378 100644 --- a/drivers/net/wireless/wl12xx/spi.c +++ b/drivers/net/wireless/wl12xx/spi.c | |||
| @@ -110,9 +110,8 @@ static void wl1271_spi_reset(struct wl1271 *wl) | |||
| 110 | spi_message_add_tail(&t, &m); | 110 | spi_message_add_tail(&t, &m); |
| 111 | 111 | ||
| 112 | spi_sync(wl_to_spi(wl), &m); | 112 | spi_sync(wl_to_spi(wl), &m); |
| 113 | kfree(cmd); | ||
| 114 | |||
| 115 | wl1271_dump(DEBUG_SPI, "spi reset -> ", cmd, WSPI_INIT_CMD_LEN); | 113 | wl1271_dump(DEBUG_SPI, "spi reset -> ", cmd, WSPI_INIT_CMD_LEN); |
| 114 | kfree(cmd); | ||
| 116 | } | 115 | } |
| 117 | 116 | ||
| 118 | static void wl1271_spi_init(struct wl1271 *wl) | 117 | static void wl1271_spi_init(struct wl1271 *wl) |
diff --git a/drivers/net/xen-netfront.c b/drivers/net/xen-netfront.c index 546de574982..da1f1212034 100644 --- a/drivers/net/xen-netfront.c +++ b/drivers/net/xen-netfront.c | |||
| @@ -120,6 +120,9 @@ struct netfront_info { | |||
| 120 | unsigned long rx_pfn_array[NET_RX_RING_SIZE]; | 120 | unsigned long rx_pfn_array[NET_RX_RING_SIZE]; |
| 121 | struct multicall_entry rx_mcl[NET_RX_RING_SIZE+1]; | 121 | struct multicall_entry rx_mcl[NET_RX_RING_SIZE+1]; |
| 122 | struct mmu_update rx_mmu[NET_RX_RING_SIZE]; | 122 | struct mmu_update rx_mmu[NET_RX_RING_SIZE]; |
| 123 | |||
| 124 | /* Statistics */ | ||
| 125 | int rx_gso_checksum_fixup; | ||
| 123 | }; | 126 | }; |
| 124 | 127 | ||
| 125 | struct netfront_rx_info { | 128 | struct netfront_rx_info { |
| @@ -770,11 +773,29 @@ static RING_IDX xennet_fill_frags(struct netfront_info *np, | |||
| 770 | return cons; | 773 | return cons; |
| 771 | } | 774 | } |
| 772 | 775 | ||
| 773 | static int skb_checksum_setup(struct sk_buff *skb) | 776 | static int checksum_setup(struct net_device *dev, struct sk_buff *skb) |
| 774 | { | 777 | { |
| 775 | struct iphdr *iph; | 778 | struct iphdr *iph; |
| 776 | unsigned char *th; | 779 | unsigned char *th; |
| 777 | int err = -EPROTO; | 780 | int err = -EPROTO; |
| 781 | int recalculate_partial_csum = 0; | ||
| 782 | |||
| 783 | /* | ||
| 784 | * A GSO SKB must be CHECKSUM_PARTIAL. However some buggy | ||
| 785 | * peers can fail to set NETRXF_csum_blank when sending a GSO | ||
| 786 | * frame. In this case force the SKB to CHECKSUM_PARTIAL and | ||
| 787 | * recalculate the partial checksum. | ||
| 788 | */ | ||
| 789 | if (skb->ip_summed != CHECKSUM_PARTIAL && skb_is_gso(skb)) { | ||
| 790 | struct netfront_info *np = netdev_priv(dev); | ||
| 791 | np->rx_gso_checksum_fixup++; | ||
| 792 | skb->ip_summed = CHECKSUM_PARTIAL; | ||
| 793 | recalculate_partial_csum = 1; | ||
| 794 | } | ||
| 795 | |||
| 796 | /* A non-CHECKSUM_PARTIAL SKB does not require setup. */ | ||
| 797 | if (skb->ip_summed != CHECKSUM_PARTIAL) | ||
| 798 | return 0; | ||
| 778 | 799 | ||
| 779 | if (skb->protocol != htons(ETH_P_IP)) | 800 | if (skb->protocol != htons(ETH_P_IP)) |
| 780 | goto out; | 801 | goto out; |
| @@ -788,9 +809,23 @@ static int skb_checksum_setup(struct sk_buff *skb) | |||
| 788 | switch (iph->protocol) { | 809 | switch (iph->protocol) { |
| 789 | case IPPROTO_TCP: | 810 | case IPPROTO_TCP: |
| 790 | skb->csum_offset = offsetof(struct tcphdr, check); | 811 | skb->csum_offset = offsetof(struct tcphdr, check); |
| 812 | |||
| 813 | if (recalculate_partial_csum) { | ||
| 814 | struct tcphdr *tcph = (struct tcphdr *)th; | ||
| 815 | tcph->check = ~csum_tcpudp_magic(iph->saddr, iph->daddr, | ||
| 816 | skb->len - iph->ihl*4, | ||
| 817 | IPPROTO_TCP, 0); | ||
| 818 | } | ||
| 791 | break; | 819 | break; |
| 792 | case IPPROTO_UDP: | 820 | case IPPROTO_UDP: |
| 793 | skb->csum_offset = offsetof(struct udphdr, check); | 821 | skb->csum_offset = offsetof(struct udphdr, check); |
| 822 | |||
| 823 | if (recalculate_partial_csum) { | ||
| 824 | struct udphdr *udph = (struct udphdr *)th; | ||
| 825 | udph->check = ~csum_tcpudp_magic(iph->saddr, iph->daddr, | ||
| 826 | skb->len - iph->ihl*4, | ||
| 827 | IPPROTO_UDP, 0); | ||
| 828 | } | ||
| 794 | break; | 829 | break; |
| 795 | default: | 830 | default: |
| 796 | if (net_ratelimit()) | 831 | if (net_ratelimit()) |
| @@ -829,13 +864,11 @@ static int handle_incoming_queue(struct net_device *dev, | |||
| 829 | /* Ethernet work: Delayed to here as it peeks the header. */ | 864 | /* Ethernet work: Delayed to here as it peeks the header. */ |
| 830 | skb->protocol = eth_type_trans(skb, dev); | 865 | skb->protocol = eth_type_trans(skb, dev); |
| 831 | 866 | ||
| 832 | if (skb->ip_summed == CHECKSUM_PARTIAL) { | 867 | if (checksum_setup(dev, skb)) { |
| 833 | if (skb_checksum_setup(skb)) { | 868 | kfree_skb(skb); |
| 834 | kfree_skb(skb); | 869 | packets_dropped++; |
| 835 | packets_dropped++; | 870 | dev->stats.rx_errors++; |
| 836 | dev->stats.rx_errors++; | 871 | continue; |
| 837 | continue; | ||
| 838 | } | ||
| 839 | } | 872 | } |
| 840 | 873 | ||
| 841 | dev->stats.rx_packets++; | 874 | dev->stats.rx_packets++; |
| @@ -1632,12 +1665,59 @@ static void netback_changed(struct xenbus_device *dev, | |||
| 1632 | } | 1665 | } |
| 1633 | } | 1666 | } |
| 1634 | 1667 | ||
| 1668 | static const struct xennet_stat { | ||
| 1669 | char name[ETH_GSTRING_LEN]; | ||
| 1670 | u16 offset; | ||
| 1671 | } xennet_stats[] = { | ||
| 1672 | { | ||
| 1673 | "rx_gso_checksum_fixup", | ||
| 1674 | offsetof(struct netfront_info, rx_gso_checksum_fixup) | ||
| 1675 | }, | ||
| 1676 | }; | ||
| 1677 | |||
| 1678 | static int xennet_get_sset_count(struct net_device *dev, int string_set) | ||
| 1679 | { | ||
| 1680 | switch (string_set) { | ||
| 1681 | case ETH_SS_STATS: | ||
| 1682 | return ARRAY_SIZE(xennet_stats); | ||
| 1683 | default: | ||
| 1684 | return -EINVAL; | ||
| 1685 | } | ||
| 1686 | } | ||
| 1687 | |||
| 1688 | static void xennet_get_ethtool_stats(struct net_device *dev, | ||
| 1689 | struct ethtool_stats *stats, u64 * data) | ||
| 1690 | { | ||
| 1691 | void *np = netdev_priv(dev); | ||
| 1692 | int i; | ||
| 1693 | |||
| 1694 | for (i = 0; i < ARRAY_SIZE(xennet_stats); i++) | ||
| 1695 | data[i] = *(int *)(np + xennet_stats[i].offset); | ||
| 1696 | } | ||
| 1697 | |||
| 1698 | static void xennet_get_strings(struct net_device *dev, u32 stringset, u8 * data) | ||
| 1699 | { | ||
| 1700 | int i; | ||
| 1701 | |||
| 1702 | switch (stringset) { | ||
| 1703 | case ETH_SS_STATS: | ||
| 1704 | for (i = 0; i < ARRAY_SIZE(xennet_stats); i++) | ||
| 1705 | memcpy(data + i * ETH_GSTRING_LEN, | ||
| 1706 | xennet_stats[i].name, ETH_GSTRING_LEN); | ||
| 1707 | break; | ||
| 1708 | } | ||
| 1709 | } | ||
| 1710 | |||
| 1635 | static const struct ethtool_ops xennet_ethtool_ops = | 1711 | static const struct ethtool_ops xennet_ethtool_ops = |
| 1636 | { | 1712 | { |
| 1637 | .set_tx_csum = ethtool_op_set_tx_csum, | 1713 | .set_tx_csum = ethtool_op_set_tx_csum, |
| 1638 | .set_sg = xennet_set_sg, | 1714 | .set_sg = xennet_set_sg, |
| 1639 | .set_tso = xennet_set_tso, | 1715 | .set_tso = xennet_set_tso, |
| 1640 | .get_link = ethtool_op_get_link, | 1716 | .get_link = ethtool_op_get_link, |
| 1717 | |||
| 1718 | .get_sset_count = xennet_get_sset_count, | ||
| 1719 | .get_ethtool_stats = xennet_get_ethtool_stats, | ||
| 1720 | .get_strings = xennet_get_strings, | ||
| 1641 | }; | 1721 | }; |
| 1642 | 1722 | ||
| 1643 | #ifdef CONFIG_SYSFS | 1723 | #ifdef CONFIG_SYSFS |
diff --git a/drivers/parport/share.c b/drivers/parport/share.c index a2d9d1e5926..a848e02e6be 100644 --- a/drivers/parport/share.c +++ b/drivers/parport/share.c | |||
| @@ -678,7 +678,7 @@ void parport_unregister_device(struct pardevice *dev) | |||
| 678 | 678 | ||
| 679 | /* Make sure we haven't left any pointers around in the wait | 679 | /* Make sure we haven't left any pointers around in the wait |
| 680 | * list. */ | 680 | * list. */ |
| 681 | spin_lock (&port->waitlist_lock); | 681 | spin_lock_irq(&port->waitlist_lock); |
| 682 | if (dev->waitprev || dev->waitnext || port->waithead == dev) { | 682 | if (dev->waitprev || dev->waitnext || port->waithead == dev) { |
| 683 | if (dev->waitprev) | 683 | if (dev->waitprev) |
| 684 | dev->waitprev->waitnext = dev->waitnext; | 684 | dev->waitprev->waitnext = dev->waitnext; |
| @@ -689,7 +689,7 @@ void parport_unregister_device(struct pardevice *dev) | |||
| 689 | else | 689 | else |
| 690 | port->waittail = dev->waitprev; | 690 | port->waittail = dev->waitprev; |
| 691 | } | 691 | } |
| 692 | spin_unlock (&port->waitlist_lock); | 692 | spin_unlock_irq(&port->waitlist_lock); |
| 693 | 693 | ||
| 694 | kfree(dev->state); | 694 | kfree(dev->state); |
| 695 | kfree(dev); | 695 | kfree(dev); |
diff --git a/drivers/platform/x86/intel_scu_ipc.c b/drivers/platform/x86/intel_scu_ipc.c index 1752ef006d2..a91d510a798 100644 --- a/drivers/platform/x86/intel_scu_ipc.c +++ b/drivers/platform/x86/intel_scu_ipc.c | |||
| @@ -26,7 +26,6 @@ | |||
| 26 | #include <linux/sfi.h> | 26 | #include <linux/sfi.h> |
| 27 | #include <asm/mrst.h> | 27 | #include <asm/mrst.h> |
| 28 | #include <asm/intel_scu_ipc.h> | 28 | #include <asm/intel_scu_ipc.h> |
| 29 | #include <asm/mrst.h> | ||
| 30 | 29 | ||
| 31 | /* IPC defines the following message types */ | 30 | /* IPC defines the following message types */ |
| 32 | #define IPCMSG_WATCHDOG_TIMER 0xF8 /* Set Kernel Watchdog Threshold */ | 31 | #define IPCMSG_WATCHDOG_TIMER 0xF8 /* Set Kernel Watchdog Threshold */ |
| @@ -161,7 +160,7 @@ static int pwr_reg_rdwr(u16 *addr, u8 *data, u32 count, u32 op, u32 id) | |||
| 161 | { | 160 | { |
| 162 | int i, nc, bytes, d; | 161 | int i, nc, bytes, d; |
| 163 | u32 offset = 0; | 162 | u32 offset = 0; |
| 164 | u32 err = 0; | 163 | int err; |
| 165 | u8 cbuf[IPC_WWBUF_SIZE] = { }; | 164 | u8 cbuf[IPC_WWBUF_SIZE] = { }; |
| 166 | u32 *wbuf = (u32 *)&cbuf; | 165 | u32 *wbuf = (u32 *)&cbuf; |
| 167 | 166 | ||
| @@ -404,7 +403,7 @@ EXPORT_SYMBOL(intel_scu_ipc_update_register); | |||
| 404 | */ | 403 | */ |
| 405 | int intel_scu_ipc_simple_command(int cmd, int sub) | 404 | int intel_scu_ipc_simple_command(int cmd, int sub) |
| 406 | { | 405 | { |
| 407 | u32 err = 0; | 406 | int err; |
| 408 | 407 | ||
| 409 | mutex_lock(&ipclock); | 408 | mutex_lock(&ipclock); |
| 410 | if (ipcdev.pdev == NULL) { | 409 | if (ipcdev.pdev == NULL) { |
| @@ -434,8 +433,7 @@ EXPORT_SYMBOL(intel_scu_ipc_simple_command); | |||
| 434 | int intel_scu_ipc_command(int cmd, int sub, u32 *in, int inlen, | 433 | int intel_scu_ipc_command(int cmd, int sub, u32 *in, int inlen, |
| 435 | u32 *out, int outlen) | 434 | u32 *out, int outlen) |
| 436 | { | 435 | { |
| 437 | u32 err = 0; | 436 | int i, err; |
| 438 | int i = 0; | ||
| 439 | 437 | ||
| 440 | mutex_lock(&ipclock); | 438 | mutex_lock(&ipclock); |
| 441 | if (ipcdev.pdev == NULL) { | 439 | if (ipcdev.pdev == NULL) { |
diff --git a/drivers/platform/x86/intel_scu_ipcutil.c b/drivers/platform/x86/intel_scu_ipcutil.c index ba3231d0819..b93a03259c1 100644 --- a/drivers/platform/x86/intel_scu_ipcutil.c +++ b/drivers/platform/x86/intel_scu_ipcutil.c | |||
| @@ -128,6 +128,6 @@ static void __exit ipc_module_exit(void) | |||
| 128 | module_init(ipc_module_init); | 128 | module_init(ipc_module_init); |
| 129 | module_exit(ipc_module_exit); | 129 | module_exit(ipc_module_exit); |
| 130 | 130 | ||
| 131 | MODULE_LICENSE("GPL V2"); | 131 | MODULE_LICENSE("GPL v2"); |
| 132 | MODULE_DESCRIPTION("Utility driver for intel scu ipc"); | 132 | MODULE_DESCRIPTION("Utility driver for intel scu ipc"); |
| 133 | MODULE_AUTHOR("Sreedhara <sreedhara.ds@intel.com>"); | 133 | MODULE_AUTHOR("Sreedhara <sreedhara.ds@intel.com>"); |
diff --git a/drivers/pps/clients/pps-ktimer.c b/drivers/pps/clients/pps-ktimer.c index 2728469d388..82583b0ff82 100644 --- a/drivers/pps/clients/pps-ktimer.c +++ b/drivers/pps/clients/pps-ktimer.c | |||
| @@ -46,8 +46,6 @@ static void pps_ktimer_event(unsigned long ptr) | |||
| 46 | /* First of all we get the time stamp... */ | 46 | /* First of all we get the time stamp... */ |
| 47 | pps_get_ts(&ts); | 47 | pps_get_ts(&ts); |
| 48 | 48 | ||
| 49 | dev_info(pps->dev, "PPS event at %lu\n", jiffies); | ||
| 50 | |||
| 51 | pps_event(pps, &ts, PPS_CAPTUREASSERT, NULL); | 49 | pps_event(pps, &ts, PPS_CAPTUREASSERT, NULL); |
| 52 | 50 | ||
| 53 | mod_timer(&ktimer, jiffies + HZ); | 51 | mod_timer(&ktimer, jiffies + HZ); |
diff --git a/drivers/pps/clients/pps_parport.c b/drivers/pps/clients/pps_parport.c index 32221efd9ca..c571d6dd8f6 100644 --- a/drivers/pps/clients/pps_parport.c +++ b/drivers/pps/clients/pps_parport.c | |||
| @@ -163,7 +163,7 @@ static void parport_attach(struct parport *port) | |||
| 163 | } | 163 | } |
| 164 | 164 | ||
| 165 | device->pardev = parport_register_device(port, KBUILD_MODNAME, | 165 | device->pardev = parport_register_device(port, KBUILD_MODNAME, |
| 166 | NULL, NULL, parport_irq, 0, device); | 166 | NULL, NULL, parport_irq, PARPORT_FLAG_EXCL, device); |
| 167 | if (!device->pardev) { | 167 | if (!device->pardev) { |
| 168 | pr_err("couldn't register with %s\n", port->name); | 168 | pr_err("couldn't register with %s\n", port->name); |
| 169 | goto err_free; | 169 | goto err_free; |
diff --git a/drivers/pps/generators/pps_gen_parport.c b/drivers/pps/generators/pps_gen_parport.c index 5c32f8dacf5..b93af3ebb5b 100644 --- a/drivers/pps/generators/pps_gen_parport.c +++ b/drivers/pps/generators/pps_gen_parport.c | |||
| @@ -198,7 +198,7 @@ static void parport_attach(struct parport *port) | |||
| 198 | } | 198 | } |
| 199 | 199 | ||
| 200 | device.pardev = parport_register_device(port, KBUILD_MODNAME, | 200 | device.pardev = parport_register_device(port, KBUILD_MODNAME, |
| 201 | NULL, NULL, NULL, 0, &device); | 201 | NULL, NULL, NULL, PARPORT_FLAG_EXCL, &device); |
| 202 | if (!device.pardev) { | 202 | if (!device.pardev) { |
| 203 | pr_err("couldn't register with %s\n", port->name); | 203 | pr_err("couldn't register with %s\n", port->name); |
| 204 | return; | 204 | return; |
diff --git a/drivers/rapidio/rio-scan.c b/drivers/rapidio/rio-scan.c index 467e82bd092..a50391b6ba2 100644 --- a/drivers/rapidio/rio-scan.c +++ b/drivers/rapidio/rio-scan.c | |||
| @@ -943,6 +943,8 @@ static int rio_enum_complete(struct rio_mport *port) | |||
| 943 | * @port: Master port to send transactions | 943 | * @port: Master port to send transactions |
| 944 | * @destid: Current destination ID in network | 944 | * @destid: Current destination ID in network |
| 945 | * @hopcount: Number of hops into the network | 945 | * @hopcount: Number of hops into the network |
| 946 | * @prev: previous rio_dev | ||
| 947 | * @prev_port: previous port number | ||
| 946 | * | 948 | * |
| 947 | * Recursively discovers a RIO network. Transactions are sent via the | 949 | * Recursively discovers a RIO network. Transactions are sent via the |
| 948 | * master port passed in @port. | 950 | * master port passed in @port. |
diff --git a/drivers/rtc/Kconfig b/drivers/rtc/Kconfig index 4941cade319..cdd97192dc6 100644 --- a/drivers/rtc/Kconfig +++ b/drivers/rtc/Kconfig | |||
| @@ -97,18 +97,6 @@ config RTC_INTF_DEV | |||
| 97 | 97 | ||
| 98 | If unsure, say Y. | 98 | If unsure, say Y. |
| 99 | 99 | ||
| 100 | config RTC_INTF_DEV_UIE_EMUL | ||
| 101 | bool "RTC UIE emulation on dev interface" | ||
| 102 | depends on RTC_INTF_DEV | ||
| 103 | help | ||
| 104 | Provides an emulation for RTC_UIE if the underlying rtc chip | ||
| 105 | driver does not expose RTC_UIE ioctls. Those requests generate | ||
| 106 | once-per-second update interrupts, used for synchronization. | ||
| 107 | |||
| 108 | The emulation code will read the time from the hardware | ||
| 109 | clock several times per second, please enable this option | ||
| 110 | only if you know that you really need it. | ||
| 111 | |||
| 112 | config RTC_DRV_TEST | 100 | config RTC_DRV_TEST |
| 113 | tristate "Test driver/device" | 101 | tristate "Test driver/device" |
| 114 | help | 102 | help |
diff --git a/drivers/rtc/interface.c b/drivers/rtc/interface.c index 90384b9f6b2..925006d3310 100644 --- a/drivers/rtc/interface.c +++ b/drivers/rtc/interface.c | |||
| @@ -16,6 +16,9 @@ | |||
| 16 | #include <linux/log2.h> | 16 | #include <linux/log2.h> |
| 17 | #include <linux/workqueue.h> | 17 | #include <linux/workqueue.h> |
| 18 | 18 | ||
| 19 | static int rtc_timer_enqueue(struct rtc_device *rtc, struct rtc_timer *timer); | ||
| 20 | static void rtc_timer_remove(struct rtc_device *rtc, struct rtc_timer *timer); | ||
| 21 | |||
| 19 | static int __rtc_read_time(struct rtc_device *rtc, struct rtc_time *tm) | 22 | static int __rtc_read_time(struct rtc_device *rtc, struct rtc_time *tm) |
| 20 | { | 23 | { |
| 21 | int err; | 24 | int err; |
| @@ -120,12 +123,18 @@ int rtc_read_alarm(struct rtc_device *rtc, struct rtc_wkalrm *alarm) | |||
| 120 | err = mutex_lock_interruptible(&rtc->ops_lock); | 123 | err = mutex_lock_interruptible(&rtc->ops_lock); |
| 121 | if (err) | 124 | if (err) |
| 122 | return err; | 125 | return err; |
| 123 | alarm->enabled = rtc->aie_timer.enabled; | 126 | if (rtc->ops == NULL) |
| 124 | if (alarm->enabled) | 127 | err = -ENODEV; |
| 128 | else if (!rtc->ops->read_alarm) | ||
| 129 | err = -EINVAL; | ||
| 130 | else { | ||
| 131 | memset(alarm, 0, sizeof(struct rtc_wkalrm)); | ||
| 132 | alarm->enabled = rtc->aie_timer.enabled; | ||
| 125 | alarm->time = rtc_ktime_to_tm(rtc->aie_timer.node.expires); | 133 | alarm->time = rtc_ktime_to_tm(rtc->aie_timer.node.expires); |
| 134 | } | ||
| 126 | mutex_unlock(&rtc->ops_lock); | 135 | mutex_unlock(&rtc->ops_lock); |
| 127 | 136 | ||
| 128 | return 0; | 137 | return err; |
| 129 | } | 138 | } |
| 130 | EXPORT_SYMBOL_GPL(rtc_read_alarm); | 139 | EXPORT_SYMBOL_GPL(rtc_read_alarm); |
| 131 | 140 | ||
| @@ -175,16 +184,14 @@ int rtc_set_alarm(struct rtc_device *rtc, struct rtc_wkalrm *alarm) | |||
| 175 | return err; | 184 | return err; |
| 176 | if (rtc->aie_timer.enabled) { | 185 | if (rtc->aie_timer.enabled) { |
| 177 | rtc_timer_remove(rtc, &rtc->aie_timer); | 186 | rtc_timer_remove(rtc, &rtc->aie_timer); |
| 178 | rtc->aie_timer.enabled = 0; | ||
| 179 | } | 187 | } |
| 180 | rtc->aie_timer.node.expires = rtc_tm_to_ktime(alarm->time); | 188 | rtc->aie_timer.node.expires = rtc_tm_to_ktime(alarm->time); |
| 181 | rtc->aie_timer.period = ktime_set(0, 0); | 189 | rtc->aie_timer.period = ktime_set(0, 0); |
| 182 | if (alarm->enabled) { | 190 | if (alarm->enabled) { |
| 183 | rtc->aie_timer.enabled = 1; | 191 | err = rtc_timer_enqueue(rtc, &rtc->aie_timer); |
| 184 | rtc_timer_enqueue(rtc, &rtc->aie_timer); | ||
| 185 | } | 192 | } |
| 186 | mutex_unlock(&rtc->ops_lock); | 193 | mutex_unlock(&rtc->ops_lock); |
| 187 | return 0; | 194 | return err; |
| 188 | } | 195 | } |
| 189 | EXPORT_SYMBOL_GPL(rtc_set_alarm); | 196 | EXPORT_SYMBOL_GPL(rtc_set_alarm); |
| 190 | 197 | ||
| @@ -195,15 +202,15 @@ int rtc_alarm_irq_enable(struct rtc_device *rtc, unsigned int enabled) | |||
| 195 | return err; | 202 | return err; |
| 196 | 203 | ||
| 197 | if (rtc->aie_timer.enabled != enabled) { | 204 | if (rtc->aie_timer.enabled != enabled) { |
| 198 | if (enabled) { | 205 | if (enabled) |
| 199 | rtc->aie_timer.enabled = 1; | 206 | err = rtc_timer_enqueue(rtc, &rtc->aie_timer); |
| 200 | rtc_timer_enqueue(rtc, &rtc->aie_timer); | 207 | else |
| 201 | } else { | ||
| 202 | rtc_timer_remove(rtc, &rtc->aie_timer); | 208 | rtc_timer_remove(rtc, &rtc->aie_timer); |
| 203 | rtc->aie_timer.enabled = 0; | ||
| 204 | } | ||
| 205 | } | 209 | } |
| 206 | 210 | ||
| 211 | if (err) | ||
| 212 | return err; | ||
| 213 | |||
| 207 | if (!rtc->ops) | 214 | if (!rtc->ops) |
| 208 | err = -ENODEV; | 215 | err = -ENODEV; |
| 209 | else if (!rtc->ops->alarm_irq_enable) | 216 | else if (!rtc->ops->alarm_irq_enable) |
| @@ -235,12 +242,9 @@ int rtc_update_irq_enable(struct rtc_device *rtc, unsigned int enabled) | |||
| 235 | now = rtc_tm_to_ktime(tm); | 242 | now = rtc_tm_to_ktime(tm); |
| 236 | rtc->uie_rtctimer.node.expires = ktime_add(now, onesec); | 243 | rtc->uie_rtctimer.node.expires = ktime_add(now, onesec); |
| 237 | rtc->uie_rtctimer.period = ktime_set(1, 0); | 244 | rtc->uie_rtctimer.period = ktime_set(1, 0); |
| 238 | rtc->uie_rtctimer.enabled = 1; | 245 | err = rtc_timer_enqueue(rtc, &rtc->uie_rtctimer); |
| 239 | rtc_timer_enqueue(rtc, &rtc->uie_rtctimer); | 246 | } else |
| 240 | } else { | ||
| 241 | rtc_timer_remove(rtc, &rtc->uie_rtctimer); | 247 | rtc_timer_remove(rtc, &rtc->uie_rtctimer); |
| 242 | rtc->uie_rtctimer.enabled = 0; | ||
| 243 | } | ||
| 244 | 248 | ||
| 245 | out: | 249 | out: |
| 246 | mutex_unlock(&rtc->ops_lock); | 250 | mutex_unlock(&rtc->ops_lock); |
| @@ -488,10 +492,13 @@ EXPORT_SYMBOL_GPL(rtc_irq_set_freq); | |||
| 488 | * Enqueues a timer onto the rtc devices timerqueue and sets | 492 | * Enqueues a timer onto the rtc devices timerqueue and sets |
| 489 | * the next alarm event appropriately. | 493 | * the next alarm event appropriately. |
| 490 | * | 494 | * |
| 495 | * Sets the enabled bit on the added timer. | ||
| 496 | * | ||
| 491 | * Must hold ops_lock for proper serialization of timerqueue | 497 | * Must hold ops_lock for proper serialization of timerqueue |
| 492 | */ | 498 | */ |
| 493 | void rtc_timer_enqueue(struct rtc_device *rtc, struct rtc_timer *timer) | 499 | static int rtc_timer_enqueue(struct rtc_device *rtc, struct rtc_timer *timer) |
| 494 | { | 500 | { |
| 501 | timer->enabled = 1; | ||
| 495 | timerqueue_add(&rtc->timerqueue, &timer->node); | 502 | timerqueue_add(&rtc->timerqueue, &timer->node); |
| 496 | if (&timer->node == timerqueue_getnext(&rtc->timerqueue)) { | 503 | if (&timer->node == timerqueue_getnext(&rtc->timerqueue)) { |
| 497 | struct rtc_wkalrm alarm; | 504 | struct rtc_wkalrm alarm; |
| @@ -501,7 +508,13 @@ void rtc_timer_enqueue(struct rtc_device *rtc, struct rtc_timer *timer) | |||
| 501 | err = __rtc_set_alarm(rtc, &alarm); | 508 | err = __rtc_set_alarm(rtc, &alarm); |
| 502 | if (err == -ETIME) | 509 | if (err == -ETIME) |
| 503 | schedule_work(&rtc->irqwork); | 510 | schedule_work(&rtc->irqwork); |
| 511 | else if (err) { | ||
| 512 | timerqueue_del(&rtc->timerqueue, &timer->node); | ||
| 513 | timer->enabled = 0; | ||
| 514 | return err; | ||
| 515 | } | ||
| 504 | } | 516 | } |
| 517 | return 0; | ||
| 505 | } | 518 | } |
| 506 | 519 | ||
| 507 | /** | 520 | /** |
| @@ -512,13 +525,15 @@ void rtc_timer_enqueue(struct rtc_device *rtc, struct rtc_timer *timer) | |||
| 512 | * Removes a timer onto the rtc devices timerqueue and sets | 525 | * Removes a timer onto the rtc devices timerqueue and sets |
| 513 | * the next alarm event appropriately. | 526 | * the next alarm event appropriately. |
| 514 | * | 527 | * |
| 528 | * Clears the enabled bit on the removed timer. | ||
| 529 | * | ||
| 515 | * Must hold ops_lock for proper serialization of timerqueue | 530 | * Must hold ops_lock for proper serialization of timerqueue |
| 516 | */ | 531 | */ |
| 517 | void rtc_timer_remove(struct rtc_device *rtc, struct rtc_timer *timer) | 532 | static void rtc_timer_remove(struct rtc_device *rtc, struct rtc_timer *timer) |
| 518 | { | 533 | { |
| 519 | struct timerqueue_node *next = timerqueue_getnext(&rtc->timerqueue); | 534 | struct timerqueue_node *next = timerqueue_getnext(&rtc->timerqueue); |
| 520 | timerqueue_del(&rtc->timerqueue, &timer->node); | 535 | timerqueue_del(&rtc->timerqueue, &timer->node); |
| 521 | 536 | timer->enabled = 0; | |
| 522 | if (next == &timer->node) { | 537 | if (next == &timer->node) { |
| 523 | struct rtc_wkalrm alarm; | 538 | struct rtc_wkalrm alarm; |
| 524 | int err; | 539 | int err; |
| @@ -626,8 +641,7 @@ int rtc_timer_start(struct rtc_device *rtc, struct rtc_timer* timer, | |||
| 626 | timer->node.expires = expires; | 641 | timer->node.expires = expires; |
| 627 | timer->period = period; | 642 | timer->period = period; |
| 628 | 643 | ||
| 629 | timer->enabled = 1; | 644 | ret = rtc_timer_enqueue(rtc, timer); |
| 630 | rtc_timer_enqueue(rtc, timer); | ||
| 631 | 645 | ||
| 632 | mutex_unlock(&rtc->ops_lock); | 646 | mutex_unlock(&rtc->ops_lock); |
| 633 | return ret; | 647 | return ret; |
| @@ -645,7 +659,6 @@ int rtc_timer_cancel(struct rtc_device *rtc, struct rtc_timer* timer) | |||
| 645 | mutex_lock(&rtc->ops_lock); | 659 | mutex_lock(&rtc->ops_lock); |
| 646 | if (timer->enabled) | 660 | if (timer->enabled) |
| 647 | rtc_timer_remove(rtc, timer); | 661 | rtc_timer_remove(rtc, timer); |
| 648 | timer->enabled = 0; | ||
| 649 | mutex_unlock(&rtc->ops_lock); | 662 | mutex_unlock(&rtc->ops_lock); |
| 650 | return ret; | 663 | return ret; |
| 651 | } | 664 | } |
diff --git a/drivers/s390/block/dasd_alias.c b/drivers/s390/block/dasd_alias.c index 4155805dcdf..2b771f18d1a 100644 --- a/drivers/s390/block/dasd_alias.c +++ b/drivers/s390/block/dasd_alias.c | |||
| @@ -319,6 +319,9 @@ void dasd_alias_disconnect_device_from_lcu(struct dasd_device *device) | |||
| 319 | 319 | ||
| 320 | private = (struct dasd_eckd_private *) device->private; | 320 | private = (struct dasd_eckd_private *) device->private; |
| 321 | lcu = private->lcu; | 321 | lcu = private->lcu; |
| 322 | /* nothing to do if already disconnected */ | ||
| 323 | if (!lcu) | ||
| 324 | return; | ||
| 322 | device->discipline->get_uid(device, &uid); | 325 | device->discipline->get_uid(device, &uid); |
| 323 | spin_lock_irqsave(&lcu->lock, flags); | 326 | spin_lock_irqsave(&lcu->lock, flags); |
| 324 | list_del_init(&device->alias_list); | 327 | list_del_init(&device->alias_list); |
| @@ -680,6 +683,9 @@ int dasd_alias_remove_device(struct dasd_device *device) | |||
| 680 | 683 | ||
| 681 | private = (struct dasd_eckd_private *) device->private; | 684 | private = (struct dasd_eckd_private *) device->private; |
| 682 | lcu = private->lcu; | 685 | lcu = private->lcu; |
| 686 | /* nothing to do if already removed */ | ||
| 687 | if (!lcu) | ||
| 688 | return 0; | ||
| 683 | spin_lock_irqsave(&lcu->lock, flags); | 689 | spin_lock_irqsave(&lcu->lock, flags); |
| 684 | _remove_device_from_lcu(lcu, device); | 690 | _remove_device_from_lcu(lcu, device); |
| 685 | spin_unlock_irqrestore(&lcu->lock, flags); | 691 | spin_unlock_irqrestore(&lcu->lock, flags); |
diff --git a/drivers/s390/cio/qdio_main.c b/drivers/s390/cio/qdio_main.c index e9fff2b9bce..5640c89cd9d 100644 --- a/drivers/s390/cio/qdio_main.c +++ b/drivers/s390/cio/qdio_main.c | |||
| @@ -476,7 +476,7 @@ static inline void inbound_primed(struct qdio_q *q, int count) | |||
| 476 | static int get_inbound_buffer_frontier(struct qdio_q *q) | 476 | static int get_inbound_buffer_frontier(struct qdio_q *q) |
| 477 | { | 477 | { |
| 478 | int count, stop; | 478 | int count, stop; |
| 479 | unsigned char state; | 479 | unsigned char state = 0; |
| 480 | 480 | ||
| 481 | /* | 481 | /* |
| 482 | * Don't check 128 buffers, as otherwise qdio_inbound_q_moved | 482 | * Don't check 128 buffers, as otherwise qdio_inbound_q_moved |
| @@ -643,7 +643,7 @@ void qdio_inbound_processing(unsigned long data) | |||
| 643 | static int get_outbound_buffer_frontier(struct qdio_q *q) | 643 | static int get_outbound_buffer_frontier(struct qdio_q *q) |
| 644 | { | 644 | { |
| 645 | int count, stop; | 645 | int count, stop; |
| 646 | unsigned char state; | 646 | unsigned char state = 0; |
| 647 | 647 | ||
| 648 | if (need_siga_sync(q)) | 648 | if (need_siga_sync(q)) |
| 649 | if (((queue_type(q) != QDIO_IQDIO_QFMT) && | 649 | if (((queue_type(q) != QDIO_IQDIO_QFMT) && |
diff --git a/drivers/s390/net/netiucv.c b/drivers/s390/net/netiucv.c index 65ebee0a326..b6a6356d09b 100644 --- a/drivers/s390/net/netiucv.c +++ b/drivers/s390/net/netiucv.c | |||
| @@ -565,7 +565,7 @@ static int netiucv_callback_connreq(struct iucv_path *path, | |||
| 565 | struct iucv_event ev; | 565 | struct iucv_event ev; |
| 566 | int rc; | 566 | int rc; |
| 567 | 567 | ||
| 568 | if (memcmp(iucvMagic, ipuser, sizeof(ipuser))) | 568 | if (memcmp(iucvMagic, ipuser, 16)) |
| 569 | /* ipuser must match iucvMagic. */ | 569 | /* ipuser must match iucvMagic. */ |
| 570 | return -EINVAL; | 570 | return -EINVAL; |
| 571 | rc = -EINVAL; | 571 | rc = -EINVAL; |
diff --git a/drivers/s390/net/qeth_core_main.c b/drivers/s390/net/qeth_core_main.c index 29f848bfc12..019ae58ab91 100644 --- a/drivers/s390/net/qeth_core_main.c +++ b/drivers/s390/net/qeth_core_main.c | |||
| @@ -988,16 +988,30 @@ static void qeth_get_channel_path_desc(struct qeth_card *card) | |||
| 988 | chp_dsc = (struct channelPath_dsc *)ccw_device_get_chp_desc(ccwdev, 0); | 988 | chp_dsc = (struct channelPath_dsc *)ccw_device_get_chp_desc(ccwdev, 0); |
| 989 | if (chp_dsc != NULL) { | 989 | if (chp_dsc != NULL) { |
| 990 | /* CHPP field bit 6 == 1 -> single queue */ | 990 | /* CHPP field bit 6 == 1 -> single queue */ |
| 991 | if ((chp_dsc->chpp & 0x02) == 0x02) | 991 | if ((chp_dsc->chpp & 0x02) == 0x02) { |
| 992 | if ((atomic_read(&card->qdio.state) != | ||
| 993 | QETH_QDIO_UNINITIALIZED) && | ||
| 994 | (card->qdio.no_out_queues == 4)) | ||
| 995 | /* change from 4 to 1 outbound queues */ | ||
| 996 | qeth_free_qdio_buffers(card); | ||
| 992 | card->qdio.no_out_queues = 1; | 997 | card->qdio.no_out_queues = 1; |
| 998 | if (card->qdio.default_out_queue != 0) | ||
| 999 | dev_info(&card->gdev->dev, | ||
| 1000 | "Priority Queueing not supported\n"); | ||
| 1001 | card->qdio.default_out_queue = 0; | ||
| 1002 | } else { | ||
| 1003 | if ((atomic_read(&card->qdio.state) != | ||
| 1004 | QETH_QDIO_UNINITIALIZED) && | ||
| 1005 | (card->qdio.no_out_queues == 1)) { | ||
| 1006 | /* change from 1 to 4 outbound queues */ | ||
| 1007 | qeth_free_qdio_buffers(card); | ||
| 1008 | card->qdio.default_out_queue = 2; | ||
| 1009 | } | ||
| 1010 | card->qdio.no_out_queues = 4; | ||
| 1011 | } | ||
| 993 | card->info.func_level = 0x4100 + chp_dsc->desc; | 1012 | card->info.func_level = 0x4100 + chp_dsc->desc; |
| 994 | kfree(chp_dsc); | 1013 | kfree(chp_dsc); |
| 995 | } | 1014 | } |
| 996 | if (card->qdio.no_out_queues == 1) { | ||
| 997 | card->qdio.default_out_queue = 0; | ||
| 998 | dev_info(&card->gdev->dev, | ||
| 999 | "Priority Queueing not supported\n"); | ||
| 1000 | } | ||
| 1001 | QETH_DBF_TEXT_(SETUP, 2, "nr:%x", card->qdio.no_out_queues); | 1015 | QETH_DBF_TEXT_(SETUP, 2, "nr:%x", card->qdio.no_out_queues); |
| 1002 | QETH_DBF_TEXT_(SETUP, 2, "lvl:%02x", card->info.func_level); | 1016 | QETH_DBF_TEXT_(SETUP, 2, "lvl:%02x", card->info.func_level); |
| 1003 | return; | 1017 | return; |
| @@ -1832,33 +1846,6 @@ static inline int qeth_get_initial_mtu_for_card(struct qeth_card *card) | |||
| 1832 | } | 1846 | } |
| 1833 | } | 1847 | } |
| 1834 | 1848 | ||
| 1835 | static inline int qeth_get_max_mtu_for_card(int cardtype) | ||
| 1836 | { | ||
| 1837 | switch (cardtype) { | ||
| 1838 | |||
| 1839 | case QETH_CARD_TYPE_UNKNOWN: | ||
| 1840 | case QETH_CARD_TYPE_OSD: | ||
| 1841 | case QETH_CARD_TYPE_OSN: | ||
| 1842 | case QETH_CARD_TYPE_OSM: | ||
| 1843 | case QETH_CARD_TYPE_OSX: | ||
| 1844 | return 61440; | ||
| 1845 | case QETH_CARD_TYPE_IQD: | ||
| 1846 | return 57344; | ||
| 1847 | default: | ||
| 1848 | return 1500; | ||
| 1849 | } | ||
| 1850 | } | ||
| 1851 | |||
| 1852 | static inline int qeth_get_mtu_out_of_mpc(int cardtype) | ||
| 1853 | { | ||
| 1854 | switch (cardtype) { | ||
| 1855 | case QETH_CARD_TYPE_IQD: | ||
| 1856 | return 1; | ||
| 1857 | default: | ||
| 1858 | return 0; | ||
| 1859 | } | ||
| 1860 | } | ||
| 1861 | |||
| 1862 | static inline int qeth_get_mtu_outof_framesize(int framesize) | 1849 | static inline int qeth_get_mtu_outof_framesize(int framesize) |
| 1863 | { | 1850 | { |
| 1864 | switch (framesize) { | 1851 | switch (framesize) { |
| @@ -1881,10 +1868,9 @@ static inline int qeth_mtu_is_valid(struct qeth_card *card, int mtu) | |||
| 1881 | case QETH_CARD_TYPE_OSD: | 1868 | case QETH_CARD_TYPE_OSD: |
| 1882 | case QETH_CARD_TYPE_OSM: | 1869 | case QETH_CARD_TYPE_OSM: |
| 1883 | case QETH_CARD_TYPE_OSX: | 1870 | case QETH_CARD_TYPE_OSX: |
| 1884 | return ((mtu >= 576) && (mtu <= 61440)); | ||
| 1885 | case QETH_CARD_TYPE_IQD: | 1871 | case QETH_CARD_TYPE_IQD: |
| 1886 | return ((mtu >= 576) && | 1872 | return ((mtu >= 576) && |
| 1887 | (mtu <= card->info.max_mtu + 4096 - 32)); | 1873 | (mtu <= card->info.max_mtu)); |
| 1888 | case QETH_CARD_TYPE_OSN: | 1874 | case QETH_CARD_TYPE_OSN: |
| 1889 | case QETH_CARD_TYPE_UNKNOWN: | 1875 | case QETH_CARD_TYPE_UNKNOWN: |
| 1890 | default: | 1876 | default: |
| @@ -1907,7 +1893,7 @@ static int qeth_ulp_enable_cb(struct qeth_card *card, struct qeth_reply *reply, | |||
| 1907 | memcpy(&card->token.ulp_filter_r, | 1893 | memcpy(&card->token.ulp_filter_r, |
| 1908 | QETH_ULP_ENABLE_RESP_FILTER_TOKEN(iob->data), | 1894 | QETH_ULP_ENABLE_RESP_FILTER_TOKEN(iob->data), |
| 1909 | QETH_MPC_TOKEN_LENGTH); | 1895 | QETH_MPC_TOKEN_LENGTH); |
| 1910 | if (qeth_get_mtu_out_of_mpc(card->info.type)) { | 1896 | if (card->info.type == QETH_CARD_TYPE_IQD) { |
| 1911 | memcpy(&framesize, QETH_ULP_ENABLE_RESP_MAX_MTU(iob->data), 2); | 1897 | memcpy(&framesize, QETH_ULP_ENABLE_RESP_MAX_MTU(iob->data), 2); |
| 1912 | mtu = qeth_get_mtu_outof_framesize(framesize); | 1898 | mtu = qeth_get_mtu_outof_framesize(framesize); |
| 1913 | if (!mtu) { | 1899 | if (!mtu) { |
| @@ -1915,12 +1901,21 @@ static int qeth_ulp_enable_cb(struct qeth_card *card, struct qeth_reply *reply, | |||
| 1915 | QETH_DBF_TEXT_(SETUP, 2, " rc%d", iob->rc); | 1901 | QETH_DBF_TEXT_(SETUP, 2, " rc%d", iob->rc); |
| 1916 | return 0; | 1902 | return 0; |
| 1917 | } | 1903 | } |
| 1918 | card->info.max_mtu = mtu; | 1904 | if (card->info.initial_mtu && (card->info.initial_mtu != mtu)) { |
| 1905 | /* frame size has changed */ | ||
| 1906 | if (card->dev && | ||
| 1907 | ((card->dev->mtu == card->info.initial_mtu) || | ||
| 1908 | (card->dev->mtu > mtu))) | ||
| 1909 | card->dev->mtu = mtu; | ||
| 1910 | qeth_free_qdio_buffers(card); | ||
| 1911 | } | ||
| 1919 | card->info.initial_mtu = mtu; | 1912 | card->info.initial_mtu = mtu; |
| 1913 | card->info.max_mtu = mtu; | ||
| 1920 | card->qdio.in_buf_size = mtu + 2 * PAGE_SIZE; | 1914 | card->qdio.in_buf_size = mtu + 2 * PAGE_SIZE; |
| 1921 | } else { | 1915 | } else { |
| 1922 | card->info.initial_mtu = qeth_get_initial_mtu_for_card(card); | 1916 | card->info.initial_mtu = qeth_get_initial_mtu_for_card(card); |
| 1923 | card->info.max_mtu = qeth_get_max_mtu_for_card(card->info.type); | 1917 | card->info.max_mtu = *(__u16 *)QETH_ULP_ENABLE_RESP_MAX_MTU( |
| 1918 | iob->data); | ||
| 1924 | card->qdio.in_buf_size = QETH_IN_BUF_SIZE_DEFAULT; | 1919 | card->qdio.in_buf_size = QETH_IN_BUF_SIZE_DEFAULT; |
| 1925 | } | 1920 | } |
| 1926 | 1921 | ||
| @@ -3775,6 +3770,47 @@ static inline int qeth_get_qdio_q_format(struct qeth_card *card) | |||
| 3775 | } | 3770 | } |
| 3776 | } | 3771 | } |
| 3777 | 3772 | ||
| 3773 | static void qeth_determine_capabilities(struct qeth_card *card) | ||
| 3774 | { | ||
| 3775 | int rc; | ||
| 3776 | int length; | ||
| 3777 | char *prcd; | ||
| 3778 | struct ccw_device *ddev; | ||
| 3779 | int ddev_offline = 0; | ||
| 3780 | |||
| 3781 | QETH_DBF_TEXT(SETUP, 2, "detcapab"); | ||
| 3782 | ddev = CARD_DDEV(card); | ||
| 3783 | if (!ddev->online) { | ||
| 3784 | ddev_offline = 1; | ||
| 3785 | rc = ccw_device_set_online(ddev); | ||
| 3786 | if (rc) { | ||
| 3787 | QETH_DBF_TEXT_(SETUP, 2, "3err%d", rc); | ||
| 3788 | goto out; | ||
| 3789 | } | ||
| 3790 | } | ||
| 3791 | |||
| 3792 | rc = qeth_read_conf_data(card, (void **) &prcd, &length); | ||
| 3793 | if (rc) { | ||
| 3794 | QETH_DBF_MESSAGE(2, "%s qeth_read_conf_data returned %i\n", | ||
| 3795 | dev_name(&card->gdev->dev), rc); | ||
| 3796 | QETH_DBF_TEXT_(SETUP, 2, "5err%d", rc); | ||
| 3797 | goto out_offline; | ||
| 3798 | } | ||
| 3799 | qeth_configure_unitaddr(card, prcd); | ||
| 3800 | qeth_configure_blkt_default(card, prcd); | ||
| 3801 | kfree(prcd); | ||
| 3802 | |||
| 3803 | rc = qdio_get_ssqd_desc(ddev, &card->ssqd); | ||
| 3804 | if (rc) | ||
| 3805 | QETH_DBF_TEXT_(SETUP, 2, "6err%d", rc); | ||
| 3806 | |||
| 3807 | out_offline: | ||
| 3808 | if (ddev_offline == 1) | ||
| 3809 | ccw_device_set_offline(ddev); | ||
| 3810 | out: | ||
| 3811 | return; | ||
| 3812 | } | ||
| 3813 | |||
| 3778 | static int qeth_qdio_establish(struct qeth_card *card) | 3814 | static int qeth_qdio_establish(struct qeth_card *card) |
| 3779 | { | 3815 | { |
| 3780 | struct qdio_initialize init_data; | 3816 | struct qdio_initialize init_data; |
| @@ -3905,6 +3941,7 @@ int qeth_core_hardsetup_card(struct qeth_card *card) | |||
| 3905 | 3941 | ||
| 3906 | QETH_DBF_TEXT(SETUP, 2, "hrdsetup"); | 3942 | QETH_DBF_TEXT(SETUP, 2, "hrdsetup"); |
| 3907 | atomic_set(&card->force_alloc_skb, 0); | 3943 | atomic_set(&card->force_alloc_skb, 0); |
| 3944 | qeth_get_channel_path_desc(card); | ||
| 3908 | retry: | 3945 | retry: |
| 3909 | if (retries) | 3946 | if (retries) |
| 3910 | QETH_DBF_MESSAGE(2, "%s Retrying to do IDX activates.\n", | 3947 | QETH_DBF_MESSAGE(2, "%s Retrying to do IDX activates.\n", |
| @@ -3933,6 +3970,7 @@ retriable: | |||
| 3933 | else | 3970 | else |
| 3934 | goto retry; | 3971 | goto retry; |
| 3935 | } | 3972 | } |
| 3973 | qeth_determine_capabilities(card); | ||
| 3936 | qeth_init_tokens(card); | 3974 | qeth_init_tokens(card); |
| 3937 | qeth_init_func_level(card); | 3975 | qeth_init_func_level(card); |
| 3938 | rc = qeth_idx_activate_channel(&card->read, qeth_idx_read_cb); | 3976 | rc = qeth_idx_activate_channel(&card->read, qeth_idx_read_cb); |
| @@ -4202,41 +4240,6 @@ void qeth_core_free_discipline(struct qeth_card *card) | |||
| 4202 | card->discipline.ccwgdriver = NULL; | 4240 | card->discipline.ccwgdriver = NULL; |
| 4203 | } | 4241 | } |
| 4204 | 4242 | ||
| 4205 | static void qeth_determine_capabilities(struct qeth_card *card) | ||
| 4206 | { | ||
| 4207 | int rc; | ||
| 4208 | int length; | ||
| 4209 | char *prcd; | ||
| 4210 | |||
| 4211 | QETH_DBF_TEXT(SETUP, 2, "detcapab"); | ||
| 4212 | rc = ccw_device_set_online(CARD_DDEV(card)); | ||
| 4213 | if (rc) { | ||
| 4214 | QETH_DBF_TEXT_(SETUP, 2, "3err%d", rc); | ||
| 4215 | goto out; | ||
| 4216 | } | ||
| 4217 | |||
| 4218 | |||
| 4219 | rc = qeth_read_conf_data(card, (void **) &prcd, &length); | ||
| 4220 | if (rc) { | ||
| 4221 | QETH_DBF_MESSAGE(2, "%s qeth_read_conf_data returned %i\n", | ||
| 4222 | dev_name(&card->gdev->dev), rc); | ||
| 4223 | QETH_DBF_TEXT_(SETUP, 2, "5err%d", rc); | ||
| 4224 | goto out_offline; | ||
| 4225 | } | ||
| 4226 | qeth_configure_unitaddr(card, prcd); | ||
| 4227 | qeth_configure_blkt_default(card, prcd); | ||
| 4228 | kfree(prcd); | ||
| 4229 | |||
| 4230 | rc = qdio_get_ssqd_desc(CARD_DDEV(card), &card->ssqd); | ||
| 4231 | if (rc) | ||
| 4232 | QETH_DBF_TEXT_(SETUP, 2, "6err%d", rc); | ||
| 4233 | |||
| 4234 | out_offline: | ||
| 4235 | ccw_device_set_offline(CARD_DDEV(card)); | ||
| 4236 | out: | ||
| 4237 | return; | ||
| 4238 | } | ||
| 4239 | |||
| 4240 | static int qeth_core_probe_device(struct ccwgroup_device *gdev) | 4243 | static int qeth_core_probe_device(struct ccwgroup_device *gdev) |
| 4241 | { | 4244 | { |
| 4242 | struct qeth_card *card; | 4245 | struct qeth_card *card; |
diff --git a/drivers/s390/net/qeth_l2_main.c b/drivers/s390/net/qeth_l2_main.c index 2ac8f6aff5a..ada0fe78237 100644 --- a/drivers/s390/net/qeth_l2_main.c +++ b/drivers/s390/net/qeth_l2_main.c | |||
| @@ -573,13 +573,13 @@ static int qeth_l2_send_setmac_cb(struct qeth_card *card, | |||
| 573 | case IPA_RC_L2_DUP_LAYER3_MAC: | 573 | case IPA_RC_L2_DUP_LAYER3_MAC: |
| 574 | dev_warn(&card->gdev->dev, | 574 | dev_warn(&card->gdev->dev, |
| 575 | "MAC address %pM already exists\n", | 575 | "MAC address %pM already exists\n", |
| 576 | card->dev->dev_addr); | 576 | cmd->data.setdelmac.mac); |
| 577 | break; | 577 | break; |
| 578 | case IPA_RC_L2_MAC_NOT_AUTH_BY_HYP: | 578 | case IPA_RC_L2_MAC_NOT_AUTH_BY_HYP: |
| 579 | case IPA_RC_L2_MAC_NOT_AUTH_BY_ADP: | 579 | case IPA_RC_L2_MAC_NOT_AUTH_BY_ADP: |
| 580 | dev_warn(&card->gdev->dev, | 580 | dev_warn(&card->gdev->dev, |
| 581 | "MAC address %pM is not authorized\n", | 581 | "MAC address %pM is not authorized\n", |
| 582 | card->dev->dev_addr); | 582 | cmd->data.setdelmac.mac); |
| 583 | break; | 583 | break; |
| 584 | default: | 584 | default: |
| 585 | break; | 585 | break; |
diff --git a/drivers/s390/net/smsgiucv.c b/drivers/s390/net/smsgiucv.c index 65e1cf10494..207b7d74244 100644 --- a/drivers/s390/net/smsgiucv.c +++ b/drivers/s390/net/smsgiucv.c | |||
| @@ -60,7 +60,7 @@ static struct iucv_handler smsg_handler = { | |||
| 60 | static int smsg_path_pending(struct iucv_path *path, u8 ipvmid[8], | 60 | static int smsg_path_pending(struct iucv_path *path, u8 ipvmid[8], |
| 61 | u8 ipuser[16]) | 61 | u8 ipuser[16]) |
| 62 | { | 62 | { |
| 63 | if (strncmp(ipvmid, "*MSG ", sizeof(ipvmid)) != 0) | 63 | if (strncmp(ipvmid, "*MSG ", 8) != 0) |
| 64 | return -EINVAL; | 64 | return -EINVAL; |
| 65 | /* Path pending from *MSG. */ | 65 | /* Path pending from *MSG. */ |
| 66 | return iucv_path_accept(path, &smsg_handler, "SMSGIUCV ", NULL); | 66 | return iucv_path_accept(path, &smsg_handler, "SMSGIUCV ", NULL); |
diff --git a/drivers/scsi/arcmsr/arcmsr.h b/drivers/scsi/arcmsr/arcmsr.h index 475c31ae985..77b26f5b9c3 100644 --- a/drivers/scsi/arcmsr/arcmsr.h +++ b/drivers/scsi/arcmsr/arcmsr.h | |||
| @@ -2,7 +2,7 @@ | |||
| 2 | ******************************************************************************* | 2 | ******************************************************************************* |
| 3 | ** O.S : Linux | 3 | ** O.S : Linux |
| 4 | ** FILE NAME : arcmsr.h | 4 | ** FILE NAME : arcmsr.h |
| 5 | ** BY : Erich Chen | 5 | ** BY : Nick Cheng |
| 6 | ** Description: SCSI RAID Device Driver for | 6 | ** Description: SCSI RAID Device Driver for |
| 7 | ** ARECA RAID Host adapter | 7 | ** ARECA RAID Host adapter |
| 8 | ******************************************************************************* | 8 | ******************************************************************************* |
| @@ -46,8 +46,12 @@ | |||
| 46 | struct device_attribute; | 46 | struct device_attribute; |
| 47 | /*The limit of outstanding scsi command that firmware can handle*/ | 47 | /*The limit of outstanding scsi command that firmware can handle*/ |
| 48 | #define ARCMSR_MAX_OUTSTANDING_CMD 256 | 48 | #define ARCMSR_MAX_OUTSTANDING_CMD 256 |
| 49 | #define ARCMSR_MAX_FREECCB_NUM 320 | 49 | #ifdef CONFIG_XEN |
| 50 | #define ARCMSR_DRIVER_VERSION "Driver Version 1.20.00.15 2010/02/02" | 50 | #define ARCMSR_MAX_FREECCB_NUM 160 |
| 51 | #else | ||
| 52 | #define ARCMSR_MAX_FREECCB_NUM 320 | ||
| 53 | #endif | ||
| 54 | #define ARCMSR_DRIVER_VERSION "Driver Version 1.20.00.15 2010/08/05" | ||
| 51 | #define ARCMSR_SCSI_INITIATOR_ID 255 | 55 | #define ARCMSR_SCSI_INITIATOR_ID 255 |
| 52 | #define ARCMSR_MAX_XFER_SECTORS 512 | 56 | #define ARCMSR_MAX_XFER_SECTORS 512 |
| 53 | #define ARCMSR_MAX_XFER_SECTORS_B 4096 | 57 | #define ARCMSR_MAX_XFER_SECTORS_B 4096 |
| @@ -60,7 +64,6 @@ struct device_attribute; | |||
| 60 | #define ARCMSR_MAX_HBB_POSTQUEUE 264 | 64 | #define ARCMSR_MAX_HBB_POSTQUEUE 264 |
| 61 | #define ARCMSR_MAX_XFER_LEN 0x26000 /* 152K */ | 65 | #define ARCMSR_MAX_XFER_LEN 0x26000 /* 152K */ |
| 62 | #define ARCMSR_CDB_SG_PAGE_LENGTH 256 | 66 | #define ARCMSR_CDB_SG_PAGE_LENGTH 256 |
| 63 | #define SCSI_CMD_ARECA_SPECIFIC 0xE1 | ||
| 64 | #ifndef PCI_DEVICE_ID_ARECA_1880 | 67 | #ifndef PCI_DEVICE_ID_ARECA_1880 |
| 65 | #define PCI_DEVICE_ID_ARECA_1880 0x1880 | 68 | #define PCI_DEVICE_ID_ARECA_1880 0x1880 |
| 66 | #endif | 69 | #endif |
diff --git a/drivers/scsi/arcmsr/arcmsr_attr.c b/drivers/scsi/arcmsr/arcmsr_attr.c index a4e04c50c43..acdae33de52 100644 --- a/drivers/scsi/arcmsr/arcmsr_attr.c +++ b/drivers/scsi/arcmsr/arcmsr_attr.c | |||
| @@ -2,7 +2,7 @@ | |||
| 2 | ******************************************************************************* | 2 | ******************************************************************************* |
| 3 | ** O.S : Linux | 3 | ** O.S : Linux |
| 4 | ** FILE NAME : arcmsr_attr.c | 4 | ** FILE NAME : arcmsr_attr.c |
| 5 | ** BY : Erich Chen | 5 | ** BY : Nick Cheng |
| 6 | ** Description: attributes exported to sysfs and device host | 6 | ** Description: attributes exported to sysfs and device host |
| 7 | ******************************************************************************* | 7 | ******************************************************************************* |
| 8 | ** Copyright (C) 2002 - 2005, Areca Technology Corporation All rights reserved | 8 | ** Copyright (C) 2002 - 2005, Areca Technology Corporation All rights reserved |
diff --git a/drivers/scsi/arcmsr/arcmsr_hba.c b/drivers/scsi/arcmsr/arcmsr_hba.c index 1cadcd6b7da..984bd527c6c 100644 --- a/drivers/scsi/arcmsr/arcmsr_hba.c +++ b/drivers/scsi/arcmsr/arcmsr_hba.c | |||
| @@ -2,7 +2,7 @@ | |||
| 2 | ******************************************************************************* | 2 | ******************************************************************************* |
| 3 | ** O.S : Linux | 3 | ** O.S : Linux |
| 4 | ** FILE NAME : arcmsr_hba.c | 4 | ** FILE NAME : arcmsr_hba.c |
| 5 | ** BY : Erich Chen | 5 | ** BY : Nick Cheng |
| 6 | ** Description: SCSI RAID Device Driver for | 6 | ** Description: SCSI RAID Device Driver for |
| 7 | ** ARECA RAID Host adapter | 7 | ** ARECA RAID Host adapter |
| 8 | ******************************************************************************* | 8 | ******************************************************************************* |
| @@ -76,7 +76,7 @@ MODULE_DESCRIPTION("ARECA (ARC11xx/12xx/16xx/1880) SATA/SAS RAID Host Bus Adapte | |||
| 76 | MODULE_LICENSE("Dual BSD/GPL"); | 76 | MODULE_LICENSE("Dual BSD/GPL"); |
| 77 | MODULE_VERSION(ARCMSR_DRIVER_VERSION); | 77 | MODULE_VERSION(ARCMSR_DRIVER_VERSION); |
| 78 | static int sleeptime = 10; | 78 | static int sleeptime = 10; |
| 79 | static int retrycount = 30; | 79 | static int retrycount = 12; |
| 80 | wait_queue_head_t wait_q; | 80 | wait_queue_head_t wait_q; |
| 81 | static int arcmsr_iop_message_xfer(struct AdapterControlBlock *acb, | 81 | static int arcmsr_iop_message_xfer(struct AdapterControlBlock *acb, |
| 82 | struct scsi_cmnd *cmd); | 82 | struct scsi_cmnd *cmd); |
| @@ -187,7 +187,6 @@ int arcmsr_sleep_for_bus_reset(struct scsi_cmnd *cmd) | |||
| 187 | if (isleep > 0) { | 187 | if (isleep > 0) { |
| 188 | msleep(isleep*1000); | 188 | msleep(isleep*1000); |
| 189 | } | 189 | } |
| 190 | printk(KERN_NOTICE "wake-up\n"); | ||
| 191 | return 0; | 190 | return 0; |
| 192 | } | 191 | } |
| 193 | 192 | ||
| @@ -921,7 +920,6 @@ static void arcmsr_report_ccb_state(struct AdapterControlBlock *acb, | |||
| 921 | } | 920 | } |
| 922 | 921 | ||
| 923 | static void arcmsr_drain_donequeue(struct AdapterControlBlock *acb, struct CommandControlBlock *pCCB, bool error) | 922 | static void arcmsr_drain_donequeue(struct AdapterControlBlock *acb, struct CommandControlBlock *pCCB, bool error) |
| 924 | |||
| 925 | { | 923 | { |
| 926 | int id, lun; | 924 | int id, lun; |
| 927 | if ((pCCB->acb != acb) || (pCCB->startdone != ARCMSR_CCB_START)) { | 925 | if ((pCCB->acb != acb) || (pCCB->startdone != ARCMSR_CCB_START)) { |
| @@ -948,7 +946,7 @@ static void arcmsr_drain_donequeue(struct AdapterControlBlock *acb, struct Comma | |||
| 948 | , pCCB->startdone | 946 | , pCCB->startdone |
| 949 | , atomic_read(&acb->ccboutstandingcount)); | 947 | , atomic_read(&acb->ccboutstandingcount)); |
| 950 | return; | 948 | return; |
| 951 | } | 949 | } |
| 952 | arcmsr_report_ccb_state(acb, pCCB, error); | 950 | arcmsr_report_ccb_state(acb, pCCB, error); |
| 953 | } | 951 | } |
| 954 | 952 | ||
| @@ -981,7 +979,7 @@ static void arcmsr_done4abort_postqueue(struct AdapterControlBlock *acb) | |||
| 981 | case ACB_ADAPTER_TYPE_B: { | 979 | case ACB_ADAPTER_TYPE_B: { |
| 982 | struct MessageUnit_B *reg = acb->pmuB; | 980 | struct MessageUnit_B *reg = acb->pmuB; |
| 983 | /*clear all outbound posted Q*/ | 981 | /*clear all outbound posted Q*/ |
| 984 | writel(ARCMSR_DOORBELL_INT_CLEAR_PATTERN, ®->iop2drv_doorbell); /* clear doorbell interrupt */ | 982 | writel(ARCMSR_DOORBELL_INT_CLEAR_PATTERN, reg->iop2drv_doorbell); /* clear doorbell interrupt */ |
| 985 | for (i = 0; i < ARCMSR_MAX_HBB_POSTQUEUE; i++) { | 983 | for (i = 0; i < ARCMSR_MAX_HBB_POSTQUEUE; i++) { |
| 986 | if ((flag_ccb = readl(®->done_qbuffer[i])) != 0) { | 984 | if ((flag_ccb = readl(®->done_qbuffer[i])) != 0) { |
| 987 | writel(0, ®->done_qbuffer[i]); | 985 | writel(0, ®->done_qbuffer[i]); |
| @@ -1511,7 +1509,6 @@ static void arcmsr_hba_postqueue_isr(struct AdapterControlBlock *acb) | |||
| 1511 | arcmsr_drain_donequeue(acb, pCCB, error); | 1509 | arcmsr_drain_donequeue(acb, pCCB, error); |
| 1512 | } | 1510 | } |
| 1513 | } | 1511 | } |
| 1514 | |||
| 1515 | static void arcmsr_hbb_postqueue_isr(struct AdapterControlBlock *acb) | 1512 | static void arcmsr_hbb_postqueue_isr(struct AdapterControlBlock *acb) |
| 1516 | { | 1513 | { |
| 1517 | uint32_t index; | 1514 | uint32_t index; |
| @@ -2106,10 +2103,6 @@ static int arcmsr_queue_command_lck(struct scsi_cmnd *cmd, | |||
| 2106 | if (atomic_read(&acb->ccboutstandingcount) >= | 2103 | if (atomic_read(&acb->ccboutstandingcount) >= |
| 2107 | ARCMSR_MAX_OUTSTANDING_CMD) | 2104 | ARCMSR_MAX_OUTSTANDING_CMD) |
| 2108 | return SCSI_MLQUEUE_HOST_BUSY; | 2105 | return SCSI_MLQUEUE_HOST_BUSY; |
| 2109 | if ((scsicmd == SCSI_CMD_ARECA_SPECIFIC)) { | ||
| 2110 | printk(KERN_NOTICE "Receiveing SCSI_CMD_ARECA_SPECIFIC command..\n"); | ||
| 2111 | return 0; | ||
| 2112 | } | ||
| 2113 | ccb = arcmsr_get_freeccb(acb); | 2106 | ccb = arcmsr_get_freeccb(acb); |
| 2114 | if (!ccb) | 2107 | if (!ccb) |
| 2115 | return SCSI_MLQUEUE_HOST_BUSY; | 2108 | return SCSI_MLQUEUE_HOST_BUSY; |
| @@ -2393,6 +2386,7 @@ static int arcmsr_polling_hbb_ccbdone(struct AdapterControlBlock *acb, | |||
| 2393 | int index, rtn; | 2386 | int index, rtn; |
| 2394 | bool error; | 2387 | bool error; |
| 2395 | polling_hbb_ccb_retry: | 2388 | polling_hbb_ccb_retry: |
| 2389 | |||
| 2396 | poll_count++; | 2390 | poll_count++; |
| 2397 | /* clear doorbell interrupt */ | 2391 | /* clear doorbell interrupt */ |
| 2398 | writel(ARCMSR_DOORBELL_INT_CLEAR_PATTERN, reg->iop2drv_doorbell); | 2392 | writel(ARCMSR_DOORBELL_INT_CLEAR_PATTERN, reg->iop2drv_doorbell); |
| @@ -2663,6 +2657,7 @@ static void arcmsr_request_hba_device_map(struct AdapterControlBlock *acb) | |||
| 2663 | { | 2657 | { |
| 2664 | struct MessageUnit_A __iomem *reg = acb->pmuA; | 2658 | struct MessageUnit_A __iomem *reg = acb->pmuA; |
| 2665 | if (unlikely(atomic_read(&acb->rq_map_token) == 0) || ((acb->acb_flags & ACB_F_BUS_RESET) != 0 ) || ((acb->acb_flags & ACB_F_ABORT) != 0 )){ | 2659 | if (unlikely(atomic_read(&acb->rq_map_token) == 0) || ((acb->acb_flags & ACB_F_BUS_RESET) != 0 ) || ((acb->acb_flags & ACB_F_ABORT) != 0 )){ |
| 2660 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); | ||
| 2666 | return; | 2661 | return; |
| 2667 | } else { | 2662 | } else { |
| 2668 | acb->fw_flag = FW_NORMAL; | 2663 | acb->fw_flag = FW_NORMAL; |
| @@ -2670,8 +2665,10 @@ static void arcmsr_request_hba_device_map(struct AdapterControlBlock *acb) | |||
| 2670 | atomic_set(&acb->rq_map_token, 16); | 2665 | atomic_set(&acb->rq_map_token, 16); |
| 2671 | } | 2666 | } |
| 2672 | atomic_set(&acb->ante_token_value, atomic_read(&acb->rq_map_token)); | 2667 | atomic_set(&acb->ante_token_value, atomic_read(&acb->rq_map_token)); |
| 2673 | if (atomic_dec_and_test(&acb->rq_map_token)) | 2668 | if (atomic_dec_and_test(&acb->rq_map_token)) { |
| 2669 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); | ||
| 2674 | return; | 2670 | return; |
| 2671 | } | ||
| 2675 | writel(ARCMSR_INBOUND_MESG0_GET_CONFIG, ®->inbound_msgaddr0); | 2672 | writel(ARCMSR_INBOUND_MESG0_GET_CONFIG, ®->inbound_msgaddr0); |
| 2676 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); | 2673 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); |
| 2677 | } | 2674 | } |
| @@ -2682,15 +2679,18 @@ static void arcmsr_request_hbb_device_map(struct AdapterControlBlock *acb) | |||
| 2682 | { | 2679 | { |
| 2683 | struct MessageUnit_B __iomem *reg = acb->pmuB; | 2680 | struct MessageUnit_B __iomem *reg = acb->pmuB; |
| 2684 | if (unlikely(atomic_read(&acb->rq_map_token) == 0) || ((acb->acb_flags & ACB_F_BUS_RESET) != 0 ) || ((acb->acb_flags & ACB_F_ABORT) != 0 )){ | 2681 | if (unlikely(atomic_read(&acb->rq_map_token) == 0) || ((acb->acb_flags & ACB_F_BUS_RESET) != 0 ) || ((acb->acb_flags & ACB_F_ABORT) != 0 )){ |
| 2682 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); | ||
| 2685 | return; | 2683 | return; |
| 2686 | } else { | 2684 | } else { |
| 2687 | acb->fw_flag = FW_NORMAL; | 2685 | acb->fw_flag = FW_NORMAL; |
| 2688 | if (atomic_read(&acb->ante_token_value) == atomic_read(&acb->rq_map_token)) { | 2686 | if (atomic_read(&acb->ante_token_value) == atomic_read(&acb->rq_map_token)) { |
| 2689 | atomic_set(&acb->rq_map_token,16); | 2687 | atomic_set(&acb->rq_map_token, 16); |
| 2690 | } | 2688 | } |
| 2691 | atomic_set(&acb->ante_token_value, atomic_read(&acb->rq_map_token)); | 2689 | atomic_set(&acb->ante_token_value, atomic_read(&acb->rq_map_token)); |
| 2692 | if(atomic_dec_and_test(&acb->rq_map_token)) | 2690 | if (atomic_dec_and_test(&acb->rq_map_token)) { |
| 2691 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); | ||
| 2693 | return; | 2692 | return; |
| 2693 | } | ||
| 2694 | writel(ARCMSR_MESSAGE_GET_CONFIG, reg->drv2iop_doorbell); | 2694 | writel(ARCMSR_MESSAGE_GET_CONFIG, reg->drv2iop_doorbell); |
| 2695 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); | 2695 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); |
| 2696 | } | 2696 | } |
| @@ -2701,6 +2701,7 @@ static void arcmsr_request_hbc_device_map(struct AdapterControlBlock *acb) | |||
| 2701 | { | 2701 | { |
| 2702 | struct MessageUnit_C __iomem *reg = acb->pmuC; | 2702 | struct MessageUnit_C __iomem *reg = acb->pmuC; |
| 2703 | if (unlikely(atomic_read(&acb->rq_map_token) == 0) || ((acb->acb_flags & ACB_F_BUS_RESET) != 0) || ((acb->acb_flags & ACB_F_ABORT) != 0)) { | 2703 | if (unlikely(atomic_read(&acb->rq_map_token) == 0) || ((acb->acb_flags & ACB_F_BUS_RESET) != 0) || ((acb->acb_flags & ACB_F_ABORT) != 0)) { |
| 2704 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); | ||
| 2704 | return; | 2705 | return; |
| 2705 | } else { | 2706 | } else { |
| 2706 | acb->fw_flag = FW_NORMAL; | 2707 | acb->fw_flag = FW_NORMAL; |
| @@ -2708,8 +2709,10 @@ static void arcmsr_request_hbc_device_map(struct AdapterControlBlock *acb) | |||
| 2708 | atomic_set(&acb->rq_map_token, 16); | 2709 | atomic_set(&acb->rq_map_token, 16); |
| 2709 | } | 2710 | } |
| 2710 | atomic_set(&acb->ante_token_value, atomic_read(&acb->rq_map_token)); | 2711 | atomic_set(&acb->ante_token_value, atomic_read(&acb->rq_map_token)); |
| 2711 | if (atomic_dec_and_test(&acb->rq_map_token)) | 2712 | if (atomic_dec_and_test(&acb->rq_map_token)) { |
| 2713 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); | ||
| 2712 | return; | 2714 | return; |
| 2715 | } | ||
| 2713 | writel(ARCMSR_INBOUND_MESG0_GET_CONFIG, ®->inbound_msgaddr0); | 2716 | writel(ARCMSR_INBOUND_MESG0_GET_CONFIG, ®->inbound_msgaddr0); |
| 2714 | writel(ARCMSR_HBCMU_DRV2IOP_MESSAGE_CMD_DONE, ®->inbound_doorbell); | 2717 | writel(ARCMSR_HBCMU_DRV2IOP_MESSAGE_CMD_DONE, ®->inbound_doorbell); |
| 2715 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); | 2718 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); |
| @@ -2897,6 +2900,8 @@ static uint8_t arcmsr_iop_reset(struct AdapterControlBlock *acb) | |||
| 2897 | uint32_t intmask_org; | 2900 | uint32_t intmask_org; |
| 2898 | uint8_t rtnval = 0x00; | 2901 | uint8_t rtnval = 0x00; |
| 2899 | int i = 0; | 2902 | int i = 0; |
| 2903 | unsigned long flags; | ||
| 2904 | |||
| 2900 | if (atomic_read(&acb->ccboutstandingcount) != 0) { | 2905 | if (atomic_read(&acb->ccboutstandingcount) != 0) { |
| 2901 | /* disable all outbound interrupt */ | 2906 | /* disable all outbound interrupt */ |
| 2902 | intmask_org = arcmsr_disable_outbound_ints(acb); | 2907 | intmask_org = arcmsr_disable_outbound_ints(acb); |
| @@ -2907,7 +2912,12 @@ static uint8_t arcmsr_iop_reset(struct AdapterControlBlock *acb) | |||
| 2907 | for (i = 0; i < ARCMSR_MAX_FREECCB_NUM; i++) { | 2912 | for (i = 0; i < ARCMSR_MAX_FREECCB_NUM; i++) { |
| 2908 | ccb = acb->pccb_pool[i]; | 2913 | ccb = acb->pccb_pool[i]; |
| 2909 | if (ccb->startdone == ARCMSR_CCB_START) { | 2914 | if (ccb->startdone == ARCMSR_CCB_START) { |
| 2910 | arcmsr_ccb_complete(ccb); | 2915 | scsi_dma_unmap(ccb->pcmd); |
| 2916 | ccb->startdone = ARCMSR_CCB_DONE; | ||
| 2917 | ccb->ccb_flags = 0; | ||
| 2918 | spin_lock_irqsave(&acb->ccblist_lock, flags); | ||
| 2919 | list_add_tail(&ccb->list, &acb->ccb_free_list); | ||
| 2920 | spin_unlock_irqrestore(&acb->ccblist_lock, flags); | ||
| 2911 | } | 2921 | } |
| 2912 | } | 2922 | } |
| 2913 | atomic_set(&acb->ccboutstandingcount, 0); | 2923 | atomic_set(&acb->ccboutstandingcount, 0); |
| @@ -2920,8 +2930,7 @@ static uint8_t arcmsr_iop_reset(struct AdapterControlBlock *acb) | |||
| 2920 | 2930 | ||
| 2921 | static int arcmsr_bus_reset(struct scsi_cmnd *cmd) | 2931 | static int arcmsr_bus_reset(struct scsi_cmnd *cmd) |
| 2922 | { | 2932 | { |
| 2923 | struct AdapterControlBlock *acb = | 2933 | struct AdapterControlBlock *acb; |
| 2924 | (struct AdapterControlBlock *)cmd->device->host->hostdata; | ||
| 2925 | uint32_t intmask_org, outbound_doorbell; | 2934 | uint32_t intmask_org, outbound_doorbell; |
| 2926 | int retry_count = 0; | 2935 | int retry_count = 0; |
| 2927 | int rtn = FAILED; | 2936 | int rtn = FAILED; |
| @@ -2971,31 +2980,16 @@ sleep_again: | |||
| 2971 | atomic_set(&acb->rq_map_token, 16); | 2980 | atomic_set(&acb->rq_map_token, 16); |
| 2972 | atomic_set(&acb->ante_token_value, 16); | 2981 | atomic_set(&acb->ante_token_value, 16); |
| 2973 | acb->fw_flag = FW_NORMAL; | 2982 | acb->fw_flag = FW_NORMAL; |
| 2974 | init_timer(&acb->eternal_timer); | 2983 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); |
| 2975 | acb->eternal_timer.expires = jiffies + msecs_to_jiffies(6*HZ); | ||
| 2976 | acb->eternal_timer.data = (unsigned long) acb; | ||
| 2977 | acb->eternal_timer.function = &arcmsr_request_device_map; | ||
| 2978 | add_timer(&acb->eternal_timer); | ||
| 2979 | acb->acb_flags &= ~ACB_F_BUS_RESET; | 2984 | acb->acb_flags &= ~ACB_F_BUS_RESET; |
| 2980 | rtn = SUCCESS; | 2985 | rtn = SUCCESS; |
| 2981 | printk(KERN_ERR "arcmsr: scsi bus reset eh returns with success\n"); | 2986 | printk(KERN_ERR "arcmsr: scsi bus reset eh returns with success\n"); |
| 2982 | } else { | 2987 | } else { |
| 2983 | acb->acb_flags &= ~ACB_F_BUS_RESET; | 2988 | acb->acb_flags &= ~ACB_F_BUS_RESET; |
| 2984 | if (atomic_read(&acb->rq_map_token) == 0) { | 2989 | atomic_set(&acb->rq_map_token, 16); |
| 2985 | atomic_set(&acb->rq_map_token, 16); | 2990 | atomic_set(&acb->ante_token_value, 16); |
| 2986 | atomic_set(&acb->ante_token_value, 16); | 2991 | acb->fw_flag = FW_NORMAL; |
| 2987 | acb->fw_flag = FW_NORMAL; | 2992 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6*HZ)); |
| 2988 | init_timer(&acb->eternal_timer); | ||
| 2989 | acb->eternal_timer.expires = jiffies + msecs_to_jiffies(6*HZ); | ||
| 2990 | acb->eternal_timer.data = (unsigned long) acb; | ||
| 2991 | acb->eternal_timer.function = &arcmsr_request_device_map; | ||
| 2992 | add_timer(&acb->eternal_timer); | ||
| 2993 | } else { | ||
| 2994 | atomic_set(&acb->rq_map_token, 16); | ||
| 2995 | atomic_set(&acb->ante_token_value, 16); | ||
| 2996 | acb->fw_flag = FW_NORMAL; | ||
| 2997 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6*HZ)); | ||
| 2998 | } | ||
| 2999 | rtn = SUCCESS; | 2993 | rtn = SUCCESS; |
| 3000 | } | 2994 | } |
| 3001 | break; | 2995 | break; |
| @@ -3007,21 +3001,10 @@ sleep_again: | |||
| 3007 | rtn = FAILED; | 3001 | rtn = FAILED; |
| 3008 | } else { | 3002 | } else { |
| 3009 | acb->acb_flags &= ~ACB_F_BUS_RESET; | 3003 | acb->acb_flags &= ~ACB_F_BUS_RESET; |
| 3010 | if (atomic_read(&acb->rq_map_token) == 0) { | 3004 | atomic_set(&acb->rq_map_token, 16); |
| 3011 | atomic_set(&acb->rq_map_token, 16); | 3005 | atomic_set(&acb->ante_token_value, 16); |
| 3012 | atomic_set(&acb->ante_token_value, 16); | 3006 | acb->fw_flag = FW_NORMAL; |
| 3013 | acb->fw_flag = FW_NORMAL; | 3007 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); |
| 3014 | init_timer(&acb->eternal_timer); | ||
| 3015 | acb->eternal_timer.expires = jiffies + msecs_to_jiffies(6*HZ); | ||
| 3016 | acb->eternal_timer.data = (unsigned long) acb; | ||
| 3017 | acb->eternal_timer.function = &arcmsr_request_device_map; | ||
| 3018 | add_timer(&acb->eternal_timer); | ||
| 3019 | } else { | ||
| 3020 | atomic_set(&acb->rq_map_token, 16); | ||
| 3021 | atomic_set(&acb->ante_token_value, 16); | ||
| 3022 | acb->fw_flag = FW_NORMAL; | ||
| 3023 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6*HZ)); | ||
| 3024 | } | ||
| 3025 | rtn = SUCCESS; | 3008 | rtn = SUCCESS; |
| 3026 | } | 3009 | } |
| 3027 | break; | 3010 | break; |
| @@ -3067,31 +3050,16 @@ sleep: | |||
| 3067 | atomic_set(&acb->rq_map_token, 16); | 3050 | atomic_set(&acb->rq_map_token, 16); |
| 3068 | atomic_set(&acb->ante_token_value, 16); | 3051 | atomic_set(&acb->ante_token_value, 16); |
| 3069 | acb->fw_flag = FW_NORMAL; | 3052 | acb->fw_flag = FW_NORMAL; |
| 3070 | init_timer(&acb->eternal_timer); | 3053 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6 * HZ)); |
| 3071 | acb->eternal_timer.expires = jiffies + msecs_to_jiffies(6 * HZ); | ||
| 3072 | acb->eternal_timer.data = (unsigned long) acb; | ||
| 3073 | acb->eternal_timer.function = &arcmsr_request_device_map; | ||
| 3074 | add_timer(&acb->eternal_timer); | ||
| 3075 | acb->acb_flags &= ~ACB_F_BUS_RESET; | 3054 | acb->acb_flags &= ~ACB_F_BUS_RESET; |
| 3076 | rtn = SUCCESS; | 3055 | rtn = SUCCESS; |
| 3077 | printk(KERN_ERR "arcmsr: scsi bus reset eh returns with success\n"); | 3056 | printk(KERN_ERR "arcmsr: scsi bus reset eh returns with success\n"); |
| 3078 | } else { | 3057 | } else { |
| 3079 | acb->acb_flags &= ~ACB_F_BUS_RESET; | 3058 | acb->acb_flags &= ~ACB_F_BUS_RESET; |
| 3080 | if (atomic_read(&acb->rq_map_token) == 0) { | 3059 | atomic_set(&acb->rq_map_token, 16); |
| 3081 | atomic_set(&acb->rq_map_token, 16); | 3060 | atomic_set(&acb->ante_token_value, 16); |
| 3082 | atomic_set(&acb->ante_token_value, 16); | 3061 | acb->fw_flag = FW_NORMAL; |
| 3083 | acb->fw_flag = FW_NORMAL; | 3062 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6*HZ)); |
| 3084 | init_timer(&acb->eternal_timer); | ||
| 3085 | acb->eternal_timer.expires = jiffies + msecs_to_jiffies(6*HZ); | ||
| 3086 | acb->eternal_timer.data = (unsigned long) acb; | ||
| 3087 | acb->eternal_timer.function = &arcmsr_request_device_map; | ||
| 3088 | add_timer(&acb->eternal_timer); | ||
| 3089 | } else { | ||
| 3090 | atomic_set(&acb->rq_map_token, 16); | ||
| 3091 | atomic_set(&acb->ante_token_value, 16); | ||
| 3092 | acb->fw_flag = FW_NORMAL; | ||
| 3093 | mod_timer(&acb->eternal_timer, jiffies + msecs_to_jiffies(6*HZ)); | ||
| 3094 | } | ||
| 3095 | rtn = SUCCESS; | 3063 | rtn = SUCCESS; |
| 3096 | } | 3064 | } |
| 3097 | break; | 3065 | break; |
diff --git a/drivers/scsi/libsas/sas_scsi_host.c b/drivers/scsi/libsas/sas_scsi_host.c index 5815cbeb27a..9a7aaf5f131 100644 --- a/drivers/scsi/libsas/sas_scsi_host.c +++ b/drivers/scsi/libsas/sas_scsi_host.c | |||
| @@ -646,6 +646,7 @@ void sas_scsi_recover_host(struct Scsi_Host *shost) | |||
| 646 | 646 | ||
| 647 | spin_lock_irqsave(shost->host_lock, flags); | 647 | spin_lock_irqsave(shost->host_lock, flags); |
| 648 | list_splice_init(&shost->eh_cmd_q, &eh_work_q); | 648 | list_splice_init(&shost->eh_cmd_q, &eh_work_q); |
| 649 | shost->host_eh_scheduled = 0; | ||
| 649 | spin_unlock_irqrestore(shost->host_lock, flags); | 650 | spin_unlock_irqrestore(shost->host_lock, flags); |
| 650 | 651 | ||
| 651 | SAS_DPRINTK("Enter %s\n", __func__); | 652 | SAS_DPRINTK("Enter %s\n", __func__); |
diff --git a/drivers/scsi/mpt2sas/mpt2sas_base.c b/drivers/scsi/mpt2sas/mpt2sas_base.c index b2a817055b8..9ead0399808 100644 --- a/drivers/scsi/mpt2sas/mpt2sas_base.c +++ b/drivers/scsi/mpt2sas/mpt2sas_base.c | |||
| @@ -2176,9 +2176,9 @@ _base_allocate_memory_pools(struct MPT2SAS_ADAPTER *ioc, int sleep_flag) | |||
| 2176 | /* adjust hba_queue_depth, reply_free_queue_depth, | 2176 | /* adjust hba_queue_depth, reply_free_queue_depth, |
| 2177 | * and queue_size | 2177 | * and queue_size |
| 2178 | */ | 2178 | */ |
| 2179 | ioc->hba_queue_depth -= queue_diff; | 2179 | ioc->hba_queue_depth -= (queue_diff / 2); |
| 2180 | ioc->reply_free_queue_depth -= queue_diff; | 2180 | ioc->reply_free_queue_depth -= (queue_diff / 2); |
| 2181 | queue_size -= queue_diff; | 2181 | queue_size = facts->MaxReplyDescriptorPostQueueDepth; |
| 2182 | } | 2182 | } |
| 2183 | ioc->reply_post_queue_depth = queue_size; | 2183 | ioc->reply_post_queue_depth = queue_size; |
| 2184 | 2184 | ||
| @@ -3941,6 +3941,8 @@ mpt2sas_base_detach(struct MPT2SAS_ADAPTER *ioc) | |||
| 3941 | static void | 3941 | static void |
| 3942 | _base_reset_handler(struct MPT2SAS_ADAPTER *ioc, int reset_phase) | 3942 | _base_reset_handler(struct MPT2SAS_ADAPTER *ioc, int reset_phase) |
| 3943 | { | 3943 | { |
| 3944 | mpt2sas_scsih_reset_handler(ioc, reset_phase); | ||
| 3945 | mpt2sas_ctl_reset_handler(ioc, reset_phase); | ||
| 3944 | switch (reset_phase) { | 3946 | switch (reset_phase) { |
| 3945 | case MPT2_IOC_PRE_RESET: | 3947 | case MPT2_IOC_PRE_RESET: |
| 3946 | dtmprintk(ioc, printk(MPT2SAS_INFO_FMT "%s: " | 3948 | dtmprintk(ioc, printk(MPT2SAS_INFO_FMT "%s: " |
| @@ -3971,8 +3973,6 @@ _base_reset_handler(struct MPT2SAS_ADAPTER *ioc, int reset_phase) | |||
| 3971 | "MPT2_IOC_DONE_RESET\n", ioc->name, __func__)); | 3973 | "MPT2_IOC_DONE_RESET\n", ioc->name, __func__)); |
| 3972 | break; | 3974 | break; |
| 3973 | } | 3975 | } |
| 3974 | mpt2sas_scsih_reset_handler(ioc, reset_phase); | ||
| 3975 | mpt2sas_ctl_reset_handler(ioc, reset_phase); | ||
| 3976 | } | 3976 | } |
| 3977 | 3977 | ||
| 3978 | /** | 3978 | /** |
| @@ -4026,6 +4026,7 @@ mpt2sas_base_hard_reset_handler(struct MPT2SAS_ADAPTER *ioc, int sleep_flag, | |||
| 4026 | { | 4026 | { |
| 4027 | int r; | 4027 | int r; |
| 4028 | unsigned long flags; | 4028 | unsigned long flags; |
| 4029 | u8 pe_complete = ioc->wait_for_port_enable_to_complete; | ||
| 4029 | 4030 | ||
| 4030 | dtmprintk(ioc, printk(MPT2SAS_INFO_FMT "%s: enter\n", ioc->name, | 4031 | dtmprintk(ioc, printk(MPT2SAS_INFO_FMT "%s: enter\n", ioc->name, |
| 4031 | __func__)); | 4032 | __func__)); |
| @@ -4068,6 +4069,14 @@ mpt2sas_base_hard_reset_handler(struct MPT2SAS_ADAPTER *ioc, int sleep_flag, | |||
| 4068 | if (r) | 4069 | if (r) |
| 4069 | goto out; | 4070 | goto out; |
| 4070 | _base_reset_handler(ioc, MPT2_IOC_AFTER_RESET); | 4071 | _base_reset_handler(ioc, MPT2_IOC_AFTER_RESET); |
| 4072 | |||
| 4073 | /* If this hard reset is called while port enable is active, then | ||
| 4074 | * there is no reason to call make_ioc_operational | ||
| 4075 | */ | ||
| 4076 | if (pe_complete) { | ||
| 4077 | r = -EFAULT; | ||
| 4078 | goto out; | ||
| 4079 | } | ||
| 4071 | r = _base_make_ioc_operational(ioc, sleep_flag); | 4080 | r = _base_make_ioc_operational(ioc, sleep_flag); |
| 4072 | if (!r) | 4081 | if (!r) |
| 4073 | _base_reset_handler(ioc, MPT2_IOC_DONE_RESET); | 4082 | _base_reset_handler(ioc, MPT2_IOC_DONE_RESET); |
diff --git a/drivers/scsi/mpt2sas/mpt2sas_scsih.c b/drivers/scsi/mpt2sas/mpt2sas_scsih.c index eda347c5797..5ded3db6e31 100644 --- a/drivers/scsi/mpt2sas/mpt2sas_scsih.c +++ b/drivers/scsi/mpt2sas/mpt2sas_scsih.c | |||
| @@ -819,7 +819,7 @@ _scsih_is_end_device(u32 device_info) | |||
| 819 | } | 819 | } |
| 820 | 820 | ||
| 821 | /** | 821 | /** |
| 822 | * mptscsih_get_scsi_lookup - returns scmd entry | 822 | * _scsih_scsi_lookup_get - returns scmd entry |
| 823 | * @ioc: per adapter object | 823 | * @ioc: per adapter object |
| 824 | * @smid: system request message index | 824 | * @smid: system request message index |
| 825 | * | 825 | * |
| @@ -832,6 +832,28 @@ _scsih_scsi_lookup_get(struct MPT2SAS_ADAPTER *ioc, u16 smid) | |||
| 832 | } | 832 | } |
| 833 | 833 | ||
| 834 | /** | 834 | /** |
| 835 | * _scsih_scsi_lookup_get_clear - returns scmd entry | ||
| 836 | * @ioc: per adapter object | ||
| 837 | * @smid: system request message index | ||
| 838 | * | ||
| 839 | * Returns the smid stored scmd pointer. | ||
| 840 | * Then will derefrence the stored scmd pointer. | ||
| 841 | */ | ||
| 842 | static inline struct scsi_cmnd * | ||
| 843 | _scsih_scsi_lookup_get_clear(struct MPT2SAS_ADAPTER *ioc, u16 smid) | ||
| 844 | { | ||
| 845 | unsigned long flags; | ||
| 846 | struct scsi_cmnd *scmd; | ||
| 847 | |||
| 848 | spin_lock_irqsave(&ioc->scsi_lookup_lock, flags); | ||
| 849 | scmd = ioc->scsi_lookup[smid - 1].scmd; | ||
| 850 | ioc->scsi_lookup[smid - 1].scmd = NULL; | ||
| 851 | spin_unlock_irqrestore(&ioc->scsi_lookup_lock, flags); | ||
| 852 | |||
| 853 | return scmd; | ||
| 854 | } | ||
| 855 | |||
| 856 | /** | ||
| 835 | * _scsih_scsi_lookup_find_by_scmd - scmd lookup | 857 | * _scsih_scsi_lookup_find_by_scmd - scmd lookup |
| 836 | * @ioc: per adapter object | 858 | * @ioc: per adapter object |
| 837 | * @smid: system request message index | 859 | * @smid: system request message index |
| @@ -2981,9 +3003,6 @@ _scsih_check_topo_delete_events(struct MPT2SAS_ADAPTER *ioc, | |||
| 2981 | u16 handle; | 3003 | u16 handle; |
| 2982 | 3004 | ||
| 2983 | for (i = 0 ; i < event_data->NumEntries; i++) { | 3005 | for (i = 0 ; i < event_data->NumEntries; i++) { |
| 2984 | if (event_data->PHY[i].PhyStatus & | ||
| 2985 | MPI2_EVENT_SAS_TOPO_PHYSTATUS_VACANT) | ||
| 2986 | continue; | ||
| 2987 | handle = le16_to_cpu(event_data->PHY[i].AttachedDevHandle); | 3006 | handle = le16_to_cpu(event_data->PHY[i].AttachedDevHandle); |
| 2988 | if (!handle) | 3007 | if (!handle) |
| 2989 | continue; | 3008 | continue; |
| @@ -3210,7 +3229,7 @@ _scsih_flush_running_cmds(struct MPT2SAS_ADAPTER *ioc) | |||
| 3210 | u16 count = 0; | 3229 | u16 count = 0; |
| 3211 | 3230 | ||
| 3212 | for (smid = 1; smid <= ioc->scsiio_depth; smid++) { | 3231 | for (smid = 1; smid <= ioc->scsiio_depth; smid++) { |
| 3213 | scmd = _scsih_scsi_lookup_get(ioc, smid); | 3232 | scmd = _scsih_scsi_lookup_get_clear(ioc, smid); |
| 3214 | if (!scmd) | 3233 | if (!scmd) |
| 3215 | continue; | 3234 | continue; |
| 3216 | count++; | 3235 | count++; |
| @@ -3804,7 +3823,7 @@ _scsih_io_done(struct MPT2SAS_ADAPTER *ioc, u16 smid, u8 msix_index, u32 reply) | |||
| 3804 | u32 response_code = 0; | 3823 | u32 response_code = 0; |
| 3805 | 3824 | ||
| 3806 | mpi_reply = mpt2sas_base_get_reply_virt_addr(ioc, reply); | 3825 | mpi_reply = mpt2sas_base_get_reply_virt_addr(ioc, reply); |
| 3807 | scmd = _scsih_scsi_lookup_get(ioc, smid); | 3826 | scmd = _scsih_scsi_lookup_get_clear(ioc, smid); |
| 3808 | if (scmd == NULL) | 3827 | if (scmd == NULL) |
| 3809 | return 1; | 3828 | return 1; |
| 3810 | 3829 | ||
| @@ -5005,6 +5024,12 @@ _scsih_sas_device_status_change_event(struct MPT2SAS_ADAPTER *ioc, | |||
| 5005 | event_data); | 5024 | event_data); |
| 5006 | #endif | 5025 | #endif |
| 5007 | 5026 | ||
| 5027 | /* In MPI Revision K (0xC), the internal device reset complete was | ||
| 5028 | * implemented, so avoid setting tm_busy flag for older firmware. | ||
| 5029 | */ | ||
| 5030 | if ((ioc->facts.HeaderVersion >> 8) < 0xC) | ||
| 5031 | return; | ||
| 5032 | |||
| 5008 | if (event_data->ReasonCode != | 5033 | if (event_data->ReasonCode != |
| 5009 | MPI2_EVENT_SAS_DEV_STAT_RC_INTERNAL_DEVICE_RESET && | 5034 | MPI2_EVENT_SAS_DEV_STAT_RC_INTERNAL_DEVICE_RESET && |
| 5010 | event_data->ReasonCode != | 5035 | event_data->ReasonCode != |
| @@ -5099,6 +5124,7 @@ _scsih_sas_broadcast_primative_event(struct MPT2SAS_ADAPTER *ioc, | |||
| 5099 | struct fw_event_work *fw_event) | 5124 | struct fw_event_work *fw_event) |
| 5100 | { | 5125 | { |
| 5101 | struct scsi_cmnd *scmd; | 5126 | struct scsi_cmnd *scmd; |
| 5127 | struct scsi_device *sdev; | ||
| 5102 | u16 smid, handle; | 5128 | u16 smid, handle; |
| 5103 | u32 lun; | 5129 | u32 lun; |
| 5104 | struct MPT2SAS_DEVICE *sas_device_priv_data; | 5130 | struct MPT2SAS_DEVICE *sas_device_priv_data; |
| @@ -5109,12 +5135,17 @@ _scsih_sas_broadcast_primative_event(struct MPT2SAS_ADAPTER *ioc, | |||
| 5109 | Mpi2EventDataSasBroadcastPrimitive_t *event_data = fw_event->event_data; | 5135 | Mpi2EventDataSasBroadcastPrimitive_t *event_data = fw_event->event_data; |
| 5110 | #endif | 5136 | #endif |
| 5111 | u16 ioc_status; | 5137 | u16 ioc_status; |
| 5138 | unsigned long flags; | ||
| 5139 | int r; | ||
| 5140 | |||
| 5112 | dewtprintk(ioc, printk(MPT2SAS_INFO_FMT "broadcast primative: " | 5141 | dewtprintk(ioc, printk(MPT2SAS_INFO_FMT "broadcast primative: " |
| 5113 | "phy number(%d), width(%d)\n", ioc->name, event_data->PhyNum, | 5142 | "phy number(%d), width(%d)\n", ioc->name, event_data->PhyNum, |
| 5114 | event_data->PortWidth)); | 5143 | event_data->PortWidth)); |
| 5115 | dtmprintk(ioc, printk(MPT2SAS_INFO_FMT "%s: enter\n", ioc->name, | 5144 | dtmprintk(ioc, printk(MPT2SAS_INFO_FMT "%s: enter\n", ioc->name, |
| 5116 | __func__)); | 5145 | __func__)); |
| 5117 | 5146 | ||
| 5147 | spin_lock_irqsave(&ioc->scsi_lookup_lock, flags); | ||
| 5148 | ioc->broadcast_aen_busy = 0; | ||
| 5118 | termination_count = 0; | 5149 | termination_count = 0; |
| 5119 | query_count = 0; | 5150 | query_count = 0; |
| 5120 | mpi_reply = ioc->tm_cmds.reply; | 5151 | mpi_reply = ioc->tm_cmds.reply; |
| @@ -5122,7 +5153,8 @@ _scsih_sas_broadcast_primative_event(struct MPT2SAS_ADAPTER *ioc, | |||
| 5122 | scmd = _scsih_scsi_lookup_get(ioc, smid); | 5153 | scmd = _scsih_scsi_lookup_get(ioc, smid); |
| 5123 | if (!scmd) | 5154 | if (!scmd) |
| 5124 | continue; | 5155 | continue; |
| 5125 | sas_device_priv_data = scmd->device->hostdata; | 5156 | sdev = scmd->device; |
| 5157 | sas_device_priv_data = sdev->hostdata; | ||
| 5126 | if (!sas_device_priv_data || !sas_device_priv_data->sas_target) | 5158 | if (!sas_device_priv_data || !sas_device_priv_data->sas_target) |
| 5127 | continue; | 5159 | continue; |
| 5128 | /* skip hidden raid components */ | 5160 | /* skip hidden raid components */ |
| @@ -5138,6 +5170,7 @@ _scsih_sas_broadcast_primative_event(struct MPT2SAS_ADAPTER *ioc, | |||
| 5138 | lun = sas_device_priv_data->lun; | 5170 | lun = sas_device_priv_data->lun; |
| 5139 | query_count++; | 5171 | query_count++; |
| 5140 | 5172 | ||
| 5173 | spin_unlock_irqrestore(&ioc->scsi_lookup_lock, flags); | ||
| 5141 | mpt2sas_scsih_issue_tm(ioc, handle, 0, 0, lun, | 5174 | mpt2sas_scsih_issue_tm(ioc, handle, 0, 0, lun, |
| 5142 | MPI2_SCSITASKMGMT_TASKTYPE_QUERY_TASK, smid, 30, NULL); | 5175 | MPI2_SCSITASKMGMT_TASKTYPE_QUERY_TASK, smid, 30, NULL); |
| 5143 | ioc->tm_cmds.status = MPT2_CMD_NOT_USED; | 5176 | ioc->tm_cmds.status = MPT2_CMD_NOT_USED; |
| @@ -5147,14 +5180,20 @@ _scsih_sas_broadcast_primative_event(struct MPT2SAS_ADAPTER *ioc, | |||
| 5147 | (mpi_reply->ResponseCode == | 5180 | (mpi_reply->ResponseCode == |
| 5148 | MPI2_SCSITASKMGMT_RSP_TM_SUCCEEDED || | 5181 | MPI2_SCSITASKMGMT_RSP_TM_SUCCEEDED || |
| 5149 | mpi_reply->ResponseCode == | 5182 | mpi_reply->ResponseCode == |
| 5150 | MPI2_SCSITASKMGMT_RSP_IO_QUEUED_ON_IOC)) | 5183 | MPI2_SCSITASKMGMT_RSP_IO_QUEUED_ON_IOC)) { |
| 5184 | spin_lock_irqsave(&ioc->scsi_lookup_lock, flags); | ||
| 5151 | continue; | 5185 | continue; |
| 5152 | 5186 | } | |
| 5153 | mpt2sas_scsih_issue_tm(ioc, handle, 0, 0, lun, | 5187 | r = mpt2sas_scsih_issue_tm(ioc, handle, sdev->channel, sdev->id, |
| 5154 | MPI2_SCSITASKMGMT_TASKTYPE_ABRT_TASK_SET, 0, 30, NULL); | 5188 | sdev->lun, MPI2_SCSITASKMGMT_TASKTYPE_ABORT_TASK, smid, 30, |
| 5189 | scmd); | ||
| 5190 | if (r == FAILED) | ||
| 5191 | sdev_printk(KERN_WARNING, sdev, "task abort: FAILED " | ||
| 5192 | "scmd(%p)\n", scmd); | ||
| 5155 | termination_count += le32_to_cpu(mpi_reply->TerminationCount); | 5193 | termination_count += le32_to_cpu(mpi_reply->TerminationCount); |
| 5194 | spin_lock_irqsave(&ioc->scsi_lookup_lock, flags); | ||
| 5156 | } | 5195 | } |
| 5157 | ioc->broadcast_aen_busy = 0; | 5196 | spin_unlock_irqrestore(&ioc->scsi_lookup_lock, flags); |
| 5158 | 5197 | ||
| 5159 | dtmprintk(ioc, printk(MPT2SAS_INFO_FMT | 5198 | dtmprintk(ioc, printk(MPT2SAS_INFO_FMT |
| 5160 | "%s - exit, query_count = %d termination_count = %d\n", | 5199 | "%s - exit, query_count = %d termination_count = %d\n", |
| @@ -6626,6 +6665,7 @@ _scsih_remove(struct pci_dev *pdev) | |||
| 6626 | destroy_workqueue(wq); | 6665 | destroy_workqueue(wq); |
| 6627 | 6666 | ||
| 6628 | /* release all the volumes */ | 6667 | /* release all the volumes */ |
| 6668 | _scsih_ir_shutdown(ioc); | ||
| 6629 | list_for_each_entry_safe(raid_device, next, &ioc->raid_device_list, | 6669 | list_for_each_entry_safe(raid_device, next, &ioc->raid_device_list, |
| 6630 | list) { | 6670 | list) { |
| 6631 | if (raid_device->starget) { | 6671 | if (raid_device->starget) { |
diff --git a/drivers/sh/intc/chip.c b/drivers/sh/intc/chip.c index de885a0f917..f33e2dd9793 100644 --- a/drivers/sh/intc/chip.c +++ b/drivers/sh/intc/chip.c | |||
| @@ -173,7 +173,8 @@ int intc_set_priority(unsigned int irq, unsigned int prio) | |||
| 173 | return 0; | 173 | return 0; |
| 174 | } | 174 | } |
| 175 | 175 | ||
| 176 | #define VALID(x) (x | 0x80) | 176 | #define SENSE_VALID_FLAG 0x80 |
| 177 | #define VALID(x) (x | SENSE_VALID_FLAG) | ||
| 177 | 178 | ||
| 178 | static unsigned char intc_irq_sense_table[IRQ_TYPE_SENSE_MASK + 1] = { | 179 | static unsigned char intc_irq_sense_table[IRQ_TYPE_SENSE_MASK + 1] = { |
| 179 | [IRQ_TYPE_EDGE_FALLING] = VALID(0), | 180 | [IRQ_TYPE_EDGE_FALLING] = VALID(0), |
| @@ -201,7 +202,8 @@ static int intc_set_type(struct irq_data *data, unsigned int type) | |||
| 201 | ihp = intc_find_irq(d->sense, d->nr_sense, irq); | 202 | ihp = intc_find_irq(d->sense, d->nr_sense, irq); |
| 202 | if (ihp) { | 203 | if (ihp) { |
| 203 | addr = INTC_REG(d, _INTC_ADDR_E(ihp->handle), 0); | 204 | addr = INTC_REG(d, _INTC_ADDR_E(ihp->handle), 0); |
| 204 | intc_reg_fns[_INTC_FN(ihp->handle)](addr, ihp->handle, value); | 205 | intc_reg_fns[_INTC_FN(ihp->handle)](addr, ihp->handle, |
| 206 | value & ~SENSE_VALID_FLAG); | ||
| 205 | } | 207 | } |
| 206 | 208 | ||
| 207 | return 0; | 209 | return 0; |
diff --git a/drivers/staging/ath6kl/miscdrv/ar3kps/ar3kpsconfig.c b/drivers/staging/ath6kl/miscdrv/ar3kps/ar3kpsconfig.c index 0e298dba9fc..29b8ab44ea4 100644 --- a/drivers/staging/ath6kl/miscdrv/ar3kps/ar3kpsconfig.c +++ b/drivers/staging/ath6kl/miscdrv/ar3kps/ar3kpsconfig.c | |||
| @@ -360,8 +360,8 @@ int PSSendOps(void *arg) | |||
| 360 | status = 1; | 360 | status = 1; |
| 361 | goto complete; | 361 | goto complete; |
| 362 | } | 362 | } |
| 363 | len = (firmware->size > MAX_BDADDR_FORMAT_LENGTH)? MAX_BDADDR_FORMAT_LENGTH: firmware->size; | 363 | len = min(firmware->size, MAX_BDADDR_FORMAT_LENGTH - 1); |
| 364 | memcpy(config_bdaddr, firmware->data,len); | 364 | memcpy(config_bdaddr, firmware->data, len); |
| 365 | config_bdaddr[len] = '\0'; | 365 | config_bdaddr[len] = '\0'; |
| 366 | write_bdaddr(hdev,config_bdaddr,BDADDR_TYPE_STRING); | 366 | write_bdaddr(hdev,config_bdaddr,BDADDR_TYPE_STRING); |
| 367 | A_RELEASE_FIRMWARE(firmware); | 367 | A_RELEASE_FIRMWARE(firmware); |
diff --git a/drivers/staging/brcm80211/sys/wl_mac80211.c b/drivers/staging/brcm80211/sys/wl_mac80211.c index bdd629d72a7..f1235884cc5 100644 --- a/drivers/staging/brcm80211/sys/wl_mac80211.c +++ b/drivers/staging/brcm80211/sys/wl_mac80211.c | |||
| @@ -209,11 +209,8 @@ static void wl_ops_stop(struct ieee80211_hw *hw) | |||
| 209 | struct wl_info *wl = hw->priv; | 209 | struct wl_info *wl = hw->priv; |
| 210 | ASSERT(wl); | 210 | ASSERT(wl); |
| 211 | WL_LOCK(wl); | 211 | WL_LOCK(wl); |
| 212 | wl_down(wl); | ||
| 213 | ieee80211_stop_queues(hw); | 212 | ieee80211_stop_queues(hw); |
| 214 | WL_UNLOCK(wl); | 213 | WL_UNLOCK(wl); |
| 215 | |||
| 216 | return; | ||
| 217 | } | 214 | } |
| 218 | 215 | ||
| 219 | static int | 216 | static int |
| @@ -246,7 +243,14 @@ wl_ops_add_interface(struct ieee80211_hw *hw, struct ieee80211_vif *vif) | |||
| 246 | static void | 243 | static void |
| 247 | wl_ops_remove_interface(struct ieee80211_hw *hw, struct ieee80211_vif *vif) | 244 | wl_ops_remove_interface(struct ieee80211_hw *hw, struct ieee80211_vif *vif) |
| 248 | { | 245 | { |
| 249 | return; | 246 | struct wl_info *wl; |
| 247 | |||
| 248 | wl = HW_TO_WL(hw); | ||
| 249 | |||
| 250 | /* put driver in down state */ | ||
| 251 | WL_LOCK(wl); | ||
| 252 | wl_down(wl); | ||
| 253 | WL_UNLOCK(wl); | ||
| 250 | } | 254 | } |
| 251 | 255 | ||
| 252 | static int | 256 | static int |
| @@ -779,7 +783,7 @@ static struct wl_info *wl_attach(u16 vendor, u16 device, unsigned long regs, | |||
| 779 | wl_found++; | 783 | wl_found++; |
| 780 | return wl; | 784 | return wl; |
| 781 | 785 | ||
| 782 | fail: | 786 | fail: |
| 783 | wl_free(wl); | 787 | wl_free(wl); |
| 784 | fail1: | 788 | fail1: |
| 785 | return NULL; | 789 | return NULL; |
| @@ -1090,7 +1094,6 @@ wl_pci_probe(struct pci_dev *pdev, const struct pci_device_id *ent) | |||
| 1090 | return 0; | 1094 | return 0; |
| 1091 | } | 1095 | } |
| 1092 | 1096 | ||
| 1093 | #ifdef LINUXSTA_PS | ||
| 1094 | static int wl_suspend(struct pci_dev *pdev, pm_message_t state) | 1097 | static int wl_suspend(struct pci_dev *pdev, pm_message_t state) |
| 1095 | { | 1098 | { |
| 1096 | struct wl_info *wl; | 1099 | struct wl_info *wl; |
| @@ -1105,11 +1108,12 @@ static int wl_suspend(struct pci_dev *pdev, pm_message_t state) | |||
| 1105 | return -ENODEV; | 1108 | return -ENODEV; |
| 1106 | } | 1109 | } |
| 1107 | 1110 | ||
| 1111 | /* only need to flag hw is down for proper resume */ | ||
| 1108 | WL_LOCK(wl); | 1112 | WL_LOCK(wl); |
| 1109 | wl_down(wl); | ||
| 1110 | wl->pub->hw_up = false; | 1113 | wl->pub->hw_up = false; |
| 1111 | WL_UNLOCK(wl); | 1114 | WL_UNLOCK(wl); |
| 1112 | pci_save_state(pdev, wl->pci_psstate); | 1115 | |
| 1116 | pci_save_state(pdev); | ||
| 1113 | pci_disable_device(pdev); | 1117 | pci_disable_device(pdev); |
| 1114 | return pci_set_power_state(pdev, PCI_D3hot); | 1118 | return pci_set_power_state(pdev, PCI_D3hot); |
| 1115 | } | 1119 | } |
| @@ -1133,7 +1137,7 @@ static int wl_resume(struct pci_dev *pdev) | |||
| 1133 | if (err) | 1137 | if (err) |
| 1134 | return err; | 1138 | return err; |
| 1135 | 1139 | ||
| 1136 | pci_restore_state(pdev, wl->pci_psstate); | 1140 | pci_restore_state(pdev); |
| 1137 | 1141 | ||
| 1138 | err = pci_enable_device(pdev); | 1142 | err = pci_enable_device(pdev); |
| 1139 | if (err) | 1143 | if (err) |
| @@ -1145,13 +1149,12 @@ static int wl_resume(struct pci_dev *pdev) | |||
| 1145 | if ((val & 0x0000ff00) != 0) | 1149 | if ((val & 0x0000ff00) != 0) |
| 1146 | pci_write_config_dword(pdev, 0x40, val & 0xffff00ff); | 1150 | pci_write_config_dword(pdev, 0x40, val & 0xffff00ff); |
| 1147 | 1151 | ||
| 1148 | WL_LOCK(wl); | 1152 | /* |
| 1149 | err = wl_up(wl); | 1153 | * done. driver will be put in up state |
| 1150 | WL_UNLOCK(wl); | 1154 | * in wl_ops_add_interface() call. |
| 1151 | 1155 | */ | |
| 1152 | return err; | 1156 | return err; |
| 1153 | } | 1157 | } |
| 1154 | #endif /* LINUXSTA_PS */ | ||
| 1155 | 1158 | ||
| 1156 | static void wl_remove(struct pci_dev *pdev) | 1159 | static void wl_remove(struct pci_dev *pdev) |
| 1157 | { | 1160 | { |
| @@ -1184,14 +1187,12 @@ static void wl_remove(struct pci_dev *pdev) | |||
| 1184 | } | 1187 | } |
| 1185 | 1188 | ||
| 1186 | static struct pci_driver wl_pci_driver = { | 1189 | static struct pci_driver wl_pci_driver = { |
| 1187 | .name = "brcm80211", | 1190 | .name = "brcm80211", |
| 1188 | .probe = wl_pci_probe, | 1191 | .probe = wl_pci_probe, |
| 1189 | #ifdef LINUXSTA_PS | 1192 | .suspend = wl_suspend, |
| 1190 | .suspend = wl_suspend, | 1193 | .resume = wl_resume, |
| 1191 | .resume = wl_resume, | 1194 | .remove = __devexit_p(wl_remove), |
| 1192 | #endif /* LINUXSTA_PS */ | 1195 | .id_table = wl_id_table, |
| 1193 | .remove = __devexit_p(wl_remove), | ||
| 1194 | .id_table = wl_id_table, | ||
| 1195 | }; | 1196 | }; |
| 1196 | 1197 | ||
| 1197 | /** | 1198 | /** |
diff --git a/drivers/staging/brcm80211/sys/wlc_mac80211.c b/drivers/staging/brcm80211/sys/wlc_mac80211.c index 1d5d01ac0a9..a1303863686 100644 --- a/drivers/staging/brcm80211/sys/wlc_mac80211.c +++ b/drivers/staging/brcm80211/sys/wlc_mac80211.c | |||
| @@ -5126,7 +5126,6 @@ wlc_sendpkt_mac80211(struct wlc_info *wlc, struct sk_buff *sdu, | |||
| 5126 | fifo = prio2fifo[prio]; | 5126 | fifo = prio2fifo[prio]; |
| 5127 | 5127 | ||
| 5128 | ASSERT((uint) skb_headroom(sdu) >= TXOFF); | 5128 | ASSERT((uint) skb_headroom(sdu) >= TXOFF); |
| 5129 | ASSERT(!(sdu->cloned)); | ||
| 5130 | ASSERT(!(sdu->next)); | 5129 | ASSERT(!(sdu->next)); |
| 5131 | ASSERT(!(sdu->prev)); | 5130 | ASSERT(!(sdu->prev)); |
| 5132 | ASSERT(fifo < NFIFO); | 5131 | ASSERT(fifo < NFIFO); |
diff --git a/drivers/staging/comedi/drivers/ni_labpc.c b/drivers/staging/comedi/drivers/ni_labpc.c index 4d1868d04ba..0728c3c0cb0 100644 --- a/drivers/staging/comedi/drivers/ni_labpc.c +++ b/drivers/staging/comedi/drivers/ni_labpc.c | |||
| @@ -575,7 +575,8 @@ int labpc_common_attach(struct comedi_device *dev, unsigned long iobase, | |||
| 575 | /* grab our IRQ */ | 575 | /* grab our IRQ */ |
| 576 | if (irq) { | 576 | if (irq) { |
| 577 | isr_flags = 0; | 577 | isr_flags = 0; |
| 578 | if (thisboard->bustype == pci_bustype) | 578 | if (thisboard->bustype == pci_bustype |
| 579 | || thisboard->bustype == pcmcia_bustype) | ||
| 579 | isr_flags |= IRQF_SHARED; | 580 | isr_flags |= IRQF_SHARED; |
| 580 | if (request_irq(irq, labpc_interrupt, isr_flags, | 581 | if (request_irq(irq, labpc_interrupt, isr_flags, |
| 581 | driver_labpc.driver_name, dev)) { | 582 | driver_labpc.driver_name, dev)) { |
diff --git a/drivers/staging/hv/blkvsc_drv.c b/drivers/staging/hv/blkvsc_drv.c index b3d05fcfe6d..4fb809485d9 100644 --- a/drivers/staging/hv/blkvsc_drv.c +++ b/drivers/staging/hv/blkvsc_drv.c | |||
| @@ -368,6 +368,7 @@ static int blkvsc_probe(struct device *device) | |||
| 368 | blkdev->gd->first_minor = 0; | 368 | blkdev->gd->first_minor = 0; |
| 369 | blkdev->gd->fops = &block_ops; | 369 | blkdev->gd->fops = &block_ops; |
| 370 | blkdev->gd->private_data = blkdev; | 370 | blkdev->gd->private_data = blkdev; |
| 371 | blkdev->gd->driverfs_dev = &(blkdev->device_ctx->device); | ||
| 371 | sprintf(blkdev->gd->disk_name, "hd%c", 'a' + devnum); | 372 | sprintf(blkdev->gd->disk_name, "hd%c", 'a' + devnum); |
| 372 | 373 | ||
| 373 | blkvsc_do_inquiry(blkdev); | 374 | blkvsc_do_inquiry(blkdev); |
diff --git a/drivers/staging/hv/netvsc.c b/drivers/staging/hv/netvsc.c index df9cd131e95..0edbe7483a4 100644 --- a/drivers/staging/hv/netvsc.c +++ b/drivers/staging/hv/netvsc.c | |||
| @@ -1279,7 +1279,7 @@ static void netvsc_channel_cb(void *context) | |||
| 1279 | /* ASSERT(device); */ | 1279 | /* ASSERT(device); */ |
| 1280 | 1280 | ||
| 1281 | packet = kzalloc(NETVSC_PACKET_SIZE * sizeof(unsigned char), | 1281 | packet = kzalloc(NETVSC_PACKET_SIZE * sizeof(unsigned char), |
| 1282 | GFP_KERNEL); | 1282 | GFP_ATOMIC); |
| 1283 | if (!packet) | 1283 | if (!packet) |
| 1284 | return; | 1284 | return; |
| 1285 | buffer = packet; | 1285 | buffer = packet; |
diff --git a/drivers/staging/hv/netvsc_drv.c b/drivers/staging/hv/netvsc_drv.c index 0147b407512..54706a16dc0 100644 --- a/drivers/staging/hv/netvsc_drv.c +++ b/drivers/staging/hv/netvsc_drv.c | |||
| @@ -358,7 +358,6 @@ static int netvsc_probe(struct device *device) | |||
| 358 | 358 | ||
| 359 | /* Set initial state */ | 359 | /* Set initial state */ |
| 360 | netif_carrier_off(net); | 360 | netif_carrier_off(net); |
| 361 | netif_stop_queue(net); | ||
| 362 | 361 | ||
| 363 | net_device_ctx = netdev_priv(net); | 362 | net_device_ctx = netdev_priv(net); |
| 364 | net_device_ctx->device_ctx = device_ctx; | 363 | net_device_ctx->device_ctx = device_ctx; |
diff --git a/drivers/staging/iio/adc/ad7476_core.c b/drivers/staging/iio/adc/ad7476_core.c index deb68c8a6e1..b8b54da67c6 100644 --- a/drivers/staging/iio/adc/ad7476_core.c +++ b/drivers/staging/iio/adc/ad7476_core.c | |||
| @@ -68,7 +68,7 @@ static ssize_t ad7476_show_scale(struct device *dev, | |||
| 68 | /* Corresponds to Vref / 2^(bits) */ | 68 | /* Corresponds to Vref / 2^(bits) */ |
| 69 | unsigned int scale_uv = (st->int_vref_mv * 1000) >> st->chip_info->bits; | 69 | unsigned int scale_uv = (st->int_vref_mv * 1000) >> st->chip_info->bits; |
| 70 | 70 | ||
| 71 | return sprintf(buf, "%d.%d\n", scale_uv / 1000, scale_uv % 1000); | 71 | return sprintf(buf, "%d.%03d\n", scale_uv / 1000, scale_uv % 1000); |
| 72 | } | 72 | } |
| 73 | static IIO_DEVICE_ATTR(in_scale, S_IRUGO, ad7476_show_scale, NULL, 0); | 73 | static IIO_DEVICE_ATTR(in_scale, S_IRUGO, ad7476_show_scale, NULL, 0); |
| 74 | 74 | ||
diff --git a/drivers/staging/iio/adc/ad7887_core.c b/drivers/staging/iio/adc/ad7887_core.c index 685908995d4..5d85efab658 100644 --- a/drivers/staging/iio/adc/ad7887_core.c +++ b/drivers/staging/iio/adc/ad7887_core.c | |||
| @@ -68,7 +68,7 @@ static ssize_t ad7887_show_scale(struct device *dev, | |||
| 68 | /* Corresponds to Vref / 2^(bits) */ | 68 | /* Corresponds to Vref / 2^(bits) */ |
| 69 | unsigned int scale_uv = (st->int_vref_mv * 1000) >> st->chip_info->bits; | 69 | unsigned int scale_uv = (st->int_vref_mv * 1000) >> st->chip_info->bits; |
| 70 | 70 | ||
| 71 | return sprintf(buf, "%d.%d\n", scale_uv / 1000, scale_uv % 1000); | 71 | return sprintf(buf, "%d.%03d\n", scale_uv / 1000, scale_uv % 1000); |
| 72 | } | 72 | } |
| 73 | static IIO_DEVICE_ATTR(in_scale, S_IRUGO, ad7887_show_scale, NULL, 0); | 73 | static IIO_DEVICE_ATTR(in_scale, S_IRUGO, ad7887_show_scale, NULL, 0); |
| 74 | 74 | ||
diff --git a/drivers/staging/iio/adc/ad799x_core.c b/drivers/staging/iio/adc/ad799x_core.c index 6309d521a86..89ccf375a18 100644 --- a/drivers/staging/iio/adc/ad799x_core.c +++ b/drivers/staging/iio/adc/ad799x_core.c | |||
| @@ -432,7 +432,7 @@ static ssize_t ad799x_show_scale(struct device *dev, | |||
| 432 | /* Corresponds to Vref / 2^(bits) */ | 432 | /* Corresponds to Vref / 2^(bits) */ |
| 433 | unsigned int scale_uv = (st->int_vref_mv * 1000) >> st->chip_info->bits; | 433 | unsigned int scale_uv = (st->int_vref_mv * 1000) >> st->chip_info->bits; |
| 434 | 434 | ||
| 435 | return sprintf(buf, "%d.%d\n", scale_uv / 1000, scale_uv % 1000); | 435 | return sprintf(buf, "%d.%03d\n", scale_uv / 1000, scale_uv % 1000); |
| 436 | } | 436 | } |
| 437 | 437 | ||
| 438 | static IIO_DEVICE_ATTR(in_scale, S_IRUGO, ad799x_show_scale, NULL, 0); | 438 | static IIO_DEVICE_ATTR(in_scale, S_IRUGO, ad799x_show_scale, NULL, 0); |
diff --git a/drivers/staging/iio/dac/ad5446.c b/drivers/staging/iio/dac/ad5446.c index e3387cd3114..0f87ecac82f 100644 --- a/drivers/staging/iio/dac/ad5446.c +++ b/drivers/staging/iio/dac/ad5446.c | |||
| @@ -87,7 +87,7 @@ static ssize_t ad5446_show_scale(struct device *dev, | |||
| 87 | /* Corresponds to Vref / 2^(bits) */ | 87 | /* Corresponds to Vref / 2^(bits) */ |
| 88 | unsigned int scale_uv = (st->vref_mv * 1000) >> st->chip_info->bits; | 88 | unsigned int scale_uv = (st->vref_mv * 1000) >> st->chip_info->bits; |
| 89 | 89 | ||
| 90 | return sprintf(buf, "%d.%d\n", scale_uv / 1000, scale_uv % 1000); | 90 | return sprintf(buf, "%d.%03d\n", scale_uv / 1000, scale_uv % 1000); |
| 91 | } | 91 | } |
| 92 | static IIO_DEVICE_ATTR(out_scale, S_IRUGO, ad5446_show_scale, NULL, 0); | 92 | static IIO_DEVICE_ATTR(out_scale, S_IRUGO, ad5446_show_scale, NULL, 0); |
| 93 | 93 | ||
diff --git a/drivers/staging/lirc/lirc_zilog.c b/drivers/staging/lirc/lirc_zilog.c index 3fe5f416019..0aad0d7a74a 100644 --- a/drivers/staging/lirc/lirc_zilog.c +++ b/drivers/staging/lirc/lirc_zilog.c | |||
| @@ -495,7 +495,7 @@ static int send_data_block(struct IR_tx *tx, unsigned char *data_block) | |||
| 495 | /* send boot data to the IR TX device */ | 495 | /* send boot data to the IR TX device */ |
| 496 | static int send_boot_data(struct IR_tx *tx) | 496 | static int send_boot_data(struct IR_tx *tx) |
| 497 | { | 497 | { |
| 498 | int ret; | 498 | int ret, i; |
| 499 | unsigned char buf[4]; | 499 | unsigned char buf[4]; |
| 500 | 500 | ||
| 501 | /* send the boot block */ | 501 | /* send the boot block */ |
| @@ -503,7 +503,7 @@ static int send_boot_data(struct IR_tx *tx) | |||
| 503 | if (ret != 0) | 503 | if (ret != 0) |
| 504 | return ret; | 504 | return ret; |
| 505 | 505 | ||
| 506 | /* kick it off? */ | 506 | /* Hit the go button to activate the new boot data */ |
| 507 | buf[0] = 0x00; | 507 | buf[0] = 0x00; |
| 508 | buf[1] = 0x20; | 508 | buf[1] = 0x20; |
| 509 | ret = i2c_master_send(tx->c, buf, 2); | 509 | ret = i2c_master_send(tx->c, buf, 2); |
| @@ -511,7 +511,19 @@ static int send_boot_data(struct IR_tx *tx) | |||
| 511 | zilog_error("i2c_master_send failed with %d\n", ret); | 511 | zilog_error("i2c_master_send failed with %d\n", ret); |
| 512 | return ret < 0 ? ret : -EFAULT; | 512 | return ret < 0 ? ret : -EFAULT; |
| 513 | } | 513 | } |
| 514 | ret = i2c_master_send(tx->c, buf, 1); | 514 | |
| 515 | /* | ||
| 516 | * Wait for zilog to settle after hitting go post boot block upload. | ||
| 517 | * Without this delay, the HD-PVR and HVR-1950 both return an -EIO | ||
| 518 | * upon attempting to get firmware revision, and tx probe thus fails. | ||
| 519 | */ | ||
| 520 | for (i = 0; i < 10; i++) { | ||
| 521 | ret = i2c_master_send(tx->c, buf, 1); | ||
| 522 | if (ret == 1) | ||
| 523 | break; | ||
| 524 | udelay(100); | ||
| 525 | } | ||
| 526 | |||
| 515 | if (ret != 1) { | 527 | if (ret != 1) { |
| 516 | zilog_error("i2c_master_send failed with %d\n", ret); | 528 | zilog_error("i2c_master_send failed with %d\n", ret); |
| 517 | return ret < 0 ? ret : -EFAULT; | 529 | return ret < 0 ? ret : -EFAULT; |
| @@ -523,8 +535,8 @@ static int send_boot_data(struct IR_tx *tx) | |||
| 523 | zilog_error("i2c_master_recv failed with %d\n", ret); | 535 | zilog_error("i2c_master_recv failed with %d\n", ret); |
| 524 | return 0; | 536 | return 0; |
| 525 | } | 537 | } |
| 526 | if (buf[0] != 0x80) { | 538 | if ((buf[0] != 0x80) && (buf[0] != 0xa0)) { |
| 527 | zilog_error("unexpected IR TX response: %02x\n", buf[0]); | 539 | zilog_error("unexpected IR TX init response: %02x\n", buf[0]); |
| 528 | return 0; | 540 | return 0; |
| 529 | } | 541 | } |
| 530 | zilog_notify("Zilog/Hauppauge IR blaster firmware version " | 542 | zilog_notify("Zilog/Hauppauge IR blaster firmware version " |
| @@ -827,7 +839,15 @@ static int send_code(struct IR_tx *tx, unsigned int code, unsigned int key) | |||
| 827 | zilog_error("i2c_master_send failed with %d\n", ret); | 839 | zilog_error("i2c_master_send failed with %d\n", ret); |
| 828 | return ret < 0 ? ret : -EFAULT; | 840 | return ret < 0 ? ret : -EFAULT; |
| 829 | } | 841 | } |
| 830 | ret = i2c_master_send(tx->c, buf, 1); | 842 | |
| 843 | /* Give the z8 a moment to process data block */ | ||
| 844 | for (i = 0; i < 10; i++) { | ||
| 845 | ret = i2c_master_send(tx->c, buf, 1); | ||
| 846 | if (ret == 1) | ||
| 847 | break; | ||
| 848 | udelay(100); | ||
| 849 | } | ||
| 850 | |||
| 831 | if (ret != 1) { | 851 | if (ret != 1) { |
| 832 | zilog_error("i2c_master_send failed with %d\n", ret); | 852 | zilog_error("i2c_master_send failed with %d\n", ret); |
| 833 | return ret < 0 ? ret : -EFAULT; | 853 | return ret < 0 ? ret : -EFAULT; |
diff --git a/drivers/staging/msm/msm_fb.c b/drivers/staging/msm/msm_fb.c index 23fa049b51f..a2f29d46405 100644 --- a/drivers/staging/msm/msm_fb.c +++ b/drivers/staging/msm/msm_fb.c | |||
| @@ -347,7 +347,7 @@ static int msm_fb_suspend(struct platform_device *pdev, pm_message_t state) | |||
| 347 | if ((!mfd) || (mfd->key != MFD_KEY)) | 347 | if ((!mfd) || (mfd->key != MFD_KEY)) |
| 348 | return 0; | 348 | return 0; |
| 349 | 349 | ||
| 350 | acquire_console_sem(); | 350 | console_lock(); |
| 351 | fb_set_suspend(mfd->fbi, 1); | 351 | fb_set_suspend(mfd->fbi, 1); |
| 352 | 352 | ||
| 353 | ret = msm_fb_suspend_sub(mfd); | 353 | ret = msm_fb_suspend_sub(mfd); |
| @@ -358,7 +358,7 @@ static int msm_fb_suspend(struct platform_device *pdev, pm_message_t state) | |||
| 358 | pdev->dev.power.power_state = state; | 358 | pdev->dev.power.power_state = state; |
| 359 | } | 359 | } |
| 360 | 360 | ||
| 361 | release_console_sem(); | 361 | console_unlock(); |
| 362 | return ret; | 362 | return ret; |
| 363 | } | 363 | } |
| 364 | #else | 364 | #else |
| @@ -431,11 +431,11 @@ static int msm_fb_resume(struct platform_device *pdev) | |||
| 431 | if ((!mfd) || (mfd->key != MFD_KEY)) | 431 | if ((!mfd) || (mfd->key != MFD_KEY)) |
| 432 | return 0; | 432 | return 0; |
| 433 | 433 | ||
| 434 | acquire_console_sem(); | 434 | console_lock(); |
| 435 | ret = msm_fb_resume_sub(mfd); | 435 | ret = msm_fb_resume_sub(mfd); |
| 436 | pdev->dev.power.power_state = PMSG_ON; | 436 | pdev->dev.power.power_state = PMSG_ON; |
| 437 | fb_set_suspend(mfd->fbi, 1); | 437 | fb_set_suspend(mfd->fbi, 1); |
| 438 | release_console_sem(); | 438 | console_unlock(); |
| 439 | 439 | ||
| 440 | return ret; | 440 | return ret; |
| 441 | } | 441 | } |
diff --git a/drivers/staging/olpc_dcon/olpc_dcon.c b/drivers/staging/olpc_dcon/olpc_dcon.c index 9f26dc9408b..56a283d1a74 100644 --- a/drivers/staging/olpc_dcon/olpc_dcon.c +++ b/drivers/staging/olpc_dcon/olpc_dcon.c | |||
| @@ -373,17 +373,17 @@ static void dcon_source_switch(struct work_struct *work) | |||
| 373 | * | 373 | * |
| 374 | * For now, we just hope.. | 374 | * For now, we just hope.. |
| 375 | */ | 375 | */ |
| 376 | acquire_console_sem(); | 376 | console_lock(); |
| 377 | ignore_fb_events = 1; | 377 | ignore_fb_events = 1; |
| 378 | if (fb_blank(fbinfo, FB_BLANK_UNBLANK)) { | 378 | if (fb_blank(fbinfo, FB_BLANK_UNBLANK)) { |
| 379 | ignore_fb_events = 0; | 379 | ignore_fb_events = 0; |
| 380 | release_console_sem(); | 380 | console_unlock(); |
| 381 | printk(KERN_ERR "olpc-dcon: Failed to enter CPU mode\n"); | 381 | printk(KERN_ERR "olpc-dcon: Failed to enter CPU mode\n"); |
| 382 | dcon_pending = DCON_SOURCE_DCON; | 382 | dcon_pending = DCON_SOURCE_DCON; |
| 383 | return; | 383 | return; |
| 384 | } | 384 | } |
| 385 | ignore_fb_events = 0; | 385 | ignore_fb_events = 0; |
| 386 | release_console_sem(); | 386 | console_unlock(); |
| 387 | 387 | ||
| 388 | /* And turn off the DCON */ | 388 | /* And turn off the DCON */ |
| 389 | pdata->set_dconload(1); | 389 | pdata->set_dconload(1); |
| @@ -435,12 +435,12 @@ static void dcon_source_switch(struct work_struct *work) | |||
| 435 | } | 435 | } |
| 436 | } | 436 | } |
| 437 | 437 | ||
| 438 | acquire_console_sem(); | 438 | console_lock(); |
| 439 | ignore_fb_events = 1; | 439 | ignore_fb_events = 1; |
| 440 | if (fb_blank(fbinfo, FB_BLANK_POWERDOWN)) | 440 | if (fb_blank(fbinfo, FB_BLANK_POWERDOWN)) |
| 441 | printk(KERN_ERR "olpc-dcon: couldn't blank fb!\n"); | 441 | printk(KERN_ERR "olpc-dcon: couldn't blank fb!\n"); |
| 442 | ignore_fb_events = 0; | 442 | ignore_fb_events = 0; |
| 443 | release_console_sem(); | 443 | console_unlock(); |
| 444 | 444 | ||
| 445 | printk(KERN_INFO "olpc-dcon: The DCON has control\n"); | 445 | printk(KERN_INFO "olpc-dcon: The DCON has control\n"); |
| 446 | break; | 446 | break; |
diff --git a/drivers/staging/rt2860/rt_main_dev.c b/drivers/staging/rt2860/rt_main_dev.c index 701561d6b6f..236dd36d349 100644 --- a/drivers/staging/rt2860/rt_main_dev.c +++ b/drivers/staging/rt2860/rt_main_dev.c | |||
| @@ -484,8 +484,6 @@ struct net_device *RtmpPhyNetDevInit(struct rt_rtmp_adapter *pAd, | |||
| 484 | net_dev->ml_priv = (void *)pAd; | 484 | net_dev->ml_priv = (void *)pAd; |
| 485 | pAd->net_dev = net_dev; | 485 | pAd->net_dev = net_dev; |
| 486 | 486 | ||
| 487 | netif_stop_queue(net_dev); | ||
| 488 | |||
| 489 | return net_dev; | 487 | return net_dev; |
| 490 | 488 | ||
| 491 | } | 489 | } |
diff --git a/drivers/staging/rt2860/usb_main_dev.c b/drivers/staging/rt2860/usb_main_dev.c index ee68d51caa4..322bf49ee90 100644 --- a/drivers/staging/rt2860/usb_main_dev.c +++ b/drivers/staging/rt2860/usb_main_dev.c | |||
| @@ -106,6 +106,7 @@ struct usb_device_id rtusb_usb_id[] = { | |||
| 106 | {USB_DEVICE(0x0411, 0x016f)}, /* MelCo.,Inc. WLI-UC-G301N */ | 106 | {USB_DEVICE(0x0411, 0x016f)}, /* MelCo.,Inc. WLI-UC-G301N */ |
| 107 | {USB_DEVICE(0x1737, 0x0070)}, /* Linksys WUSB100 */ | 107 | {USB_DEVICE(0x1737, 0x0070)}, /* Linksys WUSB100 */ |
| 108 | {USB_DEVICE(0x1737, 0x0071)}, /* Linksys WUSB600N */ | 108 | {USB_DEVICE(0x1737, 0x0071)}, /* Linksys WUSB600N */ |
| 109 | {USB_DEVICE(0x1737, 0x0078)}, /* Linksys WUSB100v2 */ | ||
| 109 | {USB_DEVICE(0x0411, 0x00e8)}, /* Buffalo WLI-UC-G300N */ | 110 | {USB_DEVICE(0x0411, 0x00e8)}, /* Buffalo WLI-UC-G300N */ |
| 110 | {USB_DEVICE(0x050d, 0x815c)}, /* Belkin F5D8053 */ | 111 | {USB_DEVICE(0x050d, 0x815c)}, /* Belkin F5D8053 */ |
| 111 | {USB_DEVICE(0x100D, 0x9031)}, /* Motorola 2770 */ | 112 | {USB_DEVICE(0x100D, 0x9031)}, /* Motorola 2770 */ |
diff --git a/drivers/staging/rtl8712/hal_init.c b/drivers/staging/rtl8712/hal_init.c index 32088a641eb..84be383abec 100644 --- a/drivers/staging/rtl8712/hal_init.c +++ b/drivers/staging/rtl8712/hal_init.c | |||
| @@ -128,12 +128,13 @@ static u8 rtl8712_dl_fw(struct _adapter *padapter) | |||
| 128 | u8 *ptmpchar = NULL, *ppayload, *ptr; | 128 | u8 *ptmpchar = NULL, *ppayload, *ptr; |
| 129 | struct tx_desc *ptx_desc; | 129 | struct tx_desc *ptx_desc; |
| 130 | u32 txdscp_sz = sizeof(struct tx_desc); | 130 | u32 txdscp_sz = sizeof(struct tx_desc); |
| 131 | u8 ret = _FAIL; | ||
| 131 | 132 | ||
| 132 | ulfilelength = rtl871x_open_fw(padapter, &phfwfile_hdl, &pmappedfw); | 133 | ulfilelength = rtl871x_open_fw(padapter, &phfwfile_hdl, &pmappedfw); |
| 133 | if (pmappedfw && (ulfilelength > 0)) { | 134 | if (pmappedfw && (ulfilelength > 0)) { |
| 134 | update_fwhdr(&fwhdr, pmappedfw); | 135 | update_fwhdr(&fwhdr, pmappedfw); |
| 135 | if (chk_fwhdr(&fwhdr, ulfilelength) == _FAIL) | 136 | if (chk_fwhdr(&fwhdr, ulfilelength) == _FAIL) |
| 136 | goto exit_fail; | 137 | goto firmware_rel; |
| 137 | fill_fwpriv(padapter, &fwhdr.fwpriv); | 138 | fill_fwpriv(padapter, &fwhdr.fwpriv); |
| 138 | /* firmware check ok */ | 139 | /* firmware check ok */ |
| 139 | maxlen = (fwhdr.img_IMEM_size > fwhdr.img_SRAM_size) ? | 140 | maxlen = (fwhdr.img_IMEM_size > fwhdr.img_SRAM_size) ? |
| @@ -141,7 +142,7 @@ static u8 rtl8712_dl_fw(struct _adapter *padapter) | |||
| 141 | maxlen += txdscp_sz; | 142 | maxlen += txdscp_sz; |
| 142 | ptmpchar = _malloc(maxlen + FWBUFF_ALIGN_SZ); | 143 | ptmpchar = _malloc(maxlen + FWBUFF_ALIGN_SZ); |
| 143 | if (ptmpchar == NULL) | 144 | if (ptmpchar == NULL) |
| 144 | return _FAIL; | 145 | goto firmware_rel; |
| 145 | 146 | ||
| 146 | ptx_desc = (struct tx_desc *)(ptmpchar + FWBUFF_ALIGN_SZ - | 147 | ptx_desc = (struct tx_desc *)(ptmpchar + FWBUFF_ALIGN_SZ - |
| 147 | ((addr_t)(ptmpchar) & (FWBUFF_ALIGN_SZ - 1))); | 148 | ((addr_t)(ptmpchar) & (FWBUFF_ALIGN_SZ - 1))); |
| @@ -273,11 +274,13 @@ static u8 rtl8712_dl_fw(struct _adapter *padapter) | |||
| 273 | goto exit_fail; | 274 | goto exit_fail; |
| 274 | } else | 275 | } else |
| 275 | goto exit_fail; | 276 | goto exit_fail; |
| 276 | return _SUCCESS; | 277 | ret = _SUCCESS; |
| 277 | 278 | ||
| 278 | exit_fail: | 279 | exit_fail: |
| 279 | kfree(ptmpchar); | 280 | kfree(ptmpchar); |
| 280 | return _FAIL; | 281 | firmware_rel: |
| 282 | release_firmware((struct firmware *)phfwfile_hdl); | ||
| 283 | return ret; | ||
| 281 | } | 284 | } |
| 282 | 285 | ||
| 283 | uint rtl8712_hal_init(struct _adapter *padapter) | 286 | uint rtl8712_hal_init(struct _adapter *padapter) |
diff --git a/drivers/staging/rtl8712/usb_intf.c b/drivers/staging/rtl8712/usb_intf.c index a692ee88b9e..21ce2af447b 100644 --- a/drivers/staging/rtl8712/usb_intf.c +++ b/drivers/staging/rtl8712/usb_intf.c | |||
| @@ -47,54 +47,123 @@ static int r871xu_drv_init(struct usb_interface *pusb_intf, | |||
| 47 | static void r871xu_dev_remove(struct usb_interface *pusb_intf); | 47 | static void r871xu_dev_remove(struct usb_interface *pusb_intf); |
| 48 | 48 | ||
| 49 | static struct usb_device_id rtl871x_usb_id_tbl[] = { | 49 | static struct usb_device_id rtl871x_usb_id_tbl[] = { |
| 50 | /*92SU | 50 | |
| 51 | * Realtek */ | 51 | /* RTL8188SU */ |
| 52 | {USB_DEVICE(0x0bda, 0x8171)}, | 52 | /* Realtek */ |
| 53 | {USB_DEVICE(0x0bda, 0x8172)}, | 53 | {USB_DEVICE(0x0BDA, 0x8171)}, |
| 54 | {USB_DEVICE(0x0bda, 0x8173)}, | 54 | {USB_DEVICE(0x0bda, 0x8173)}, |
| 55 | {USB_DEVICE(0x0bda, 0x8174)}, | ||
| 56 | {USB_DEVICE(0x0bda, 0x8712)}, | 55 | {USB_DEVICE(0x0bda, 0x8712)}, |
| 57 | {USB_DEVICE(0x0bda, 0x8713)}, | 56 | {USB_DEVICE(0x0bda, 0x8713)}, |
| 58 | {USB_DEVICE(0x0bda, 0xC512)}, | 57 | {USB_DEVICE(0x0bda, 0xC512)}, |
| 59 | /* Abocom */ | 58 | /* Abocom */ |
| 60 | {USB_DEVICE(0x07B8, 0x8188)}, | 59 | {USB_DEVICE(0x07B8, 0x8188)}, |
| 60 | /* ASUS */ | ||
| 61 | {USB_DEVICE(0x0B05, 0x1786)}, | ||
| 62 | {USB_DEVICE(0x0B05, 0x1791)}, /* 11n mode disable */ | ||
| 63 | /* Belkin */ | ||
| 64 | {USB_DEVICE(0x050D, 0x945A)}, | ||
| 61 | /* Corega */ | 65 | /* Corega */ |
| 62 | {USB_DEVICE(0x07aa, 0x0047)}, | 66 | {USB_DEVICE(0x07AA, 0x0047)}, |
| 63 | /* Dlink */ | 67 | /* D-Link */ |
| 64 | {USB_DEVICE(0x07d1, 0x3303)}, | 68 | {USB_DEVICE(0x2001, 0x3306)}, |
| 65 | {USB_DEVICE(0x07d1, 0x3302)}, | 69 | {USB_DEVICE(0x07D1, 0x3306)}, /* 11n mode disable */ |
| 66 | {USB_DEVICE(0x07d1, 0x3300)}, | 70 | /* Edimax */ |
| 67 | /* Dlink for Skyworth */ | 71 | {USB_DEVICE(0x7392, 0x7611)}, |
| 68 | {USB_DEVICE(0x14b2, 0x3300)}, | ||
| 69 | {USB_DEVICE(0x14b2, 0x3301)}, | ||
| 70 | {USB_DEVICE(0x14b2, 0x3302)}, | ||
| 71 | /* EnGenius */ | 72 | /* EnGenius */ |
| 72 | {USB_DEVICE(0x1740, 0x9603)}, | 73 | {USB_DEVICE(0x1740, 0x9603)}, |
| 73 | {USB_DEVICE(0x1740, 0x9605)}, | 74 | /* Hawking */ |
| 75 | {USB_DEVICE(0x0E66, 0x0016)}, | ||
| 76 | /* Hercules */ | ||
| 77 | {USB_DEVICE(0x06F8, 0xE034)}, | ||
| 78 | {USB_DEVICE(0x06F8, 0xE032)}, | ||
| 79 | /* Logitec */ | ||
| 80 | {USB_DEVICE(0x0789, 0x0167)}, | ||
| 81 | /* PCI */ | ||
| 82 | {USB_DEVICE(0x2019, 0xAB28)}, | ||
| 83 | {USB_DEVICE(0x2019, 0xED16)}, | ||
| 84 | /* Sitecom */ | ||
| 85 | {USB_DEVICE(0x0DF6, 0x0057)}, | ||
| 86 | {USB_DEVICE(0x0DF6, 0x0045)}, | ||
| 87 | {USB_DEVICE(0x0DF6, 0x0059)}, /* 11n mode disable */ | ||
| 88 | {USB_DEVICE(0x0DF6, 0x004B)}, | ||
| 89 | {USB_DEVICE(0x0DF6, 0x0063)}, | ||
| 90 | /* Sweex */ | ||
| 91 | {USB_DEVICE(0x177F, 0x0154)}, | ||
| 92 | /* Thinkware */ | ||
| 93 | {USB_DEVICE(0x0BDA, 0x5077)}, | ||
| 94 | /* Toshiba */ | ||
| 95 | {USB_DEVICE(0x1690, 0x0752)}, | ||
| 96 | /* - */ | ||
| 97 | {USB_DEVICE(0x20F4, 0x646B)}, | ||
| 98 | {USB_DEVICE(0x083A, 0xC512)}, | ||
| 99 | |||
| 100 | /* RTL8191SU */ | ||
| 101 | /* Realtek */ | ||
| 102 | {USB_DEVICE(0x0BDA, 0x8172)}, | ||
| 103 | /* Amigo */ | ||
| 104 | {USB_DEVICE(0x0EB0, 0x9061)}, | ||
| 105 | /* ASUS/EKB */ | ||
| 106 | {USB_DEVICE(0x0BDA, 0x8172)}, | ||
| 107 | {USB_DEVICE(0x13D3, 0x3323)}, | ||
| 108 | {USB_DEVICE(0x13D3, 0x3311)}, /* 11n mode disable */ | ||
| 109 | {USB_DEVICE(0x13D3, 0x3342)}, | ||
| 110 | /* ASUS/EKBLenovo */ | ||
| 111 | {USB_DEVICE(0x13D3, 0x3333)}, | ||
| 112 | {USB_DEVICE(0x13D3, 0x3334)}, | ||
| 113 | {USB_DEVICE(0x13D3, 0x3335)}, /* 11n mode disable */ | ||
| 114 | {USB_DEVICE(0x13D3, 0x3336)}, /* 11n mode disable */ | ||
| 115 | /* ASUS/Media BOX */ | ||
| 116 | {USB_DEVICE(0x13D3, 0x3309)}, | ||
| 74 | /* Belkin */ | 117 | /* Belkin */ |
| 75 | {USB_DEVICE(0x050d, 0x815F)}, | 118 | {USB_DEVICE(0x050D, 0x815F)}, |
| 76 | {USB_DEVICE(0x050d, 0x945A)}, | 119 | /* D-Link */ |
| 77 | {USB_DEVICE(0x050d, 0x845A)}, | 120 | {USB_DEVICE(0x07D1, 0x3302)}, |
| 78 | /* Guillemot */ | 121 | {USB_DEVICE(0x07D1, 0x3300)}, |
| 79 | {USB_DEVICE(0x06f8, 0xe031)}, | 122 | {USB_DEVICE(0x07D1, 0x3303)}, |
| 80 | /* Edimax */ | 123 | /* Edimax */ |
| 81 | {USB_DEVICE(0x7392, 0x7611)}, | ||
| 82 | {USB_DEVICE(0x7392, 0x7612)}, | 124 | {USB_DEVICE(0x7392, 0x7612)}, |
| 83 | {USB_DEVICE(0x7392, 0x7622)}, | 125 | /* EnGenius */ |
| 84 | /* Sitecom */ | 126 | {USB_DEVICE(0x1740, 0x9605)}, |
| 85 | {USB_DEVICE(0x0DF6, 0x0045)}, | 127 | /* Guillemot */ |
| 128 | {USB_DEVICE(0x06F8, 0xE031)}, | ||
| 86 | /* Hawking */ | 129 | /* Hawking */ |
| 87 | {USB_DEVICE(0x0E66, 0x0015)}, | 130 | {USB_DEVICE(0x0E66, 0x0015)}, |
| 88 | {USB_DEVICE(0x0E66, 0x0016)}, | 131 | /* Mediao */ |
| 89 | {USB_DEVICE(0x0b05, 0x1786)}, | ||
| 90 | {USB_DEVICE(0x0b05, 0x1791)}, /* 11n mode disable */ | ||
| 91 | |||
| 92 | {USB_DEVICE(0x13D3, 0x3306)}, | 132 | {USB_DEVICE(0x13D3, 0x3306)}, |
| 93 | {USB_DEVICE(0x13D3, 0x3309)}, | 133 | /* PCI */ |
| 134 | {USB_DEVICE(0x2019, 0xED18)}, | ||
| 135 | {USB_DEVICE(0x2019, 0x4901)}, | ||
| 136 | /* Sitecom */ | ||
| 137 | {USB_DEVICE(0x0DF6, 0x0058)}, | ||
| 138 | {USB_DEVICE(0x0DF6, 0x0049)}, | ||
| 139 | {USB_DEVICE(0x0DF6, 0x004C)}, | ||
| 140 | {USB_DEVICE(0x0DF6, 0x0064)}, | ||
| 141 | /* Skyworth */ | ||
| 142 | {USB_DEVICE(0x14b2, 0x3300)}, | ||
| 143 | {USB_DEVICE(0x14b2, 0x3301)}, | ||
| 144 | {USB_DEVICE(0x14B2, 0x3302)}, | ||
| 145 | /* - */ | ||
| 146 | {USB_DEVICE(0x04F2, 0xAFF2)}, | ||
| 147 | {USB_DEVICE(0x04F2, 0xAFF5)}, | ||
| 148 | {USB_DEVICE(0x04F2, 0xAFF6)}, | ||
| 149 | {USB_DEVICE(0x13D3, 0x3339)}, | ||
| 150 | {USB_DEVICE(0x13D3, 0x3340)}, /* 11n mode disable */ | ||
| 151 | {USB_DEVICE(0x13D3, 0x3341)}, /* 11n mode disable */ | ||
| 94 | {USB_DEVICE(0x13D3, 0x3310)}, | 152 | {USB_DEVICE(0x13D3, 0x3310)}, |
| 95 | {USB_DEVICE(0x13D3, 0x3311)}, /* 11n mode disable */ | ||
| 96 | {USB_DEVICE(0x13D3, 0x3325)}, | 153 | {USB_DEVICE(0x13D3, 0x3325)}, |
| 97 | {USB_DEVICE(0x083A, 0xC512)}, | 154 | |
| 155 | /* RTL8192SU */ | ||
| 156 | /* Realtek */ | ||
| 157 | {USB_DEVICE(0x0BDA, 0x8174)}, | ||
| 158 | {USB_DEVICE(0x0BDA, 0x8174)}, | ||
| 159 | /* Belkin */ | ||
| 160 | {USB_DEVICE(0x050D, 0x845A)}, | ||
| 161 | /* Corega */ | ||
| 162 | {USB_DEVICE(0x07AA, 0x0051)}, | ||
| 163 | /* Edimax */ | ||
| 164 | {USB_DEVICE(0x7392, 0x7622)}, | ||
| 165 | /* NEC */ | ||
| 166 | {USB_DEVICE(0x0409, 0x02B6)}, | ||
| 98 | {} | 167 | {} |
| 99 | }; | 168 | }; |
| 100 | 169 | ||
| @@ -103,8 +172,20 @@ MODULE_DEVICE_TABLE(usb, rtl871x_usb_id_tbl); | |||
| 103 | static struct specific_device_id specific_device_id_tbl[] = { | 172 | static struct specific_device_id specific_device_id_tbl[] = { |
| 104 | {.idVendor = 0x0b05, .idProduct = 0x1791, | 173 | {.idVendor = 0x0b05, .idProduct = 0x1791, |
| 105 | .flags = SPEC_DEV_ID_DISABLE_HT}, | 174 | .flags = SPEC_DEV_ID_DISABLE_HT}, |
| 175 | {.idVendor = 0x0df6, .idProduct = 0x0059, | ||
| 176 | .flags = SPEC_DEV_ID_DISABLE_HT}, | ||
| 177 | {.idVendor = 0x13d3, .idProduct = 0x3306, | ||
| 178 | .flags = SPEC_DEV_ID_DISABLE_HT}, | ||
| 106 | {.idVendor = 0x13D3, .idProduct = 0x3311, | 179 | {.idVendor = 0x13D3, .idProduct = 0x3311, |
| 107 | .flags = SPEC_DEV_ID_DISABLE_HT}, | 180 | .flags = SPEC_DEV_ID_DISABLE_HT}, |
| 181 | {.idVendor = 0x13d3, .idProduct = 0x3335, | ||
| 182 | .flags = SPEC_DEV_ID_DISABLE_HT}, | ||
| 183 | {.idVendor = 0x13d3, .idProduct = 0x3336, | ||
| 184 | .flags = SPEC_DEV_ID_DISABLE_HT}, | ||
| 185 | {.idVendor = 0x13d3, .idProduct = 0x3340, | ||
| 186 | .flags = SPEC_DEV_ID_DISABLE_HT}, | ||
| 187 | {.idVendor = 0x13d3, .idProduct = 0x3341, | ||
| 188 | .flags = SPEC_DEV_ID_DISABLE_HT}, | ||
| 108 | {} | 189 | {} |
| 109 | }; | 190 | }; |
| 110 | 191 | ||
diff --git a/drivers/staging/sm7xx/smtcfb.c b/drivers/staging/sm7xx/smtcfb.c index 0bc113c44d3..d007e4a12c1 100644 --- a/drivers/staging/sm7xx/smtcfb.c +++ b/drivers/staging/sm7xx/smtcfb.c | |||
| @@ -1044,9 +1044,9 @@ static int __maybe_unused smtcfb_suspend(struct pci_dev *pdev, pm_message_t msg) | |||
| 1044 | 1044 | ||
| 1045 | /* when doing suspend, call fb apis and pci apis */ | 1045 | /* when doing suspend, call fb apis and pci apis */ |
| 1046 | if (msg.event == PM_EVENT_SUSPEND) { | 1046 | if (msg.event == PM_EVENT_SUSPEND) { |
| 1047 | acquire_console_sem(); | 1047 | console_lock(); |
| 1048 | fb_set_suspend(&sfb->fb, 1); | 1048 | fb_set_suspend(&sfb->fb, 1); |
| 1049 | release_console_sem(); | 1049 | console_unlock(); |
| 1050 | retv = pci_save_state(pdev); | 1050 | retv = pci_save_state(pdev); |
| 1051 | pci_disable_device(pdev); | 1051 | pci_disable_device(pdev); |
| 1052 | retv = pci_choose_state(pdev, msg); | 1052 | retv = pci_choose_state(pdev, msg); |
| @@ -1105,9 +1105,9 @@ static int __maybe_unused smtcfb_resume(struct pci_dev *pdev) | |||
| 1105 | 1105 | ||
| 1106 | smtcfb_setmode(sfb); | 1106 | smtcfb_setmode(sfb); |
| 1107 | 1107 | ||
| 1108 | acquire_console_sem(); | 1108 | console_lock(); |
| 1109 | fb_set_suspend(&sfb->fb, 0); | 1109 | fb_set_suspend(&sfb->fb, 0); |
| 1110 | release_console_sem(); | 1110 | console_unlock(); |
| 1111 | 1111 | ||
| 1112 | return 0; | 1112 | return 0; |
| 1113 | } | 1113 | } |
diff --git a/drivers/staging/speakup/kobjects.c b/drivers/staging/speakup/kobjects.c index 408bb9b3303..07a7f543259 100644 --- a/drivers/staging/speakup/kobjects.c +++ b/drivers/staging/speakup/kobjects.c | |||
| @@ -332,7 +332,7 @@ static ssize_t silent_store(struct kobject *kobj, struct kobj_attribute *attr, | |||
| 332 | unsigned long flags; | 332 | unsigned long flags; |
| 333 | 333 | ||
| 334 | len = strlen(buf); | 334 | len = strlen(buf); |
| 335 | if (len > 0 || len < 3) { | 335 | if (len > 0 && len < 3) { |
| 336 | ch = buf[0]; | 336 | ch = buf[0]; |
| 337 | if (ch == '\n') | 337 | if (ch == '\n') |
| 338 | ch = '0'; | 338 | ch = '0'; |
diff --git a/drivers/staging/ste_rmi4/synaptics_i2c_rmi4.c b/drivers/staging/ste_rmi4/synaptics_i2c_rmi4.c index e8f047e86a3..80183a7e662 100644 --- a/drivers/staging/ste_rmi4/synaptics_i2c_rmi4.c +++ b/drivers/staging/ste_rmi4/synaptics_i2c_rmi4.c | |||
| @@ -986,12 +986,6 @@ static int __devinit synaptics_rmi4_probe | |||
| 986 | input_set_abs_params(rmi4_data->input_dev, ABS_MT_TOUCH_MAJOR, 0, | 986 | input_set_abs_params(rmi4_data->input_dev, ABS_MT_TOUCH_MAJOR, 0, |
| 987 | MAX_TOUCH_MAJOR, 0, 0); | 987 | MAX_TOUCH_MAJOR, 0, 0); |
| 988 | 988 | ||
| 989 | retval = input_register_device(rmi4_data->input_dev); | ||
| 990 | if (retval) { | ||
| 991 | dev_err(&client->dev, "%s:input register failed\n", __func__); | ||
| 992 | goto err_input_register; | ||
| 993 | } | ||
| 994 | |||
| 995 | /* Clear interrupts */ | 989 | /* Clear interrupts */ |
| 996 | synaptics_rmi4_i2c_block_read(rmi4_data, | 990 | synaptics_rmi4_i2c_block_read(rmi4_data, |
| 997 | rmi4_data->fn01_data_base_addr + 1, intr_status, | 991 | rmi4_data->fn01_data_base_addr + 1, intr_status, |
| @@ -1003,15 +997,20 @@ static int __devinit synaptics_rmi4_probe | |||
| 1003 | if (retval) { | 997 | if (retval) { |
| 1004 | dev_err(&client->dev, "%s:Unable to get attn irq %d\n", | 998 | dev_err(&client->dev, "%s:Unable to get attn irq %d\n", |
| 1005 | __func__, platformdata->irq_number); | 999 | __func__, platformdata->irq_number); |
| 1006 | goto err_request_irq; | 1000 | goto err_unset_clientdata; |
| 1001 | } | ||
| 1002 | |||
| 1003 | retval = input_register_device(rmi4_data->input_dev); | ||
| 1004 | if (retval) { | ||
| 1005 | dev_err(&client->dev, "%s:input register failed\n", __func__); | ||
| 1006 | goto err_free_irq; | ||
| 1007 | } | 1007 | } |
| 1008 | 1008 | ||
| 1009 | return retval; | 1009 | return retval; |
| 1010 | 1010 | ||
| 1011 | err_request_irq: | 1011 | err_free_irq: |
| 1012 | free_irq(platformdata->irq_number, rmi4_data); | 1012 | free_irq(platformdata->irq_number, rmi4_data); |
| 1013 | input_unregister_device(rmi4_data->input_dev); | 1013 | err_unset_clientdata: |
| 1014 | err_input_register: | ||
| 1015 | i2c_set_clientdata(client, NULL); | 1014 | i2c_set_clientdata(client, NULL); |
| 1016 | err_query_dev: | 1015 | err_query_dev: |
| 1017 | if (platformdata->regulator_en) { | 1016 | if (platformdata->regulator_en) { |
diff --git a/drivers/staging/tidspbridge/core/io_sm.c b/drivers/staging/tidspbridge/core/io_sm.c index 571864555dd..27e0aa81a58 100644 --- a/drivers/staging/tidspbridge/core/io_sm.c +++ b/drivers/staging/tidspbridge/core/io_sm.c | |||
| @@ -949,7 +949,7 @@ func_end: | |||
| 949 | * Calls the Bridge's CHNL_ISR to determine if this interrupt is ours, then | 949 | * Calls the Bridge's CHNL_ISR to determine if this interrupt is ours, then |
| 950 | * schedules a DPC to dispatch I/O. | 950 | * schedules a DPC to dispatch I/O. |
| 951 | */ | 951 | */ |
| 952 | void io_mbox_msg(u32 msg) | 952 | int io_mbox_msg(struct notifier_block *self, unsigned long len, void *msg) |
| 953 | { | 953 | { |
| 954 | struct io_mgr *pio_mgr; | 954 | struct io_mgr *pio_mgr; |
| 955 | struct dev_object *dev_obj; | 955 | struct dev_object *dev_obj; |
| @@ -959,9 +959,9 @@ void io_mbox_msg(u32 msg) | |||
| 959 | dev_get_io_mgr(dev_obj, &pio_mgr); | 959 | dev_get_io_mgr(dev_obj, &pio_mgr); |
| 960 | 960 | ||
| 961 | if (!pio_mgr) | 961 | if (!pio_mgr) |
| 962 | return; | 962 | return NOTIFY_BAD; |
| 963 | 963 | ||
| 964 | pio_mgr->intr_val = (u16)msg; | 964 | pio_mgr->intr_val = (u16)((u32)msg); |
| 965 | if (pio_mgr->intr_val & MBX_PM_CLASS) | 965 | if (pio_mgr->intr_val & MBX_PM_CLASS) |
| 966 | io_dispatch_pm(pio_mgr); | 966 | io_dispatch_pm(pio_mgr); |
| 967 | 967 | ||
| @@ -973,7 +973,7 @@ void io_mbox_msg(u32 msg) | |||
| 973 | spin_unlock_irqrestore(&pio_mgr->dpc_lock, flags); | 973 | spin_unlock_irqrestore(&pio_mgr->dpc_lock, flags); |
| 974 | tasklet_schedule(&pio_mgr->dpc_tasklet); | 974 | tasklet_schedule(&pio_mgr->dpc_tasklet); |
| 975 | } | 975 | } |
| 976 | return; | 976 | return NOTIFY_OK; |
| 977 | } | 977 | } |
| 978 | 978 | ||
| 979 | /* | 979 | /* |
diff --git a/drivers/staging/tidspbridge/core/tiomap3430.c b/drivers/staging/tidspbridge/core/tiomap3430.c index a3b0a183d57..a3f69f6f505 100644 --- a/drivers/staging/tidspbridge/core/tiomap3430.c +++ b/drivers/staging/tidspbridge/core/tiomap3430.c | |||
| @@ -223,6 +223,10 @@ static struct bridge_drv_interface drv_interface_fxns = { | |||
| 223 | bridge_msg_set_queue_id, | 223 | bridge_msg_set_queue_id, |
| 224 | }; | 224 | }; |
| 225 | 225 | ||
| 226 | static struct notifier_block dsp_mbox_notifier = { | ||
| 227 | .notifier_call = io_mbox_msg, | ||
| 228 | }; | ||
| 229 | |||
| 226 | static inline void flush_all(struct bridge_dev_context *dev_context) | 230 | static inline void flush_all(struct bridge_dev_context *dev_context) |
| 227 | { | 231 | { |
| 228 | if (dev_context->dw_brd_state == BRD_DSP_HIBERNATION || | 232 | if (dev_context->dw_brd_state == BRD_DSP_HIBERNATION || |
| @@ -553,7 +557,7 @@ static int bridge_brd_start(struct bridge_dev_context *dev_ctxt, | |||
| 553 | * Enable Mailbox events and also drain any pending | 557 | * Enable Mailbox events and also drain any pending |
| 554 | * stale messages. | 558 | * stale messages. |
| 555 | */ | 559 | */ |
| 556 | dev_context->mbox = omap_mbox_get("dsp"); | 560 | dev_context->mbox = omap_mbox_get("dsp", &dsp_mbox_notifier); |
| 557 | if (IS_ERR(dev_context->mbox)) { | 561 | if (IS_ERR(dev_context->mbox)) { |
| 558 | dev_context->mbox = NULL; | 562 | dev_context->mbox = NULL; |
| 559 | pr_err("%s: Failed to get dsp mailbox handle\n", | 563 | pr_err("%s: Failed to get dsp mailbox handle\n", |
| @@ -563,8 +567,6 @@ static int bridge_brd_start(struct bridge_dev_context *dev_ctxt, | |||
| 563 | 567 | ||
| 564 | } | 568 | } |
| 565 | if (!status) { | 569 | if (!status) { |
| 566 | dev_context->mbox->rxq->callback = (int (*)(void *))io_mbox_msg; | ||
| 567 | |||
| 568 | /*PM_IVA2GRPSEL_PER = 0xC0;*/ | 570 | /*PM_IVA2GRPSEL_PER = 0xC0;*/ |
| 569 | temp = readl(resources->dw_per_pm_base + 0xA8); | 571 | temp = readl(resources->dw_per_pm_base + 0xA8); |
| 570 | temp = (temp & 0xFFFFFF30) | 0xC0; | 572 | temp = (temp & 0xFFFFFF30) | 0xC0; |
| @@ -685,7 +687,7 @@ static int bridge_brd_stop(struct bridge_dev_context *dev_ctxt) | |||
| 685 | /* Disable the mailbox interrupts */ | 687 | /* Disable the mailbox interrupts */ |
| 686 | if (dev_context->mbox) { | 688 | if (dev_context->mbox) { |
| 687 | omap_mbox_disable_irq(dev_context->mbox, IRQ_RX); | 689 | omap_mbox_disable_irq(dev_context->mbox, IRQ_RX); |
| 688 | omap_mbox_put(dev_context->mbox); | 690 | omap_mbox_put(dev_context->mbox, &dsp_mbox_notifier); |
| 689 | dev_context->mbox = NULL; | 691 | dev_context->mbox = NULL; |
| 690 | } | 692 | } |
| 691 | /* Reset IVA2 clocks*/ | 693 | /* Reset IVA2 clocks*/ |
| @@ -786,10 +788,7 @@ static int bridge_dev_create(struct bridge_dev_context | |||
| 786 | 788 | ||
| 787 | pt_attrs = kzalloc(sizeof(struct pg_table_attrs), GFP_KERNEL); | 789 | pt_attrs = kzalloc(sizeof(struct pg_table_attrs), GFP_KERNEL); |
| 788 | if (pt_attrs != NULL) { | 790 | if (pt_attrs != NULL) { |
| 789 | /* Assuming that we use only DSP's memory map | 791 | pt_attrs->l1_size = SZ_16K; /* 4096 entries of 32 bits */ |
| 790 | * until 0x4000:0000 , we would need only 1024 | ||
| 791 | * L1 enties i.e L1 size = 4K */ | ||
| 792 | pt_attrs->l1_size = 0x1000; | ||
| 793 | align_size = pt_attrs->l1_size; | 792 | align_size = pt_attrs->l1_size; |
| 794 | /* Align sizes are expected to be power of 2 */ | 793 | /* Align sizes are expected to be power of 2 */ |
| 795 | /* we like to get aligned on L1 table size */ | 794 | /* we like to get aligned on L1 table size */ |
diff --git a/drivers/staging/tidspbridge/include/dspbridge/io_sm.h b/drivers/staging/tidspbridge/include/dspbridge/io_sm.h index 18aec55d864..8242c70e09d 100644 --- a/drivers/staging/tidspbridge/include/dspbridge/io_sm.h +++ b/drivers/staging/tidspbridge/include/dspbridge/io_sm.h | |||
| @@ -72,22 +72,17 @@ extern void io_dpc(unsigned long ref_data); | |||
| 72 | /* | 72 | /* |
| 73 | * ======== io_mbox_msg ======== | 73 | * ======== io_mbox_msg ======== |
| 74 | * Purpose: | 74 | * Purpose: |
| 75 | * Main interrupt handler for the shared memory Bridge channel manager. | 75 | * Main message handler for the shared memory Bridge channel manager. |
| 76 | * Calls the Bridge's chnlsm_isr to determine if this interrupt is ours, | 76 | * Determine if this message is ours, then schedules a DPC to |
| 77 | * then schedules a DPC to dispatch I/O. | 77 | * dispatch I/O. |
| 78 | * Parameters: | 78 | * Parameters: |
| 79 | * ref_data: Pointer to the channel manager object for this board. | 79 | * self: Pointer to its own notifier_block struct. |
| 80 | * Set in an initial call to ISR_Install(). | 80 | * len: Length of message. |
| 81 | * msg: Message code received. | ||
| 81 | * Returns: | 82 | * Returns: |
| 82 | * TRUE if interrupt handled; FALSE otherwise. | 83 | * NOTIFY_OK if handled; NOTIFY_BAD otherwise. |
| 83 | * Requires: | ||
| 84 | * Must be in locked memory if executing in kernel mode. | ||
| 85 | * Must only call functions which are in locked memory if Kernel mode. | ||
| 86 | * Must only call asynchronous services. | ||
| 87 | * Interrupts are disabled and EOI for this interrupt has been sent. | ||
| 88 | * Ensures: | ||
| 89 | */ | 84 | */ |
| 90 | void io_mbox_msg(u32 msg); | 85 | int io_mbox_msg(struct notifier_block *self, unsigned long len, void *msg); |
| 91 | 86 | ||
| 92 | /* | 87 | /* |
| 93 | * ======== io_request_chnl ======== | 88 | * ======== io_request_chnl ======== |
diff --git a/drivers/staging/usbip/stub.h b/drivers/staging/usbip/stub.h index 30dbfb6d16f..d73267961ef 100644 --- a/drivers/staging/usbip/stub.h +++ b/drivers/staging/usbip/stub.h | |||
| @@ -32,6 +32,7 @@ | |||
| 32 | 32 | ||
| 33 | struct stub_device { | 33 | struct stub_device { |
| 34 | struct usb_interface *interface; | 34 | struct usb_interface *interface; |
| 35 | struct usb_device *udev; | ||
| 35 | struct list_head list; | 36 | struct list_head list; |
| 36 | 37 | ||
| 37 | struct usbip_device ud; | 38 | struct usbip_device ud; |
diff --git a/drivers/staging/usbip/stub_dev.c b/drivers/staging/usbip/stub_dev.c index b186b5fed2b..a7ce51cc890 100644 --- a/drivers/staging/usbip/stub_dev.c +++ b/drivers/staging/usbip/stub_dev.c | |||
| @@ -258,10 +258,11 @@ static void stub_shutdown_connection(struct usbip_device *ud) | |||
| 258 | static void stub_device_reset(struct usbip_device *ud) | 258 | static void stub_device_reset(struct usbip_device *ud) |
| 259 | { | 259 | { |
| 260 | struct stub_device *sdev = container_of(ud, struct stub_device, ud); | 260 | struct stub_device *sdev = container_of(ud, struct stub_device, ud); |
| 261 | struct usb_device *udev = interface_to_usbdev(sdev->interface); | 261 | struct usb_device *udev = sdev->udev; |
| 262 | int ret; | 262 | int ret; |
| 263 | 263 | ||
| 264 | usbip_udbg("device reset"); | 264 | usbip_udbg("device reset"); |
| 265 | |||
| 265 | ret = usb_lock_device_for_reset(udev, sdev->interface); | 266 | ret = usb_lock_device_for_reset(udev, sdev->interface); |
| 266 | if (ret < 0) { | 267 | if (ret < 0) { |
| 267 | dev_err(&udev->dev, "lock for reset\n"); | 268 | dev_err(&udev->dev, "lock for reset\n"); |
| @@ -309,7 +310,8 @@ static void stub_device_unusable(struct usbip_device *ud) | |||
| 309 | * | 310 | * |
| 310 | * Allocates and initializes a new stub_device struct. | 311 | * Allocates and initializes a new stub_device struct. |
| 311 | */ | 312 | */ |
| 312 | static struct stub_device *stub_device_alloc(struct usb_interface *interface) | 313 | static struct stub_device *stub_device_alloc(struct usb_device *udev, |
| 314 | struct usb_interface *interface) | ||
| 313 | { | 315 | { |
| 314 | struct stub_device *sdev; | 316 | struct stub_device *sdev; |
| 315 | int busnum = interface_to_busnum(interface); | 317 | int busnum = interface_to_busnum(interface); |
| @@ -324,7 +326,8 @@ static struct stub_device *stub_device_alloc(struct usb_interface *interface) | |||
| 324 | return NULL; | 326 | return NULL; |
| 325 | } | 327 | } |
| 326 | 328 | ||
| 327 | sdev->interface = interface; | 329 | sdev->interface = usb_get_intf(interface); |
| 330 | sdev->udev = usb_get_dev(udev); | ||
| 328 | 331 | ||
| 329 | /* | 332 | /* |
| 330 | * devid is defined with devnum when this driver is first allocated. | 333 | * devid is defined with devnum when this driver is first allocated. |
| @@ -450,11 +453,12 @@ static int stub_probe(struct usb_interface *interface, | |||
| 450 | return err; | 453 | return err; |
| 451 | } | 454 | } |
| 452 | 455 | ||
| 456 | usb_get_intf(interface); | ||
| 453 | return 0; | 457 | return 0; |
| 454 | } | 458 | } |
| 455 | 459 | ||
| 456 | /* ok. this is my device. */ | 460 | /* ok. this is my device. */ |
| 457 | sdev = stub_device_alloc(interface); | 461 | sdev = stub_device_alloc(udev, interface); |
| 458 | if (!sdev) | 462 | if (!sdev) |
| 459 | return -ENOMEM; | 463 | return -ENOMEM; |
| 460 | 464 | ||
| @@ -476,6 +480,8 @@ static int stub_probe(struct usb_interface *interface, | |||
| 476 | dev_err(&interface->dev, "create sysfs files for %s\n", | 480 | dev_err(&interface->dev, "create sysfs files for %s\n", |
| 477 | udev_busid); | 481 | udev_busid); |
| 478 | usb_set_intfdata(interface, NULL); | 482 | usb_set_intfdata(interface, NULL); |
| 483 | usb_put_intf(interface); | ||
| 484 | |||
| 479 | busid_priv->interf_count = 0; | 485 | busid_priv->interf_count = 0; |
| 480 | 486 | ||
| 481 | busid_priv->sdev = NULL; | 487 | busid_priv->sdev = NULL; |
| @@ -545,6 +551,7 @@ static void stub_disconnect(struct usb_interface *interface) | |||
| 545 | if (busid_priv->interf_count > 1) { | 551 | if (busid_priv->interf_count > 1) { |
| 546 | busid_priv->interf_count--; | 552 | busid_priv->interf_count--; |
| 547 | shutdown_busid(busid_priv); | 553 | shutdown_busid(busid_priv); |
| 554 | usb_put_intf(interface); | ||
| 548 | return; | 555 | return; |
| 549 | } | 556 | } |
| 550 | 557 | ||
| @@ -554,6 +561,9 @@ static void stub_disconnect(struct usb_interface *interface) | |||
| 554 | /* 1. shutdown the current connection */ | 561 | /* 1. shutdown the current connection */ |
| 555 | shutdown_busid(busid_priv); | 562 | shutdown_busid(busid_priv); |
| 556 | 563 | ||
| 564 | usb_put_dev(sdev->udev); | ||
| 565 | usb_put_intf(interface); | ||
| 566 | |||
| 557 | /* 3. free sdev */ | 567 | /* 3. free sdev */ |
| 558 | busid_priv->sdev = NULL; | 568 | busid_priv->sdev = NULL; |
| 559 | stub_device_free(sdev); | 569 | stub_device_free(sdev); |
diff --git a/drivers/staging/usbip/stub_rx.c b/drivers/staging/usbip/stub_rx.c index 3de6fd2539d..ae6ac82754a 100644 --- a/drivers/staging/usbip/stub_rx.c +++ b/drivers/staging/usbip/stub_rx.c | |||
| @@ -364,7 +364,7 @@ static struct stub_priv *stub_priv_alloc(struct stub_device *sdev, | |||
| 364 | 364 | ||
| 365 | static int get_pipe(struct stub_device *sdev, int epnum, int dir) | 365 | static int get_pipe(struct stub_device *sdev, int epnum, int dir) |
| 366 | { | 366 | { |
| 367 | struct usb_device *udev = interface_to_usbdev(sdev->interface); | 367 | struct usb_device *udev = sdev->udev; |
| 368 | struct usb_host_endpoint *ep; | 368 | struct usb_host_endpoint *ep; |
| 369 | struct usb_endpoint_descriptor *epd = NULL; | 369 | struct usb_endpoint_descriptor *epd = NULL; |
| 370 | 370 | ||
| @@ -484,7 +484,7 @@ static void stub_recv_cmd_submit(struct stub_device *sdev, | |||
| 484 | int ret; | 484 | int ret; |
| 485 | struct stub_priv *priv; | 485 | struct stub_priv *priv; |
| 486 | struct usbip_device *ud = &sdev->ud; | 486 | struct usbip_device *ud = &sdev->ud; |
| 487 | struct usb_device *udev = interface_to_usbdev(sdev->interface); | 487 | struct usb_device *udev = sdev->udev; |
| 488 | int pipe = get_pipe(sdev, pdu->base.ep, pdu->base.direction); | 488 | int pipe = get_pipe(sdev, pdu->base.ep, pdu->base.direction); |
| 489 | 489 | ||
| 490 | 490 | ||
diff --git a/drivers/staging/usbip/vhci.h b/drivers/staging/usbip/vhci.h index 41a1fe5138f..afc3b1a7188 100644 --- a/drivers/staging/usbip/vhci.h +++ b/drivers/staging/usbip/vhci.h | |||
| @@ -100,9 +100,6 @@ struct vhci_hcd { | |||
| 100 | * But, the index of this array begins from 0. | 100 | * But, the index of this array begins from 0. |
| 101 | */ | 101 | */ |
| 102 | struct vhci_device vdev[VHCI_NPORTS]; | 102 | struct vhci_device vdev[VHCI_NPORTS]; |
| 103 | |||
| 104 | /* vhci_device which has not been assiged its address yet */ | ||
| 105 | int pending_port; | ||
| 106 | }; | 103 | }; |
| 107 | 104 | ||
| 108 | 105 | ||
| @@ -119,6 +116,9 @@ void rh_port_disconnect(int rhport); | |||
| 119 | void vhci_rx_loop(struct usbip_task *ut); | 116 | void vhci_rx_loop(struct usbip_task *ut); |
| 120 | void vhci_tx_loop(struct usbip_task *ut); | 117 | void vhci_tx_loop(struct usbip_task *ut); |
| 121 | 118 | ||
| 119 | struct urb *pickup_urb_and_free_priv(struct vhci_device *vdev, | ||
| 120 | __u32 seqnum); | ||
| 121 | |||
| 122 | #define hardware (&the_controller->pdev.dev) | 122 | #define hardware (&the_controller->pdev.dev) |
| 123 | 123 | ||
| 124 | static inline struct vhci_device *port_to_vdev(__u32 port) | 124 | static inline struct vhci_device *port_to_vdev(__u32 port) |
diff --git a/drivers/staging/usbip/vhci_hcd.c b/drivers/staging/usbip/vhci_hcd.c index 08bd26a245d..a35fe61268d 100644 --- a/drivers/staging/usbip/vhci_hcd.c +++ b/drivers/staging/usbip/vhci_hcd.c | |||
| @@ -138,8 +138,6 @@ void rh_port_connect(int rhport, enum usb_device_speed speed) | |||
| 138 | * the_controller->vdev[rhport].ud.status = VDEV_CONNECT; | 138 | * the_controller->vdev[rhport].ud.status = VDEV_CONNECT; |
| 139 | * spin_unlock(&the_controller->vdev[rhport].ud.lock); */ | 139 | * spin_unlock(&the_controller->vdev[rhport].ud.lock); */ |
| 140 | 140 | ||
| 141 | the_controller->pending_port = rhport; | ||
| 142 | |||
| 143 | spin_unlock_irqrestore(&the_controller->lock, flags); | 141 | spin_unlock_irqrestore(&the_controller->lock, flags); |
| 144 | 142 | ||
| 145 | usb_hcd_poll_rh_status(vhci_to_hcd(the_controller)); | 143 | usb_hcd_poll_rh_status(vhci_to_hcd(the_controller)); |
| @@ -559,6 +557,7 @@ static int vhci_urb_enqueue(struct usb_hcd *hcd, struct urb *urb, | |||
| 559 | struct device *dev = &urb->dev->dev; | 557 | struct device *dev = &urb->dev->dev; |
| 560 | int ret = 0; | 558 | int ret = 0; |
| 561 | unsigned long flags; | 559 | unsigned long flags; |
| 560 | struct vhci_device *vdev; | ||
| 562 | 561 | ||
| 563 | usbip_dbg_vhci_hc("enter, usb_hcd %p urb %p mem_flags %d\n", | 562 | usbip_dbg_vhci_hc("enter, usb_hcd %p urb %p mem_flags %d\n", |
| 564 | hcd, urb, mem_flags); | 563 | hcd, urb, mem_flags); |
| @@ -574,6 +573,18 @@ static int vhci_urb_enqueue(struct usb_hcd *hcd, struct urb *urb, | |||
| 574 | return urb->status; | 573 | return urb->status; |
| 575 | } | 574 | } |
| 576 | 575 | ||
| 576 | vdev = port_to_vdev(urb->dev->portnum-1); | ||
| 577 | |||
| 578 | /* refuse enqueue for dead connection */ | ||
| 579 | spin_lock(&vdev->ud.lock); | ||
| 580 | if (vdev->ud.status == VDEV_ST_NULL || vdev->ud.status == VDEV_ST_ERROR) { | ||
| 581 | usbip_uerr("enqueue for inactive port %d\n", vdev->rhport); | ||
| 582 | spin_unlock(&vdev->ud.lock); | ||
| 583 | spin_unlock_irqrestore(&the_controller->lock, flags); | ||
| 584 | return -ENODEV; | ||
| 585 | } | ||
| 586 | spin_unlock(&vdev->ud.lock); | ||
| 587 | |||
| 577 | ret = usb_hcd_link_urb_to_ep(hcd, urb); | 588 | ret = usb_hcd_link_urb_to_ep(hcd, urb); |
| 578 | if (ret) | 589 | if (ret) |
| 579 | goto no_need_unlink; | 590 | goto no_need_unlink; |
| @@ -592,8 +603,6 @@ static int vhci_urb_enqueue(struct usb_hcd *hcd, struct urb *urb, | |||
| 592 | __u8 type = usb_pipetype(urb->pipe); | 603 | __u8 type = usb_pipetype(urb->pipe); |
| 593 | struct usb_ctrlrequest *ctrlreq = | 604 | struct usb_ctrlrequest *ctrlreq = |
| 594 | (struct usb_ctrlrequest *) urb->setup_packet; | 605 | (struct usb_ctrlrequest *) urb->setup_packet; |
| 595 | struct vhci_device *vdev = | ||
| 596 | port_to_vdev(the_controller->pending_port); | ||
| 597 | 606 | ||
| 598 | if (type != PIPE_CONTROL || !ctrlreq) { | 607 | if (type != PIPE_CONTROL || !ctrlreq) { |
| 599 | dev_err(dev, "invalid request to devnum 0\n"); | 608 | dev_err(dev, "invalid request to devnum 0\n"); |
| @@ -607,7 +616,9 @@ static int vhci_urb_enqueue(struct usb_hcd *hcd, struct urb *urb, | |||
| 607 | dev_info(dev, "SetAddress Request (%d) to port %d\n", | 616 | dev_info(dev, "SetAddress Request (%d) to port %d\n", |
| 608 | ctrlreq->wValue, vdev->rhport); | 617 | ctrlreq->wValue, vdev->rhport); |
| 609 | 618 | ||
| 610 | vdev->udev = urb->dev; | 619 | if (vdev->udev) |
| 620 | usb_put_dev(vdev->udev); | ||
| 621 | vdev->udev = usb_get_dev(urb->dev); | ||
| 611 | 622 | ||
| 612 | spin_lock(&vdev->ud.lock); | 623 | spin_lock(&vdev->ud.lock); |
| 613 | vdev->ud.status = VDEV_ST_USED; | 624 | vdev->ud.status = VDEV_ST_USED; |
| @@ -627,8 +638,9 @@ static int vhci_urb_enqueue(struct usb_hcd *hcd, struct urb *urb, | |||
| 627 | "Get_Descriptor to device 0 " | 638 | "Get_Descriptor to device 0 " |
| 628 | "(get max pipe size)\n"); | 639 | "(get max pipe size)\n"); |
| 629 | 640 | ||
| 630 | /* FIXME: reference count? (usb_get_dev()) */ | 641 | if (vdev->udev) |
| 631 | vdev->udev = urb->dev; | 642 | usb_put_dev(vdev->udev); |
| 643 | vdev->udev = usb_get_dev(urb->dev); | ||
| 632 | goto out; | 644 | goto out; |
| 633 | 645 | ||
| 634 | default: | 646 | default: |
| @@ -805,7 +817,6 @@ static int vhci_urb_dequeue(struct usb_hcd *hcd, struct urb *urb, int status) | |||
| 805 | return 0; | 817 | return 0; |
| 806 | } | 818 | } |
| 807 | 819 | ||
| 808 | |||
| 809 | static void vhci_device_unlink_cleanup(struct vhci_device *vdev) | 820 | static void vhci_device_unlink_cleanup(struct vhci_device *vdev) |
| 810 | { | 821 | { |
| 811 | struct vhci_unlink *unlink, *tmp; | 822 | struct vhci_unlink *unlink, *tmp; |
| @@ -813,11 +824,34 @@ static void vhci_device_unlink_cleanup(struct vhci_device *vdev) | |||
| 813 | spin_lock(&vdev->priv_lock); | 824 | spin_lock(&vdev->priv_lock); |
| 814 | 825 | ||
| 815 | list_for_each_entry_safe(unlink, tmp, &vdev->unlink_tx, list) { | 826 | list_for_each_entry_safe(unlink, tmp, &vdev->unlink_tx, list) { |
| 827 | usbip_uinfo("unlink cleanup tx %lu\n", unlink->unlink_seqnum); | ||
| 816 | list_del(&unlink->list); | 828 | list_del(&unlink->list); |
| 817 | kfree(unlink); | 829 | kfree(unlink); |
| 818 | } | 830 | } |
| 819 | 831 | ||
| 820 | list_for_each_entry_safe(unlink, tmp, &vdev->unlink_rx, list) { | 832 | list_for_each_entry_safe(unlink, tmp, &vdev->unlink_rx, list) { |
| 833 | struct urb *urb; | ||
| 834 | |||
| 835 | /* give back URB of unanswered unlink request */ | ||
| 836 | usbip_uinfo("unlink cleanup rx %lu\n", unlink->unlink_seqnum); | ||
| 837 | |||
| 838 | urb = pickup_urb_and_free_priv(vdev, unlink->unlink_seqnum); | ||
| 839 | if (!urb) { | ||
| 840 | usbip_uinfo("the urb (seqnum %lu) was already given back\n", | ||
| 841 | unlink->unlink_seqnum); | ||
| 842 | list_del(&unlink->list); | ||
| 843 | kfree(unlink); | ||
| 844 | continue; | ||
| 845 | } | ||
| 846 | |||
| 847 | urb->status = -ENODEV; | ||
| 848 | |||
| 849 | spin_lock(&the_controller->lock); | ||
| 850 | usb_hcd_unlink_urb_from_ep(vhci_to_hcd(the_controller), urb); | ||
| 851 | spin_unlock(&the_controller->lock); | ||
| 852 | |||
| 853 | usb_hcd_giveback_urb(vhci_to_hcd(the_controller), urb, urb->status); | ||
| 854 | |||
| 821 | list_del(&unlink->list); | 855 | list_del(&unlink->list); |
| 822 | kfree(unlink); | 856 | kfree(unlink); |
| 823 | } | 857 | } |
| @@ -887,6 +921,10 @@ static void vhci_device_reset(struct usbip_device *ud) | |||
| 887 | vdev->speed = 0; | 921 | vdev->speed = 0; |
| 888 | vdev->devid = 0; | 922 | vdev->devid = 0; |
| 889 | 923 | ||
| 924 | if (vdev->udev) | ||
| 925 | usb_put_dev(vdev->udev); | ||
| 926 | vdev->udev = NULL; | ||
| 927 | |||
| 890 | ud->tcp_socket = NULL; | 928 | ud->tcp_socket = NULL; |
| 891 | 929 | ||
| 892 | ud->status = VDEV_ST_NULL; | 930 | ud->status = VDEV_ST_NULL; |
diff --git a/drivers/staging/usbip/vhci_rx.c b/drivers/staging/usbip/vhci_rx.c index 8147d7202b2..bf699147094 100644 --- a/drivers/staging/usbip/vhci_rx.c +++ b/drivers/staging/usbip/vhci_rx.c | |||
| @@ -23,16 +23,14 @@ | |||
| 23 | #include "vhci.h" | 23 | #include "vhci.h" |
| 24 | 24 | ||
| 25 | 25 | ||
| 26 | /* get URB from transmitted urb queue */ | 26 | /* get URB from transmitted urb queue. caller must hold vdev->priv_lock */ |
| 27 | static struct urb *pickup_urb_and_free_priv(struct vhci_device *vdev, | 27 | struct urb *pickup_urb_and_free_priv(struct vhci_device *vdev, |
| 28 | __u32 seqnum) | 28 | __u32 seqnum) |
| 29 | { | 29 | { |
| 30 | struct vhci_priv *priv, *tmp; | 30 | struct vhci_priv *priv, *tmp; |
| 31 | struct urb *urb = NULL; | 31 | struct urb *urb = NULL; |
| 32 | int status; | 32 | int status; |
| 33 | 33 | ||
| 34 | spin_lock(&vdev->priv_lock); | ||
| 35 | |||
| 36 | list_for_each_entry_safe(priv, tmp, &vdev->priv_rx, list) { | 34 | list_for_each_entry_safe(priv, tmp, &vdev->priv_rx, list) { |
| 37 | if (priv->seqnum == seqnum) { | 35 | if (priv->seqnum == seqnum) { |
| 38 | urb = priv->urb; | 36 | urb = priv->urb; |
| @@ -63,8 +61,6 @@ static struct urb *pickup_urb_and_free_priv(struct vhci_device *vdev, | |||
| 63 | } | 61 | } |
| 64 | } | 62 | } |
| 65 | 63 | ||
| 66 | spin_unlock(&vdev->priv_lock); | ||
| 67 | |||
| 68 | return urb; | 64 | return urb; |
| 69 | } | 65 | } |
| 70 | 66 | ||
| @@ -74,9 +70,11 @@ static void vhci_recv_ret_submit(struct vhci_device *vdev, | |||
| 74 | struct usbip_device *ud = &vdev->ud; | 70 | struct usbip_device *ud = &vdev->ud; |
| 75 | struct urb *urb; | 71 | struct urb *urb; |
| 76 | 72 | ||
| 73 | spin_lock(&vdev->priv_lock); | ||
| 77 | 74 | ||
| 78 | urb = pickup_urb_and_free_priv(vdev, pdu->base.seqnum); | 75 | urb = pickup_urb_and_free_priv(vdev, pdu->base.seqnum); |
| 79 | 76 | ||
| 77 | spin_unlock(&vdev->priv_lock); | ||
| 80 | 78 | ||
| 81 | if (!urb) { | 79 | if (!urb) { |
| 82 | usbip_uerr("cannot find a urb of seqnum %u\n", | 80 | usbip_uerr("cannot find a urb of seqnum %u\n", |
| @@ -161,7 +159,12 @@ static void vhci_recv_ret_unlink(struct vhci_device *vdev, | |||
| 161 | return; | 159 | return; |
| 162 | } | 160 | } |
| 163 | 161 | ||
| 162 | spin_lock(&vdev->priv_lock); | ||
| 163 | |||
| 164 | urb = pickup_urb_and_free_priv(vdev, unlink->unlink_seqnum); | 164 | urb = pickup_urb_and_free_priv(vdev, unlink->unlink_seqnum); |
| 165 | |||
| 166 | spin_unlock(&vdev->priv_lock); | ||
| 167 | |||
| 165 | if (!urb) { | 168 | if (!urb) { |
| 166 | /* | 169 | /* |
| 167 | * I get the result of a unlink request. But, it seems that I | 170 | * I get the result of a unlink request. But, it seems that I |
| @@ -190,6 +193,19 @@ static void vhci_recv_ret_unlink(struct vhci_device *vdev, | |||
| 190 | return; | 193 | return; |
| 191 | } | 194 | } |
| 192 | 195 | ||
| 196 | static int vhci_priv_tx_empty(struct vhci_device *vdev) | ||
| 197 | { | ||
| 198 | int empty = 0; | ||
| 199 | |||
| 200 | spin_lock(&vdev->priv_lock); | ||
| 201 | |||
| 202 | empty = list_empty(&vdev->priv_rx); | ||
| 203 | |||
| 204 | spin_unlock(&vdev->priv_lock); | ||
| 205 | |||
| 206 | return empty; | ||
| 207 | } | ||
| 208 | |||
| 193 | /* recv a pdu */ | 209 | /* recv a pdu */ |
| 194 | static void vhci_rx_pdu(struct usbip_device *ud) | 210 | static void vhci_rx_pdu(struct usbip_device *ud) |
| 195 | { | 211 | { |
| @@ -202,11 +218,29 @@ static void vhci_rx_pdu(struct usbip_device *ud) | |||
| 202 | 218 | ||
| 203 | memset(&pdu, 0, sizeof(pdu)); | 219 | memset(&pdu, 0, sizeof(pdu)); |
| 204 | 220 | ||
| 205 | |||
| 206 | /* 1. receive a pdu header */ | 221 | /* 1. receive a pdu header */ |
| 207 | ret = usbip_xmit(0, ud->tcp_socket, (char *) &pdu, sizeof(pdu), 0); | 222 | ret = usbip_xmit(0, ud->tcp_socket, (char *) &pdu, sizeof(pdu), 0); |
| 223 | if (ret < 0) { | ||
| 224 | if (ret == -ECONNRESET) | ||
| 225 | usbip_uinfo("connection reset by peer\n"); | ||
| 226 | else if (ret == -EAGAIN) { | ||
| 227 | /* ignore if connection was idle */ | ||
| 228 | if (vhci_priv_tx_empty(vdev)) | ||
| 229 | return; | ||
| 230 | usbip_uinfo("connection timed out with pending urbs\n"); | ||
| 231 | } else if (ret != -ERESTARTSYS) | ||
| 232 | usbip_uinfo("xmit failed %d\n", ret); | ||
| 233 | |||
| 234 | usbip_event_add(ud, VDEV_EVENT_ERROR_TCP); | ||
| 235 | return; | ||
| 236 | } | ||
| 237 | if (ret == 0) { | ||
| 238 | usbip_uinfo("connection closed"); | ||
| 239 | usbip_event_add(ud, VDEV_EVENT_DOWN); | ||
| 240 | return; | ||
| 241 | } | ||
| 208 | if (ret != sizeof(pdu)) { | 242 | if (ret != sizeof(pdu)) { |
| 209 | usbip_uerr("receiving pdu failed! size is %d, should be %d\n", | 243 | usbip_uerr("received pdu size is %d, should be %d\n", |
| 210 | ret, (unsigned int)sizeof(pdu)); | 244 | ret, (unsigned int)sizeof(pdu)); |
| 211 | usbip_event_add(ud, VDEV_EVENT_ERROR_TCP); | 245 | usbip_event_add(ud, VDEV_EVENT_ERROR_TCP); |
| 212 | return; | 246 | return; |
diff --git a/drivers/staging/vme/bridges/Module.symvers b/drivers/staging/vme/bridges/Module.symvers deleted file mode 100644 index e69de29bb2d..00000000000 --- a/drivers/staging/vme/bridges/Module.symvers +++ /dev/null | |||
diff --git a/drivers/staging/xgifb/vb_setmode.c b/drivers/staging/xgifb/vb_setmode.c index 7016fdd2509..e19b932492e 100644 --- a/drivers/staging/xgifb/vb_setmode.c +++ b/drivers/staging/xgifb/vb_setmode.c | |||
| @@ -3954,8 +3954,8 @@ void XGI_GetCRT2ResInfo(unsigned short ModeNo, unsigned short ModeIdIndex, | |||
| 3954 | unsigned char XGI_IsLCDDualLink(struct vb_device_info *pVBInfo) | 3954 | unsigned char XGI_IsLCDDualLink(struct vb_device_info *pVBInfo) |
| 3955 | { | 3955 | { |
| 3956 | 3956 | ||
| 3957 | if ((((pVBInfo->VBInfo & SetCRT2ToLCD) | SetCRT2ToLCDA)) | 3957 | if ((pVBInfo->VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) && |
| 3958 | && (pVBInfo->LCDInfo & SetLCDDualLink)) /* shampoo0129 */ | 3958 | (pVBInfo->LCDInfo & SetLCDDualLink)) /* shampoo0129 */ |
| 3959 | return 1; | 3959 | return 1; |
| 3960 | 3960 | ||
| 3961 | return 0; | 3961 | return 0; |
| @@ -8773,7 +8773,7 @@ unsigned short XGI_GetVCLK2Ptr(unsigned short ModeNo, | |||
| 8773 | 8773 | ||
| 8774 | if (pVBInfo->IF_DEF_LVDS == 0) { | 8774 | if (pVBInfo->IF_DEF_LVDS == 0) { |
| 8775 | CRT2Index = CRT2Index >> 6; /* for LCD */ | 8775 | CRT2Index = CRT2Index >> 6; /* for LCD */ |
| 8776 | if (((pVBInfo->VBInfo & SetCRT2ToLCD) | SetCRT2ToLCDA)) { /*301b*/ | 8776 | if (pVBInfo->VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) { /*301b*/ |
| 8777 | if (pVBInfo->LCDResInfo != Panel1024x768) | 8777 | if (pVBInfo->LCDResInfo != Panel1024x768) |
| 8778 | VCLKIndex = LCDXlat2VCLK[CRT2Index]; | 8778 | VCLKIndex = LCDXlat2VCLK[CRT2Index]; |
| 8779 | else | 8779 | else |
diff --git a/drivers/tty/n_hdlc.c b/drivers/tty/n_hdlc.c index 47d32281032..52fc0c9a636 100644 --- a/drivers/tty/n_hdlc.c +++ b/drivers/tty/n_hdlc.c | |||
| @@ -581,8 +581,9 @@ static ssize_t n_hdlc_tty_read(struct tty_struct *tty, struct file *file, | |||
| 581 | __u8 __user *buf, size_t nr) | 581 | __u8 __user *buf, size_t nr) |
| 582 | { | 582 | { |
| 583 | struct n_hdlc *n_hdlc = tty2n_hdlc(tty); | 583 | struct n_hdlc *n_hdlc = tty2n_hdlc(tty); |
| 584 | int ret; | 584 | int ret = 0; |
| 585 | struct n_hdlc_buf *rbuf; | 585 | struct n_hdlc_buf *rbuf; |
| 586 | DECLARE_WAITQUEUE(wait, current); | ||
| 586 | 587 | ||
| 587 | if (debuglevel >= DEBUG_LEVEL_INFO) | 588 | if (debuglevel >= DEBUG_LEVEL_INFO) |
| 588 | printk("%s(%d)n_hdlc_tty_read() called\n",__FILE__,__LINE__); | 589 | printk("%s(%d)n_hdlc_tty_read() called\n",__FILE__,__LINE__); |
| @@ -598,57 +599,55 @@ static ssize_t n_hdlc_tty_read(struct tty_struct *tty, struct file *file, | |||
| 598 | return -EFAULT; | 599 | return -EFAULT; |
| 599 | } | 600 | } |
| 600 | 601 | ||
| 601 | tty_lock(); | 602 | add_wait_queue(&tty->read_wait, &wait); |
| 602 | 603 | ||
| 603 | for (;;) { | 604 | for (;;) { |
| 604 | if (test_bit(TTY_OTHER_CLOSED, &tty->flags)) { | 605 | if (test_bit(TTY_OTHER_CLOSED, &tty->flags)) { |
| 605 | tty_unlock(); | 606 | ret = -EIO; |
| 606 | return -EIO; | 607 | break; |
| 607 | } | 608 | } |
| 609 | if (tty_hung_up_p(file)) | ||
| 610 | break; | ||
| 608 | 611 | ||
| 609 | n_hdlc = tty2n_hdlc (tty); | 612 | set_current_state(TASK_INTERRUPTIBLE); |
| 610 | if (!n_hdlc || n_hdlc->magic != HDLC_MAGIC || | ||
| 611 | tty != n_hdlc->tty) { | ||
| 612 | tty_unlock(); | ||
| 613 | return 0; | ||
| 614 | } | ||
| 615 | 613 | ||
| 616 | rbuf = n_hdlc_buf_get(&n_hdlc->rx_buf_list); | 614 | rbuf = n_hdlc_buf_get(&n_hdlc->rx_buf_list); |
| 617 | if (rbuf) | 615 | if (rbuf) { |
| 616 | if (rbuf->count > nr) { | ||
| 617 | /* too large for caller's buffer */ | ||
| 618 | ret = -EOVERFLOW; | ||
| 619 | } else { | ||
| 620 | if (copy_to_user(buf, rbuf->buf, rbuf->count)) | ||
| 621 | ret = -EFAULT; | ||
| 622 | else | ||
| 623 | ret = rbuf->count; | ||
| 624 | } | ||
| 625 | |||
| 626 | if (n_hdlc->rx_free_buf_list.count > | ||
| 627 | DEFAULT_RX_BUF_COUNT) | ||
| 628 | kfree(rbuf); | ||
| 629 | else | ||
| 630 | n_hdlc_buf_put(&n_hdlc->rx_free_buf_list, rbuf); | ||
| 618 | break; | 631 | break; |
| 632 | } | ||
| 619 | 633 | ||
| 620 | /* no data */ | 634 | /* no data */ |
| 621 | if (file->f_flags & O_NONBLOCK) { | 635 | if (file->f_flags & O_NONBLOCK) { |
| 622 | tty_unlock(); | 636 | ret = -EAGAIN; |
| 623 | return -EAGAIN; | 637 | break; |
| 624 | } | 638 | } |
| 625 | 639 | ||
| 626 | interruptible_sleep_on (&tty->read_wait); | 640 | schedule(); |
| 641 | |||
| 627 | if (signal_pending(current)) { | 642 | if (signal_pending(current)) { |
| 628 | tty_unlock(); | 643 | ret = -EINTR; |
| 629 | return -EINTR; | 644 | break; |
| 630 | } | 645 | } |
| 631 | } | 646 | } |
| 632 | 647 | ||
| 633 | if (rbuf->count > nr) | 648 | remove_wait_queue(&tty->read_wait, &wait); |
| 634 | /* frame too large for caller's buffer (discard frame) */ | 649 | __set_current_state(TASK_RUNNING); |
| 635 | ret = -EOVERFLOW; | 650 | |
| 636 | else { | ||
| 637 | /* Copy the data to the caller's buffer */ | ||
| 638 | if (copy_to_user(buf, rbuf->buf, rbuf->count)) | ||
| 639 | ret = -EFAULT; | ||
| 640 | else | ||
| 641 | ret = rbuf->count; | ||
| 642 | } | ||
| 643 | |||
| 644 | /* return HDLC buffer to free list unless the free list */ | ||
| 645 | /* count has exceeded the default value, in which case the */ | ||
| 646 | /* buffer is freed back to the OS to conserve memory */ | ||
| 647 | if (n_hdlc->rx_free_buf_list.count > DEFAULT_RX_BUF_COUNT) | ||
| 648 | kfree(rbuf); | ||
| 649 | else | ||
| 650 | n_hdlc_buf_put(&n_hdlc->rx_free_buf_list,rbuf); | ||
| 651 | tty_unlock(); | ||
| 652 | return ret; | 651 | return ret; |
| 653 | 652 | ||
| 654 | } /* end of n_hdlc_tty_read() */ | 653 | } /* end of n_hdlc_tty_read() */ |
| @@ -691,14 +690,15 @@ static ssize_t n_hdlc_tty_write(struct tty_struct *tty, struct file *file, | |||
| 691 | count = maxframe; | 690 | count = maxframe; |
| 692 | } | 691 | } |
| 693 | 692 | ||
| 694 | tty_lock(); | ||
| 695 | |||
| 696 | add_wait_queue(&tty->write_wait, &wait); | 693 | add_wait_queue(&tty->write_wait, &wait); |
| 697 | set_current_state(TASK_INTERRUPTIBLE); | 694 | |
| 695 | for (;;) { | ||
| 696 | set_current_state(TASK_INTERRUPTIBLE); | ||
| 698 | 697 | ||
| 699 | /* Allocate transmit buffer */ | 698 | tbuf = n_hdlc_buf_get(&n_hdlc->tx_free_buf_list); |
| 700 | /* sleep until transmit buffer available */ | 699 | if (tbuf) |
| 701 | while (!(tbuf = n_hdlc_buf_get(&n_hdlc->tx_free_buf_list))) { | 700 | break; |
| 701 | |||
| 702 | if (file->f_flags & O_NONBLOCK) { | 702 | if (file->f_flags & O_NONBLOCK) { |
| 703 | error = -EAGAIN; | 703 | error = -EAGAIN; |
| 704 | break; | 704 | break; |
| @@ -719,7 +719,7 @@ static ssize_t n_hdlc_tty_write(struct tty_struct *tty, struct file *file, | |||
| 719 | } | 719 | } |
| 720 | } | 720 | } |
| 721 | 721 | ||
| 722 | set_current_state(TASK_RUNNING); | 722 | __set_current_state(TASK_RUNNING); |
| 723 | remove_wait_queue(&tty->write_wait, &wait); | 723 | remove_wait_queue(&tty->write_wait, &wait); |
| 724 | 724 | ||
| 725 | if (!error) { | 725 | if (!error) { |
| @@ -731,7 +731,7 @@ static ssize_t n_hdlc_tty_write(struct tty_struct *tty, struct file *file, | |||
| 731 | n_hdlc_buf_put(&n_hdlc->tx_buf_list,tbuf); | 731 | n_hdlc_buf_put(&n_hdlc->tx_buf_list,tbuf); |
| 732 | n_hdlc_send_frames(n_hdlc,tty); | 732 | n_hdlc_send_frames(n_hdlc,tty); |
| 733 | } | 733 | } |
| 734 | tty_unlock(); | 734 | |
| 735 | return error; | 735 | return error; |
| 736 | 736 | ||
| 737 | } /* end of n_hdlc_tty_write() */ | 737 | } /* end of n_hdlc_tty_write() */ |
diff --git a/drivers/tty/serial/8250.c b/drivers/tty/serial/8250.c index b25e6e49053..3975df6f7fd 100644 --- a/drivers/tty/serial/8250.c +++ b/drivers/tty/serial/8250.c | |||
| @@ -236,7 +236,8 @@ static const struct serial8250_config uart_config[] = { | |||
| 236 | .fifo_size = 128, | 236 | .fifo_size = 128, |
| 237 | .tx_loadsz = 128, | 237 | .tx_loadsz = 128, |
| 238 | .fcr = UART_FCR_ENABLE_FIFO | UART_FCR_R_TRIG_10, | 238 | .fcr = UART_FCR_ENABLE_FIFO | UART_FCR_R_TRIG_10, |
| 239 | .flags = UART_CAP_FIFO | UART_CAP_EFR | UART_CAP_SLEEP, | 239 | /* UART_CAP_EFR breaks billionon CF bluetooth card. */ |
| 240 | .flags = UART_CAP_FIFO | UART_CAP_SLEEP, | ||
| 240 | }, | 241 | }, |
| 241 | [PORT_16654] = { | 242 | [PORT_16654] = { |
| 242 | .name = "ST16654", | 243 | .name = "ST16654", |
diff --git a/drivers/tty/serial/Kconfig b/drivers/tty/serial/Kconfig index b1682d7f1d8..2b8334601c8 100644 --- a/drivers/tty/serial/Kconfig +++ b/drivers/tty/serial/Kconfig | |||
| @@ -1518,6 +1518,7 @@ config SERIAL_BCM63XX_CONSOLE | |||
| 1518 | config SERIAL_GRLIB_GAISLER_APBUART | 1518 | config SERIAL_GRLIB_GAISLER_APBUART |
| 1519 | tristate "GRLIB APBUART serial support" | 1519 | tristate "GRLIB APBUART serial support" |
| 1520 | depends on OF | 1520 | depends on OF |
| 1521 | select SERIAL_CORE | ||
| 1521 | ---help--- | 1522 | ---help--- |
| 1522 | Add support for the GRLIB APBUART serial port. | 1523 | Add support for the GRLIB APBUART serial port. |
| 1523 | 1524 | ||
diff --git a/drivers/tty/serial/sb1250-duart.c b/drivers/tty/serial/sb1250-duart.c index a2f2b325449..602d9845c52 100644 --- a/drivers/tty/serial/sb1250-duart.c +++ b/drivers/tty/serial/sb1250-duart.c | |||
| @@ -829,7 +829,7 @@ static void __init sbd_probe_duarts(void) | |||
| 829 | #ifdef CONFIG_SERIAL_SB1250_DUART_CONSOLE | 829 | #ifdef CONFIG_SERIAL_SB1250_DUART_CONSOLE |
| 830 | /* | 830 | /* |
| 831 | * Serial console stuff. Very basic, polling driver for doing serial | 831 | * Serial console stuff. Very basic, polling driver for doing serial |
| 832 | * console output. The console_sem is held by the caller, so we | 832 | * console output. The console_lock is held by the caller, so we |
| 833 | * shouldn't be interrupted for more console activity. | 833 | * shouldn't be interrupted for more console activity. |
| 834 | */ | 834 | */ |
| 835 | static void sbd_console_putchar(struct uart_port *uport, int ch) | 835 | static void sbd_console_putchar(struct uart_port *uport, int ch) |
diff --git a/drivers/tty/sysrq.c b/drivers/tty/sysrq.c index c556ed9db13..8e0dd254eb1 100644 --- a/drivers/tty/sysrq.c +++ b/drivers/tty/sysrq.c | |||
| @@ -46,7 +46,7 @@ | |||
| 46 | #include <asm/irq_regs.h> | 46 | #include <asm/irq_regs.h> |
| 47 | 47 | ||
| 48 | /* Whether we react on sysrq keys or just ignore them */ | 48 | /* Whether we react on sysrq keys or just ignore them */ |
| 49 | static int __read_mostly sysrq_enabled = 1; | 49 | static int __read_mostly sysrq_enabled = SYSRQ_DEFAULT_ENABLE; |
| 50 | static bool __read_mostly sysrq_always_enabled; | 50 | static bool __read_mostly sysrq_always_enabled; |
| 51 | 51 | ||
| 52 | static bool sysrq_on(void) | 52 | static bool sysrq_on(void) |
diff --git a/drivers/tty/tty_io.c b/drivers/tty/tty_io.c index 464d09d9787..0065da4b11c 100644 --- a/drivers/tty/tty_io.c +++ b/drivers/tty/tty_io.c | |||
| @@ -3256,8 +3256,8 @@ static ssize_t show_cons_active(struct device *dev, | |||
| 3256 | struct console *c; | 3256 | struct console *c; |
| 3257 | ssize_t count = 0; | 3257 | ssize_t count = 0; |
| 3258 | 3258 | ||
| 3259 | acquire_console_sem(); | 3259 | console_lock(); |
| 3260 | for (c = console_drivers; c; c = c->next) { | 3260 | for_each_console(c) { |
| 3261 | if (!c->device) | 3261 | if (!c->device) |
| 3262 | continue; | 3262 | continue; |
| 3263 | if (!c->write) | 3263 | if (!c->write) |
| @@ -3271,7 +3271,7 @@ static ssize_t show_cons_active(struct device *dev, | |||
| 3271 | while (i--) | 3271 | while (i--) |
| 3272 | count += sprintf(buf + count, "%s%d%c", | 3272 | count += sprintf(buf + count, "%s%d%c", |
| 3273 | cs[i]->name, cs[i]->index, i ? ' ':'\n'); | 3273 | cs[i]->name, cs[i]->index, i ? ' ':'\n'); |
| 3274 | release_console_sem(); | 3274 | console_unlock(); |
| 3275 | 3275 | ||
| 3276 | return count; | 3276 | return count; |
| 3277 | } | 3277 | } |
| @@ -3306,7 +3306,7 @@ int __init tty_init(void) | |||
| 3306 | if (IS_ERR(consdev)) | 3306 | if (IS_ERR(consdev)) |
| 3307 | consdev = NULL; | 3307 | consdev = NULL; |
| 3308 | else | 3308 | else |
| 3309 | device_create_file(consdev, &dev_attr_active); | 3309 | WARN_ON(device_create_file(consdev, &dev_attr_active) < 0); |
| 3310 | 3310 | ||
| 3311 | #ifdef CONFIG_VT | 3311 | #ifdef CONFIG_VT |
| 3312 | vty_init(&console_fops); | 3312 | vty_init(&console_fops); |
diff --git a/drivers/tty/vt/selection.c b/drivers/tty/vt/selection.c index ebae344ce91..c956ed6c83a 100644 --- a/drivers/tty/vt/selection.c +++ b/drivers/tty/vt/selection.c | |||
| @@ -316,9 +316,9 @@ int paste_selection(struct tty_struct *tty) | |||
| 316 | /* always called with BTM from vt_ioctl */ | 316 | /* always called with BTM from vt_ioctl */ |
| 317 | WARN_ON(!tty_locked()); | 317 | WARN_ON(!tty_locked()); |
| 318 | 318 | ||
| 319 | acquire_console_sem(); | 319 | console_lock(); |
| 320 | poke_blanked_console(); | 320 | poke_blanked_console(); |
| 321 | release_console_sem(); | 321 | console_unlock(); |
| 322 | 322 | ||
| 323 | ld = tty_ldisc_ref(tty); | 323 | ld = tty_ldisc_ref(tty); |
| 324 | if (!ld) { | 324 | if (!ld) { |
diff --git a/drivers/tty/vt/vc_screen.c b/drivers/tty/vt/vc_screen.c index eab3a1ff99e..a672ed192d3 100644 --- a/drivers/tty/vt/vc_screen.c +++ b/drivers/tty/vt/vc_screen.c | |||
| @@ -202,7 +202,7 @@ vcs_read(struct file *file, char __user *buf, size_t count, loff_t *ppos) | |||
| 202 | /* Select the proper current console and verify | 202 | /* Select the proper current console and verify |
| 203 | * sanity of the situation under the console lock. | 203 | * sanity of the situation under the console lock. |
| 204 | */ | 204 | */ |
| 205 | acquire_console_sem(); | 205 | console_lock(); |
| 206 | 206 | ||
| 207 | attr = (currcons & 128); | 207 | attr = (currcons & 128); |
| 208 | currcons = (currcons & 127); | 208 | currcons = (currcons & 127); |
| @@ -336,9 +336,9 @@ vcs_read(struct file *file, char __user *buf, size_t count, loff_t *ppos) | |||
| 336 | * the pagefault handling code may want to call printk(). | 336 | * the pagefault handling code may want to call printk(). |
| 337 | */ | 337 | */ |
| 338 | 338 | ||
| 339 | release_console_sem(); | 339 | console_unlock(); |
| 340 | ret = copy_to_user(buf, con_buf_start, orig_count); | 340 | ret = copy_to_user(buf, con_buf_start, orig_count); |
| 341 | acquire_console_sem(); | 341 | console_lock(); |
| 342 | 342 | ||
| 343 | if (ret) { | 343 | if (ret) { |
| 344 | read += (orig_count - ret); | 344 | read += (orig_count - ret); |
| @@ -354,7 +354,7 @@ vcs_read(struct file *file, char __user *buf, size_t count, loff_t *ppos) | |||
| 354 | if (read) | 354 | if (read) |
| 355 | ret = read; | 355 | ret = read; |
| 356 | unlock_out: | 356 | unlock_out: |
| 357 | release_console_sem(); | 357 | console_unlock(); |
| 358 | mutex_unlock(&con_buf_mtx); | 358 | mutex_unlock(&con_buf_mtx); |
| 359 | return ret; | 359 | return ret; |
| 360 | } | 360 | } |
| @@ -379,7 +379,7 @@ vcs_write(struct file *file, const char __user *buf, size_t count, loff_t *ppos) | |||
| 379 | /* Select the proper current console and verify | 379 | /* Select the proper current console and verify |
| 380 | * sanity of the situation under the console lock. | 380 | * sanity of the situation under the console lock. |
| 381 | */ | 381 | */ |
| 382 | acquire_console_sem(); | 382 | console_lock(); |
| 383 | 383 | ||
| 384 | attr = (currcons & 128); | 384 | attr = (currcons & 128); |
| 385 | currcons = (currcons & 127); | 385 | currcons = (currcons & 127); |
| @@ -414,9 +414,9 @@ vcs_write(struct file *file, const char __user *buf, size_t count, loff_t *ppos) | |||
| 414 | /* Temporarily drop the console lock so that we can read | 414 | /* Temporarily drop the console lock so that we can read |
| 415 | * in the write data from userspace safely. | 415 | * in the write data from userspace safely. |
| 416 | */ | 416 | */ |
| 417 | release_console_sem(); | 417 | console_unlock(); |
| 418 | ret = copy_from_user(con_buf, buf, this_round); | 418 | ret = copy_from_user(con_buf, buf, this_round); |
| 419 | acquire_console_sem(); | 419 | console_lock(); |
| 420 | 420 | ||
| 421 | if (ret) { | 421 | if (ret) { |
| 422 | this_round -= ret; | 422 | this_round -= ret; |
| @@ -542,7 +542,7 @@ vcs_write(struct file *file, const char __user *buf, size_t count, loff_t *ppos) | |||
| 542 | vcs_scr_updated(vc); | 542 | vcs_scr_updated(vc); |
| 543 | 543 | ||
| 544 | unlock_out: | 544 | unlock_out: |
| 545 | release_console_sem(); | 545 | console_unlock(); |
| 546 | 546 | ||
| 547 | mutex_unlock(&con_buf_mtx); | 547 | mutex_unlock(&con_buf_mtx); |
| 548 | 548 | ||
diff --git a/drivers/tty/vt/vt.c b/drivers/tty/vt/vt.c index 76407eca9ab..147ede3423d 100644 --- a/drivers/tty/vt/vt.c +++ b/drivers/tty/vt/vt.c | |||
| @@ -1003,9 +1003,9 @@ static int vt_resize(struct tty_struct *tty, struct winsize *ws) | |||
| 1003 | struct vc_data *vc = tty->driver_data; | 1003 | struct vc_data *vc = tty->driver_data; |
| 1004 | int ret; | 1004 | int ret; |
| 1005 | 1005 | ||
| 1006 | acquire_console_sem(); | 1006 | console_lock(); |
| 1007 | ret = vc_do_resize(tty, vc, ws->ws_col, ws->ws_row); | 1007 | ret = vc_do_resize(tty, vc, ws->ws_col, ws->ws_row); |
| 1008 | release_console_sem(); | 1008 | console_unlock(); |
| 1009 | return ret; | 1009 | return ret; |
| 1010 | } | 1010 | } |
| 1011 | 1011 | ||
| @@ -1271,7 +1271,7 @@ static void default_attr(struct vc_data *vc) | |||
| 1271 | vc->vc_color = vc->vc_def_color; | 1271 | vc->vc_color = vc->vc_def_color; |
| 1272 | } | 1272 | } |
| 1273 | 1273 | ||
| 1274 | /* console_sem is held */ | 1274 | /* console_lock is held */ |
| 1275 | static void csi_m(struct vc_data *vc) | 1275 | static void csi_m(struct vc_data *vc) |
| 1276 | { | 1276 | { |
| 1277 | int i; | 1277 | int i; |
| @@ -1415,7 +1415,7 @@ int mouse_reporting(void) | |||
| 1415 | return vc_cons[fg_console].d->vc_report_mouse; | 1415 | return vc_cons[fg_console].d->vc_report_mouse; |
| 1416 | } | 1416 | } |
| 1417 | 1417 | ||
| 1418 | /* console_sem is held */ | 1418 | /* console_lock is held */ |
| 1419 | static void set_mode(struct vc_data *vc, int on_off) | 1419 | static void set_mode(struct vc_data *vc, int on_off) |
| 1420 | { | 1420 | { |
| 1421 | int i; | 1421 | int i; |
| @@ -1485,7 +1485,7 @@ static void set_mode(struct vc_data *vc, int on_off) | |||
| 1485 | } | 1485 | } |
| 1486 | } | 1486 | } |
| 1487 | 1487 | ||
| 1488 | /* console_sem is held */ | 1488 | /* console_lock is held */ |
| 1489 | static void setterm_command(struct vc_data *vc) | 1489 | static void setterm_command(struct vc_data *vc) |
| 1490 | { | 1490 | { |
| 1491 | switch(vc->vc_par[0]) { | 1491 | switch(vc->vc_par[0]) { |
| @@ -1545,7 +1545,7 @@ static void setterm_command(struct vc_data *vc) | |||
| 1545 | } | 1545 | } |
| 1546 | } | 1546 | } |
| 1547 | 1547 | ||
| 1548 | /* console_sem is held */ | 1548 | /* console_lock is held */ |
| 1549 | static void csi_at(struct vc_data *vc, unsigned int nr) | 1549 | static void csi_at(struct vc_data *vc, unsigned int nr) |
| 1550 | { | 1550 | { |
| 1551 | if (nr > vc->vc_cols - vc->vc_x) | 1551 | if (nr > vc->vc_cols - vc->vc_x) |
| @@ -1555,7 +1555,7 @@ static void csi_at(struct vc_data *vc, unsigned int nr) | |||
| 1555 | insert_char(vc, nr); | 1555 | insert_char(vc, nr); |
| 1556 | } | 1556 | } |
| 1557 | 1557 | ||
| 1558 | /* console_sem is held */ | 1558 | /* console_lock is held */ |
| 1559 | static void csi_L(struct vc_data *vc, unsigned int nr) | 1559 | static void csi_L(struct vc_data *vc, unsigned int nr) |
| 1560 | { | 1560 | { |
| 1561 | if (nr > vc->vc_rows - vc->vc_y) | 1561 | if (nr > vc->vc_rows - vc->vc_y) |
| @@ -1566,7 +1566,7 @@ static void csi_L(struct vc_data *vc, unsigned int nr) | |||
| 1566 | vc->vc_need_wrap = 0; | 1566 | vc->vc_need_wrap = 0; |
| 1567 | } | 1567 | } |
| 1568 | 1568 | ||
| 1569 | /* console_sem is held */ | 1569 | /* console_lock is held */ |
| 1570 | static void csi_P(struct vc_data *vc, unsigned int nr) | 1570 | static void csi_P(struct vc_data *vc, unsigned int nr) |
| 1571 | { | 1571 | { |
| 1572 | if (nr > vc->vc_cols - vc->vc_x) | 1572 | if (nr > vc->vc_cols - vc->vc_x) |
| @@ -1576,7 +1576,7 @@ static void csi_P(struct vc_data *vc, unsigned int nr) | |||
| 1576 | delete_char(vc, nr); | 1576 | delete_char(vc, nr); |
| 1577 | } | 1577 | } |
| 1578 | 1578 | ||
| 1579 | /* console_sem is held */ | 1579 | /* console_lock is held */ |
| 1580 | static void csi_M(struct vc_data *vc, unsigned int nr) | 1580 | static void csi_M(struct vc_data *vc, unsigned int nr) |
| 1581 | { | 1581 | { |
| 1582 | if (nr > vc->vc_rows - vc->vc_y) | 1582 | if (nr > vc->vc_rows - vc->vc_y) |
| @@ -1587,7 +1587,7 @@ static void csi_M(struct vc_data *vc, unsigned int nr) | |||
| 1587 | vc->vc_need_wrap = 0; | 1587 | vc->vc_need_wrap = 0; |
| 1588 | } | 1588 | } |
| 1589 | 1589 | ||
| 1590 | /* console_sem is held (except via vc_init->reset_terminal */ | 1590 | /* console_lock is held (except via vc_init->reset_terminal */ |
| 1591 | static void save_cur(struct vc_data *vc) | 1591 | static void save_cur(struct vc_data *vc) |
| 1592 | { | 1592 | { |
| 1593 | vc->vc_saved_x = vc->vc_x; | 1593 | vc->vc_saved_x = vc->vc_x; |
| @@ -1603,7 +1603,7 @@ static void save_cur(struct vc_data *vc) | |||
| 1603 | vc->vc_saved_G1 = vc->vc_G1_charset; | 1603 | vc->vc_saved_G1 = vc->vc_G1_charset; |
| 1604 | } | 1604 | } |
| 1605 | 1605 | ||
| 1606 | /* console_sem is held */ | 1606 | /* console_lock is held */ |
| 1607 | static void restore_cur(struct vc_data *vc) | 1607 | static void restore_cur(struct vc_data *vc) |
| 1608 | { | 1608 | { |
| 1609 | gotoxy(vc, vc->vc_saved_x, vc->vc_saved_y); | 1609 | gotoxy(vc, vc->vc_saved_x, vc->vc_saved_y); |
| @@ -1625,7 +1625,7 @@ enum { ESnormal, ESesc, ESsquare, ESgetpars, ESgotpars, ESfunckey, | |||
| 1625 | EShash, ESsetG0, ESsetG1, ESpercent, ESignore, ESnonstd, | 1625 | EShash, ESsetG0, ESsetG1, ESpercent, ESignore, ESnonstd, |
| 1626 | ESpalette }; | 1626 | ESpalette }; |
| 1627 | 1627 | ||
| 1628 | /* console_sem is held (except via vc_init()) */ | 1628 | /* console_lock is held (except via vc_init()) */ |
| 1629 | static void reset_terminal(struct vc_data *vc, int do_clear) | 1629 | static void reset_terminal(struct vc_data *vc, int do_clear) |
| 1630 | { | 1630 | { |
| 1631 | vc->vc_top = 0; | 1631 | vc->vc_top = 0; |
| @@ -1685,7 +1685,7 @@ static void reset_terminal(struct vc_data *vc, int do_clear) | |||
| 1685 | csi_J(vc, 2); | 1685 | csi_J(vc, 2); |
| 1686 | } | 1686 | } |
| 1687 | 1687 | ||
| 1688 | /* console_sem is held */ | 1688 | /* console_lock is held */ |
| 1689 | static void do_con_trol(struct tty_struct *tty, struct vc_data *vc, int c) | 1689 | static void do_con_trol(struct tty_struct *tty, struct vc_data *vc, int c) |
| 1690 | { | 1690 | { |
| 1691 | /* | 1691 | /* |
| @@ -2119,7 +2119,7 @@ static int is_double_width(uint32_t ucs) | |||
| 2119 | return bisearch(ucs, double_width, ARRAY_SIZE(double_width) - 1); | 2119 | return bisearch(ucs, double_width, ARRAY_SIZE(double_width) - 1); |
| 2120 | } | 2120 | } |
| 2121 | 2121 | ||
| 2122 | /* acquires console_sem */ | 2122 | /* acquires console_lock */ |
| 2123 | static int do_con_write(struct tty_struct *tty, const unsigned char *buf, int count) | 2123 | static int do_con_write(struct tty_struct *tty, const unsigned char *buf, int count) |
| 2124 | { | 2124 | { |
| 2125 | #ifdef VT_BUF_VRAM_ONLY | 2125 | #ifdef VT_BUF_VRAM_ONLY |
| @@ -2147,11 +2147,11 @@ static int do_con_write(struct tty_struct *tty, const unsigned char *buf, int co | |||
| 2147 | 2147 | ||
| 2148 | might_sleep(); | 2148 | might_sleep(); |
| 2149 | 2149 | ||
| 2150 | acquire_console_sem(); | 2150 | console_lock(); |
| 2151 | vc = tty->driver_data; | 2151 | vc = tty->driver_data; |
| 2152 | if (vc == NULL) { | 2152 | if (vc == NULL) { |
| 2153 | printk(KERN_ERR "vt: argh, driver_data is NULL !\n"); | 2153 | printk(KERN_ERR "vt: argh, driver_data is NULL !\n"); |
| 2154 | release_console_sem(); | 2154 | console_unlock(); |
| 2155 | return 0; | 2155 | return 0; |
| 2156 | } | 2156 | } |
| 2157 | 2157 | ||
| @@ -2159,7 +2159,7 @@ static int do_con_write(struct tty_struct *tty, const unsigned char *buf, int co | |||
| 2159 | if (!vc_cons_allocated(currcons)) { | 2159 | if (!vc_cons_allocated(currcons)) { |
| 2160 | /* could this happen? */ | 2160 | /* could this happen? */ |
| 2161 | printk_once("con_write: tty %d not allocated\n", currcons+1); | 2161 | printk_once("con_write: tty %d not allocated\n", currcons+1); |
| 2162 | release_console_sem(); | 2162 | console_unlock(); |
| 2163 | return 0; | 2163 | return 0; |
| 2164 | } | 2164 | } |
| 2165 | 2165 | ||
| @@ -2375,7 +2375,7 @@ rescan_last_byte: | |||
| 2375 | } | 2375 | } |
| 2376 | FLUSH | 2376 | FLUSH |
| 2377 | console_conditional_schedule(); | 2377 | console_conditional_schedule(); |
| 2378 | release_console_sem(); | 2378 | console_unlock(); |
| 2379 | notify_update(vc); | 2379 | notify_update(vc); |
| 2380 | return n; | 2380 | return n; |
| 2381 | #undef FLUSH | 2381 | #undef FLUSH |
| @@ -2388,11 +2388,11 @@ rescan_last_byte: | |||
| 2388 | * us to do the switches asynchronously (needed when we want | 2388 | * us to do the switches asynchronously (needed when we want |
| 2389 | * to switch due to a keyboard interrupt). Synchronization | 2389 | * to switch due to a keyboard interrupt). Synchronization |
| 2390 | * with other console code and prevention of re-entrancy is | 2390 | * with other console code and prevention of re-entrancy is |
| 2391 | * ensured with console_sem. | 2391 | * ensured with console_lock. |
| 2392 | */ | 2392 | */ |
| 2393 | static void console_callback(struct work_struct *ignored) | 2393 | static void console_callback(struct work_struct *ignored) |
| 2394 | { | 2394 | { |
| 2395 | acquire_console_sem(); | 2395 | console_lock(); |
| 2396 | 2396 | ||
| 2397 | if (want_console >= 0) { | 2397 | if (want_console >= 0) { |
| 2398 | if (want_console != fg_console && | 2398 | if (want_console != fg_console && |
| @@ -2422,7 +2422,7 @@ static void console_callback(struct work_struct *ignored) | |||
| 2422 | } | 2422 | } |
| 2423 | notify_update(vc_cons[fg_console].d); | 2423 | notify_update(vc_cons[fg_console].d); |
| 2424 | 2424 | ||
| 2425 | release_console_sem(); | 2425 | console_unlock(); |
| 2426 | } | 2426 | } |
| 2427 | 2427 | ||
| 2428 | int set_console(int nr) | 2428 | int set_console(int nr) |
| @@ -2603,7 +2603,7 @@ static struct console vt_console_driver = { | |||
| 2603 | */ | 2603 | */ |
| 2604 | 2604 | ||
| 2605 | /* | 2605 | /* |
| 2606 | * Generally a bit racy with respect to console_sem(). | 2606 | * Generally a bit racy with respect to console_lock();. |
| 2607 | * | 2607 | * |
| 2608 | * There are some functions which don't need it. | 2608 | * There are some functions which don't need it. |
| 2609 | * | 2609 | * |
| @@ -2629,17 +2629,17 @@ int tioclinux(struct tty_struct *tty, unsigned long arg) | |||
| 2629 | switch (type) | 2629 | switch (type) |
| 2630 | { | 2630 | { |
| 2631 | case TIOCL_SETSEL: | 2631 | case TIOCL_SETSEL: |
| 2632 | acquire_console_sem(); | 2632 | console_lock(); |
| 2633 | ret = set_selection((struct tiocl_selection __user *)(p+1), tty); | 2633 | ret = set_selection((struct tiocl_selection __user *)(p+1), tty); |
| 2634 | release_console_sem(); | 2634 | console_unlock(); |
| 2635 | break; | 2635 | break; |
| 2636 | case TIOCL_PASTESEL: | 2636 | case TIOCL_PASTESEL: |
| 2637 | ret = paste_selection(tty); | 2637 | ret = paste_selection(tty); |
| 2638 | break; | 2638 | break; |
| 2639 | case TIOCL_UNBLANKSCREEN: | 2639 | case TIOCL_UNBLANKSCREEN: |
| 2640 | acquire_console_sem(); | 2640 | console_lock(); |
| 2641 | unblank_screen(); | 2641 | unblank_screen(); |
| 2642 | release_console_sem(); | 2642 | console_unlock(); |
| 2643 | break; | 2643 | break; |
| 2644 | case TIOCL_SELLOADLUT: | 2644 | case TIOCL_SELLOADLUT: |
| 2645 | ret = sel_loadlut(p); | 2645 | ret = sel_loadlut(p); |
| @@ -2688,10 +2688,10 @@ int tioclinux(struct tty_struct *tty, unsigned long arg) | |||
| 2688 | } | 2688 | } |
| 2689 | break; | 2689 | break; |
| 2690 | case TIOCL_BLANKSCREEN: /* until explicitly unblanked, not only poked */ | 2690 | case TIOCL_BLANKSCREEN: /* until explicitly unblanked, not only poked */ |
| 2691 | acquire_console_sem(); | 2691 | console_lock(); |
| 2692 | ignore_poke = 1; | 2692 | ignore_poke = 1; |
| 2693 | do_blank_screen(0); | 2693 | do_blank_screen(0); |
| 2694 | release_console_sem(); | 2694 | console_unlock(); |
| 2695 | break; | 2695 | break; |
| 2696 | case TIOCL_BLANKEDSCREEN: | 2696 | case TIOCL_BLANKEDSCREEN: |
| 2697 | ret = console_blanked; | 2697 | ret = console_blanked; |
| @@ -2790,11 +2790,11 @@ static void con_flush_chars(struct tty_struct *tty) | |||
| 2790 | return; | 2790 | return; |
| 2791 | 2791 | ||
| 2792 | /* if we race with con_close(), vt may be null */ | 2792 | /* if we race with con_close(), vt may be null */ |
| 2793 | acquire_console_sem(); | 2793 | console_lock(); |
| 2794 | vc = tty->driver_data; | 2794 | vc = tty->driver_data; |
| 2795 | if (vc) | 2795 | if (vc) |
| 2796 | set_cursor(vc); | 2796 | set_cursor(vc); |
| 2797 | release_console_sem(); | 2797 | console_unlock(); |
| 2798 | } | 2798 | } |
| 2799 | 2799 | ||
| 2800 | /* | 2800 | /* |
| @@ -2805,7 +2805,7 @@ static int con_open(struct tty_struct *tty, struct file *filp) | |||
| 2805 | unsigned int currcons = tty->index; | 2805 | unsigned int currcons = tty->index; |
| 2806 | int ret = 0; | 2806 | int ret = 0; |
| 2807 | 2807 | ||
| 2808 | acquire_console_sem(); | 2808 | console_lock(); |
| 2809 | if (tty->driver_data == NULL) { | 2809 | if (tty->driver_data == NULL) { |
| 2810 | ret = vc_allocate(currcons); | 2810 | ret = vc_allocate(currcons); |
| 2811 | if (ret == 0) { | 2811 | if (ret == 0) { |
| @@ -2813,7 +2813,7 @@ static int con_open(struct tty_struct *tty, struct file *filp) | |||
| 2813 | 2813 | ||
| 2814 | /* Still being freed */ | 2814 | /* Still being freed */ |
| 2815 | if (vc->port.tty) { | 2815 | if (vc->port.tty) { |
| 2816 | release_console_sem(); | 2816 | console_unlock(); |
| 2817 | return -ERESTARTSYS; | 2817 | return -ERESTARTSYS; |
| 2818 | } | 2818 | } |
| 2819 | tty->driver_data = vc; | 2819 | tty->driver_data = vc; |
| @@ -2827,11 +2827,11 @@ static int con_open(struct tty_struct *tty, struct file *filp) | |||
| 2827 | tty->termios->c_iflag |= IUTF8; | 2827 | tty->termios->c_iflag |= IUTF8; |
| 2828 | else | 2828 | else |
| 2829 | tty->termios->c_iflag &= ~IUTF8; | 2829 | tty->termios->c_iflag &= ~IUTF8; |
| 2830 | release_console_sem(); | 2830 | console_unlock(); |
| 2831 | return ret; | 2831 | return ret; |
| 2832 | } | 2832 | } |
| 2833 | } | 2833 | } |
| 2834 | release_console_sem(); | 2834 | console_unlock(); |
| 2835 | return ret; | 2835 | return ret; |
| 2836 | } | 2836 | } |
| 2837 | 2837 | ||
| @@ -2844,9 +2844,9 @@ static void con_shutdown(struct tty_struct *tty) | |||
| 2844 | { | 2844 | { |
| 2845 | struct vc_data *vc = tty->driver_data; | 2845 | struct vc_data *vc = tty->driver_data; |
| 2846 | BUG_ON(vc == NULL); | 2846 | BUG_ON(vc == NULL); |
| 2847 | acquire_console_sem(); | 2847 | console_lock(); |
| 2848 | vc->port.tty = NULL; | 2848 | vc->port.tty = NULL; |
| 2849 | release_console_sem(); | 2849 | console_unlock(); |
| 2850 | tty_shutdown(tty); | 2850 | tty_shutdown(tty); |
| 2851 | } | 2851 | } |
| 2852 | 2852 | ||
| @@ -2893,13 +2893,13 @@ static int __init con_init(void) | |||
| 2893 | struct vc_data *vc; | 2893 | struct vc_data *vc; |
| 2894 | unsigned int currcons = 0, i; | 2894 | unsigned int currcons = 0, i; |
| 2895 | 2895 | ||
| 2896 | acquire_console_sem(); | 2896 | console_lock(); |
| 2897 | 2897 | ||
| 2898 | if (conswitchp) | 2898 | if (conswitchp) |
| 2899 | display_desc = conswitchp->con_startup(); | 2899 | display_desc = conswitchp->con_startup(); |
| 2900 | if (!display_desc) { | 2900 | if (!display_desc) { |
| 2901 | fg_console = 0; | 2901 | fg_console = 0; |
| 2902 | release_console_sem(); | 2902 | console_unlock(); |
| 2903 | return 0; | 2903 | return 0; |
| 2904 | } | 2904 | } |
| 2905 | 2905 | ||
| @@ -2946,7 +2946,7 @@ static int __init con_init(void) | |||
| 2946 | printable = 1; | 2946 | printable = 1; |
| 2947 | printk("\n"); | 2947 | printk("\n"); |
| 2948 | 2948 | ||
| 2949 | release_console_sem(); | 2949 | console_unlock(); |
| 2950 | 2950 | ||
| 2951 | #ifdef CONFIG_VT_CONSOLE | 2951 | #ifdef CONFIG_VT_CONSOLE |
| 2952 | register_console(&vt_console_driver); | 2952 | register_console(&vt_console_driver); |
| @@ -2994,7 +2994,7 @@ int __init vty_init(const struct file_operations *console_fops) | |||
| 2994 | if (IS_ERR(tty0dev)) | 2994 | if (IS_ERR(tty0dev)) |
| 2995 | tty0dev = NULL; | 2995 | tty0dev = NULL; |
| 2996 | else | 2996 | else |
| 2997 | device_create_file(tty0dev, &dev_attr_active); | 2997 | WARN_ON(device_create_file(tty0dev, &dev_attr_active) < 0); |
| 2998 | 2998 | ||
| 2999 | vcs_init(); | 2999 | vcs_init(); |
| 3000 | 3000 | ||
| @@ -3037,7 +3037,7 @@ static int bind_con_driver(const struct consw *csw, int first, int last, | |||
| 3037 | if (!try_module_get(owner)) | 3037 | if (!try_module_get(owner)) |
| 3038 | return -ENODEV; | 3038 | return -ENODEV; |
| 3039 | 3039 | ||
| 3040 | acquire_console_sem(); | 3040 | console_lock(); |
| 3041 | 3041 | ||
| 3042 | /* check if driver is registered */ | 3042 | /* check if driver is registered */ |
| 3043 | for (i = 0; i < MAX_NR_CON_DRIVER; i++) { | 3043 | for (i = 0; i < MAX_NR_CON_DRIVER; i++) { |
| @@ -3122,7 +3122,7 @@ static int bind_con_driver(const struct consw *csw, int first, int last, | |||
| 3122 | 3122 | ||
| 3123 | retval = 0; | 3123 | retval = 0; |
| 3124 | err: | 3124 | err: |
| 3125 | release_console_sem(); | 3125 | console_unlock(); |
| 3126 | module_put(owner); | 3126 | module_put(owner); |
| 3127 | return retval; | 3127 | return retval; |
| 3128 | }; | 3128 | }; |
| @@ -3171,7 +3171,7 @@ int unbind_con_driver(const struct consw *csw, int first, int last, int deflt) | |||
| 3171 | if (!try_module_get(owner)) | 3171 | if (!try_module_get(owner)) |
| 3172 | return -ENODEV; | 3172 | return -ENODEV; |
| 3173 | 3173 | ||
| 3174 | acquire_console_sem(); | 3174 | console_lock(); |
| 3175 | 3175 | ||
| 3176 | /* check if driver is registered and if it is unbindable */ | 3176 | /* check if driver is registered and if it is unbindable */ |
| 3177 | for (i = 0; i < MAX_NR_CON_DRIVER; i++) { | 3177 | for (i = 0; i < MAX_NR_CON_DRIVER; i++) { |
| @@ -3185,7 +3185,7 @@ int unbind_con_driver(const struct consw *csw, int first, int last, int deflt) | |||
| 3185 | } | 3185 | } |
| 3186 | 3186 | ||
| 3187 | if (retval) { | 3187 | if (retval) { |
| 3188 | release_console_sem(); | 3188 | console_unlock(); |
| 3189 | goto err; | 3189 | goto err; |
| 3190 | } | 3190 | } |
| 3191 | 3191 | ||
| @@ -3204,12 +3204,12 @@ int unbind_con_driver(const struct consw *csw, int first, int last, int deflt) | |||
| 3204 | } | 3204 | } |
| 3205 | 3205 | ||
| 3206 | if (retval) { | 3206 | if (retval) { |
| 3207 | release_console_sem(); | 3207 | console_unlock(); |
| 3208 | goto err; | 3208 | goto err; |
| 3209 | } | 3209 | } |
| 3210 | 3210 | ||
| 3211 | if (!con_is_bound(csw)) { | 3211 | if (!con_is_bound(csw)) { |
| 3212 | release_console_sem(); | 3212 | console_unlock(); |
| 3213 | goto err; | 3213 | goto err; |
| 3214 | } | 3214 | } |
| 3215 | 3215 | ||
| @@ -3238,7 +3238,7 @@ int unbind_con_driver(const struct consw *csw, int first, int last, int deflt) | |||
| 3238 | if (!con_is_bound(csw)) | 3238 | if (!con_is_bound(csw)) |
| 3239 | con_driver->flag &= ~CON_DRIVER_FLAG_INIT; | 3239 | con_driver->flag &= ~CON_DRIVER_FLAG_INIT; |
| 3240 | 3240 | ||
| 3241 | release_console_sem(); | 3241 | console_unlock(); |
| 3242 | /* ignore return value, binding should not fail */ | 3242 | /* ignore return value, binding should not fail */ |
| 3243 | bind_con_driver(defcsw, first, last, deflt); | 3243 | bind_con_driver(defcsw, first, last, deflt); |
| 3244 | err: | 3244 | err: |
| @@ -3538,14 +3538,14 @@ int register_con_driver(const struct consw *csw, int first, int last) | |||
| 3538 | if (!try_module_get(owner)) | 3538 | if (!try_module_get(owner)) |
| 3539 | return -ENODEV; | 3539 | return -ENODEV; |
| 3540 | 3540 | ||
| 3541 | acquire_console_sem(); | 3541 | console_lock(); |
| 3542 | 3542 | ||
| 3543 | for (i = 0; i < MAX_NR_CON_DRIVER; i++) { | 3543 | for (i = 0; i < MAX_NR_CON_DRIVER; i++) { |
| 3544 | con_driver = ®istered_con_driver[i]; | 3544 | con_driver = ®istered_con_driver[i]; |
| 3545 | 3545 | ||
| 3546 | /* already registered */ | 3546 | /* already registered */ |
| 3547 | if (con_driver->con == csw) | 3547 | if (con_driver->con == csw) |
| 3548 | retval = -EINVAL; | 3548 | retval = -EBUSY; |
| 3549 | } | 3549 | } |
| 3550 | 3550 | ||
| 3551 | if (retval) | 3551 | if (retval) |
| @@ -3592,7 +3592,7 @@ int register_con_driver(const struct consw *csw, int first, int last) | |||
| 3592 | } | 3592 | } |
| 3593 | 3593 | ||
| 3594 | err: | 3594 | err: |
| 3595 | release_console_sem(); | 3595 | console_unlock(); |
| 3596 | module_put(owner); | 3596 | module_put(owner); |
| 3597 | return retval; | 3597 | return retval; |
| 3598 | } | 3598 | } |
| @@ -3613,7 +3613,7 @@ int unregister_con_driver(const struct consw *csw) | |||
| 3613 | { | 3613 | { |
| 3614 | int i, retval = -ENODEV; | 3614 | int i, retval = -ENODEV; |
| 3615 | 3615 | ||
| 3616 | acquire_console_sem(); | 3616 | console_lock(); |
| 3617 | 3617 | ||
| 3618 | /* cannot unregister a bound driver */ | 3618 | /* cannot unregister a bound driver */ |
| 3619 | if (con_is_bound(csw)) | 3619 | if (con_is_bound(csw)) |
| @@ -3639,7 +3639,7 @@ int unregister_con_driver(const struct consw *csw) | |||
| 3639 | } | 3639 | } |
| 3640 | } | 3640 | } |
| 3641 | err: | 3641 | err: |
| 3642 | release_console_sem(); | 3642 | console_unlock(); |
| 3643 | return retval; | 3643 | return retval; |
| 3644 | } | 3644 | } |
| 3645 | EXPORT_SYMBOL(unregister_con_driver); | 3645 | EXPORT_SYMBOL(unregister_con_driver); |
| @@ -3656,7 +3656,12 @@ int take_over_console(const struct consw *csw, int first, int last, int deflt) | |||
| 3656 | int err; | 3656 | int err; |
| 3657 | 3657 | ||
| 3658 | err = register_con_driver(csw, first, last); | 3658 | err = register_con_driver(csw, first, last); |
| 3659 | 3659 | /* if we get an busy error we still want to bind the console driver | |
| 3660 | * and return success, as we may have unbound the console driver | ||
| 3661 | Â * but not unregistered it. | ||
| 3662 | */ | ||
| 3663 | if (err == -EBUSY) | ||
| 3664 | err = 0; | ||
| 3660 | if (!err) | 3665 | if (!err) |
| 3661 | bind_con_driver(csw, first, last, deflt); | 3666 | bind_con_driver(csw, first, last, deflt); |
| 3662 | 3667 | ||
| @@ -3934,9 +3939,9 @@ int con_set_cmap(unsigned char __user *arg) | |||
| 3934 | { | 3939 | { |
| 3935 | int rc; | 3940 | int rc; |
| 3936 | 3941 | ||
| 3937 | acquire_console_sem(); | 3942 | console_lock(); |
| 3938 | rc = set_get_cmap (arg,1); | 3943 | rc = set_get_cmap (arg,1); |
| 3939 | release_console_sem(); | 3944 | console_unlock(); |
| 3940 | 3945 | ||
| 3941 | return rc; | 3946 | return rc; |
| 3942 | } | 3947 | } |
| @@ -3945,9 +3950,9 @@ int con_get_cmap(unsigned char __user *arg) | |||
| 3945 | { | 3950 | { |
| 3946 | int rc; | 3951 | int rc; |
| 3947 | 3952 | ||
| 3948 | acquire_console_sem(); | 3953 | console_lock(); |
| 3949 | rc = set_get_cmap (arg,0); | 3954 | rc = set_get_cmap (arg,0); |
| 3950 | release_console_sem(); | 3955 | console_unlock(); |
| 3951 | 3956 | ||
| 3952 | return rc; | 3957 | return rc; |
| 3953 | } | 3958 | } |
| @@ -3994,12 +3999,12 @@ static int con_font_get(struct vc_data *vc, struct console_font_op *op) | |||
| 3994 | } else | 3999 | } else |
| 3995 | font.data = NULL; | 4000 | font.data = NULL; |
| 3996 | 4001 | ||
| 3997 | acquire_console_sem(); | 4002 | console_lock(); |
| 3998 | if (vc->vc_sw->con_font_get) | 4003 | if (vc->vc_sw->con_font_get) |
| 3999 | rc = vc->vc_sw->con_font_get(vc, &font); | 4004 | rc = vc->vc_sw->con_font_get(vc, &font); |
| 4000 | else | 4005 | else |
| 4001 | rc = -ENOSYS; | 4006 | rc = -ENOSYS; |
| 4002 | release_console_sem(); | 4007 | console_unlock(); |
| 4003 | 4008 | ||
| 4004 | if (rc) | 4009 | if (rc) |
| 4005 | goto out; | 4010 | goto out; |
| @@ -4076,12 +4081,12 @@ static int con_font_set(struct vc_data *vc, struct console_font_op *op) | |||
| 4076 | font.data = memdup_user(op->data, size); | 4081 | font.data = memdup_user(op->data, size); |
| 4077 | if (IS_ERR(font.data)) | 4082 | if (IS_ERR(font.data)) |
| 4078 | return PTR_ERR(font.data); | 4083 | return PTR_ERR(font.data); |
| 4079 | acquire_console_sem(); | 4084 | console_lock(); |
| 4080 | if (vc->vc_sw->con_font_set) | 4085 | if (vc->vc_sw->con_font_set) |
| 4081 | rc = vc->vc_sw->con_font_set(vc, &font, op->flags); | 4086 | rc = vc->vc_sw->con_font_set(vc, &font, op->flags); |
| 4082 | else | 4087 | else |
| 4083 | rc = -ENOSYS; | 4088 | rc = -ENOSYS; |
| 4084 | release_console_sem(); | 4089 | console_unlock(); |
| 4085 | kfree(font.data); | 4090 | kfree(font.data); |
| 4086 | return rc; | 4091 | return rc; |
| 4087 | } | 4092 | } |
| @@ -4103,12 +4108,12 @@ static int con_font_default(struct vc_data *vc, struct console_font_op *op) | |||
| 4103 | else | 4108 | else |
| 4104 | name[MAX_FONT_NAME - 1] = 0; | 4109 | name[MAX_FONT_NAME - 1] = 0; |
| 4105 | 4110 | ||
| 4106 | acquire_console_sem(); | 4111 | console_lock(); |
| 4107 | if (vc->vc_sw->con_font_default) | 4112 | if (vc->vc_sw->con_font_default) |
| 4108 | rc = vc->vc_sw->con_font_default(vc, &font, s); | 4113 | rc = vc->vc_sw->con_font_default(vc, &font, s); |
| 4109 | else | 4114 | else |
| 4110 | rc = -ENOSYS; | 4115 | rc = -ENOSYS; |
| 4111 | release_console_sem(); | 4116 | console_unlock(); |
| 4112 | if (!rc) { | 4117 | if (!rc) { |
| 4113 | op->width = font.width; | 4118 | op->width = font.width; |
| 4114 | op->height = font.height; | 4119 | op->height = font.height; |
| @@ -4124,7 +4129,7 @@ static int con_font_copy(struct vc_data *vc, struct console_font_op *op) | |||
| 4124 | if (vc->vc_mode != KD_TEXT) | 4129 | if (vc->vc_mode != KD_TEXT) |
| 4125 | return -EINVAL; | 4130 | return -EINVAL; |
| 4126 | 4131 | ||
| 4127 | acquire_console_sem(); | 4132 | console_lock(); |
| 4128 | if (!vc->vc_sw->con_font_copy) | 4133 | if (!vc->vc_sw->con_font_copy) |
| 4129 | rc = -ENOSYS; | 4134 | rc = -ENOSYS; |
| 4130 | else if (con < 0 || !vc_cons_allocated(con)) | 4135 | else if (con < 0 || !vc_cons_allocated(con)) |
| @@ -4133,7 +4138,7 @@ static int con_font_copy(struct vc_data *vc, struct console_font_op *op) | |||
| 4133 | rc = 0; | 4138 | rc = 0; |
| 4134 | else | 4139 | else |
| 4135 | rc = vc->vc_sw->con_font_copy(vc, con); | 4140 | rc = vc->vc_sw->con_font_copy(vc, con); |
| 4136 | release_console_sem(); | 4141 | console_unlock(); |
| 4137 | return rc; | 4142 | return rc; |
| 4138 | } | 4143 | } |
| 4139 | 4144 | ||
diff --git a/drivers/tty/vt/vt_ioctl.c b/drivers/tty/vt/vt_ioctl.c index 6b68a0fb461..1235ebda6e1 100644 --- a/drivers/tty/vt/vt_ioctl.c +++ b/drivers/tty/vt/vt_ioctl.c | |||
| @@ -649,12 +649,12 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 649 | /* | 649 | /* |
| 650 | * explicitly blank/unblank the screen if switching modes | 650 | * explicitly blank/unblank the screen if switching modes |
| 651 | */ | 651 | */ |
| 652 | acquire_console_sem(); | 652 | console_lock(); |
| 653 | if (arg == KD_TEXT) | 653 | if (arg == KD_TEXT) |
| 654 | do_unblank_screen(1); | 654 | do_unblank_screen(1); |
| 655 | else | 655 | else |
| 656 | do_blank_screen(1); | 656 | do_blank_screen(1); |
| 657 | release_console_sem(); | 657 | console_unlock(); |
| 658 | break; | 658 | break; |
| 659 | 659 | ||
| 660 | case KDGETMODE: | 660 | case KDGETMODE: |
| @@ -893,7 +893,7 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 893 | ret = -EINVAL; | 893 | ret = -EINVAL; |
| 894 | goto out; | 894 | goto out; |
| 895 | } | 895 | } |
| 896 | acquire_console_sem(); | 896 | console_lock(); |
| 897 | vc->vt_mode = tmp; | 897 | vc->vt_mode = tmp; |
| 898 | /* the frsig is ignored, so we set it to 0 */ | 898 | /* the frsig is ignored, so we set it to 0 */ |
| 899 | vc->vt_mode.frsig = 0; | 899 | vc->vt_mode.frsig = 0; |
| @@ -901,7 +901,7 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 901 | vc->vt_pid = get_pid(task_pid(current)); | 901 | vc->vt_pid = get_pid(task_pid(current)); |
| 902 | /* no switch is required -- saw@shade.msu.ru */ | 902 | /* no switch is required -- saw@shade.msu.ru */ |
| 903 | vc->vt_newvt = -1; | 903 | vc->vt_newvt = -1; |
| 904 | release_console_sem(); | 904 | console_unlock(); |
| 905 | break; | 905 | break; |
| 906 | } | 906 | } |
| 907 | 907 | ||
| @@ -910,9 +910,9 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 910 | struct vt_mode tmp; | 910 | struct vt_mode tmp; |
| 911 | int rc; | 911 | int rc; |
| 912 | 912 | ||
| 913 | acquire_console_sem(); | 913 | console_lock(); |
| 914 | memcpy(&tmp, &vc->vt_mode, sizeof(struct vt_mode)); | 914 | memcpy(&tmp, &vc->vt_mode, sizeof(struct vt_mode)); |
| 915 | release_console_sem(); | 915 | console_unlock(); |
| 916 | 916 | ||
| 917 | rc = copy_to_user(up, &tmp, sizeof(struct vt_mode)); | 917 | rc = copy_to_user(up, &tmp, sizeof(struct vt_mode)); |
| 918 | if (rc) | 918 | if (rc) |
| @@ -965,9 +965,9 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 965 | ret = -ENXIO; | 965 | ret = -ENXIO; |
| 966 | else { | 966 | else { |
| 967 | arg--; | 967 | arg--; |
| 968 | acquire_console_sem(); | 968 | console_lock(); |
| 969 | ret = vc_allocate(arg); | 969 | ret = vc_allocate(arg); |
| 970 | release_console_sem(); | 970 | console_unlock(); |
| 971 | if (ret) | 971 | if (ret) |
| 972 | break; | 972 | break; |
| 973 | set_console(arg); | 973 | set_console(arg); |
| @@ -990,7 +990,7 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 990 | ret = -ENXIO; | 990 | ret = -ENXIO; |
| 991 | else { | 991 | else { |
| 992 | vsa.console--; | 992 | vsa.console--; |
| 993 | acquire_console_sem(); | 993 | console_lock(); |
| 994 | ret = vc_allocate(vsa.console); | 994 | ret = vc_allocate(vsa.console); |
| 995 | if (ret == 0) { | 995 | if (ret == 0) { |
| 996 | struct vc_data *nvc; | 996 | struct vc_data *nvc; |
| @@ -1003,7 +1003,7 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 1003 | put_pid(nvc->vt_pid); | 1003 | put_pid(nvc->vt_pid); |
| 1004 | nvc->vt_pid = get_pid(task_pid(current)); | 1004 | nvc->vt_pid = get_pid(task_pid(current)); |
| 1005 | } | 1005 | } |
| 1006 | release_console_sem(); | 1006 | console_unlock(); |
| 1007 | if (ret) | 1007 | if (ret) |
| 1008 | break; | 1008 | break; |
| 1009 | /* Commence switch and lock */ | 1009 | /* Commence switch and lock */ |
| @@ -1044,7 +1044,7 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 1044 | /* | 1044 | /* |
| 1045 | * Switching-from response | 1045 | * Switching-from response |
| 1046 | */ | 1046 | */ |
| 1047 | acquire_console_sem(); | 1047 | console_lock(); |
| 1048 | if (vc->vt_newvt >= 0) { | 1048 | if (vc->vt_newvt >= 0) { |
| 1049 | if (arg == 0) | 1049 | if (arg == 0) |
| 1050 | /* | 1050 | /* |
| @@ -1063,7 +1063,7 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 1063 | vc->vt_newvt = -1; | 1063 | vc->vt_newvt = -1; |
| 1064 | ret = vc_allocate(newvt); | 1064 | ret = vc_allocate(newvt); |
| 1065 | if (ret) { | 1065 | if (ret) { |
| 1066 | release_console_sem(); | 1066 | console_unlock(); |
| 1067 | break; | 1067 | break; |
| 1068 | } | 1068 | } |
| 1069 | /* | 1069 | /* |
| @@ -1083,7 +1083,7 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 1083 | if (arg != VT_ACKACQ) | 1083 | if (arg != VT_ACKACQ) |
| 1084 | ret = -EINVAL; | 1084 | ret = -EINVAL; |
| 1085 | } | 1085 | } |
| 1086 | release_console_sem(); | 1086 | console_unlock(); |
| 1087 | break; | 1087 | break; |
| 1088 | 1088 | ||
| 1089 | /* | 1089 | /* |
| @@ -1096,20 +1096,20 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 1096 | } | 1096 | } |
| 1097 | if (arg == 0) { | 1097 | if (arg == 0) { |
| 1098 | /* deallocate all unused consoles, but leave 0 */ | 1098 | /* deallocate all unused consoles, but leave 0 */ |
| 1099 | acquire_console_sem(); | 1099 | console_lock(); |
| 1100 | for (i=1; i<MAX_NR_CONSOLES; i++) | 1100 | for (i=1; i<MAX_NR_CONSOLES; i++) |
| 1101 | if (! VT_BUSY(i)) | 1101 | if (! VT_BUSY(i)) |
| 1102 | vc_deallocate(i); | 1102 | vc_deallocate(i); |
| 1103 | release_console_sem(); | 1103 | console_unlock(); |
| 1104 | } else { | 1104 | } else { |
| 1105 | /* deallocate a single console, if possible */ | 1105 | /* deallocate a single console, if possible */ |
| 1106 | arg--; | 1106 | arg--; |
| 1107 | if (VT_BUSY(arg)) | 1107 | if (VT_BUSY(arg)) |
| 1108 | ret = -EBUSY; | 1108 | ret = -EBUSY; |
| 1109 | else if (arg) { /* leave 0 */ | 1109 | else if (arg) { /* leave 0 */ |
| 1110 | acquire_console_sem(); | 1110 | console_lock(); |
| 1111 | vc_deallocate(arg); | 1111 | vc_deallocate(arg); |
| 1112 | release_console_sem(); | 1112 | console_unlock(); |
| 1113 | } | 1113 | } |
| 1114 | } | 1114 | } |
| 1115 | break; | 1115 | break; |
| @@ -1126,7 +1126,7 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 1126 | get_user(cc, &vtsizes->v_cols)) | 1126 | get_user(cc, &vtsizes->v_cols)) |
| 1127 | ret = -EFAULT; | 1127 | ret = -EFAULT; |
| 1128 | else { | 1128 | else { |
| 1129 | acquire_console_sem(); | 1129 | console_lock(); |
| 1130 | for (i = 0; i < MAX_NR_CONSOLES; i++) { | 1130 | for (i = 0; i < MAX_NR_CONSOLES; i++) { |
| 1131 | vc = vc_cons[i].d; | 1131 | vc = vc_cons[i].d; |
| 1132 | 1132 | ||
| @@ -1135,7 +1135,7 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 1135 | vc_resize(vc_cons[i].d, cc, ll); | 1135 | vc_resize(vc_cons[i].d, cc, ll); |
| 1136 | } | 1136 | } |
| 1137 | } | 1137 | } |
| 1138 | release_console_sem(); | 1138 | console_unlock(); |
| 1139 | } | 1139 | } |
| 1140 | break; | 1140 | break; |
| 1141 | } | 1141 | } |
| @@ -1187,14 +1187,14 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, | |||
| 1187 | for (i = 0; i < MAX_NR_CONSOLES; i++) { | 1187 | for (i = 0; i < MAX_NR_CONSOLES; i++) { |
| 1188 | if (!vc_cons[i].d) | 1188 | if (!vc_cons[i].d) |
| 1189 | continue; | 1189 | continue; |
| 1190 | acquire_console_sem(); | 1190 | console_lock(); |
| 1191 | if (vlin) | 1191 | if (vlin) |
| 1192 | vc_cons[i].d->vc_scan_lines = vlin; | 1192 | vc_cons[i].d->vc_scan_lines = vlin; |
| 1193 | if (clin) | 1193 | if (clin) |
| 1194 | vc_cons[i].d->vc_font.height = clin; | 1194 | vc_cons[i].d->vc_font.height = clin; |
| 1195 | vc_cons[i].d->vc_resize_user = 1; | 1195 | vc_cons[i].d->vc_resize_user = 1; |
| 1196 | vc_resize(vc_cons[i].d, cc, ll); | 1196 | vc_resize(vc_cons[i].d, cc, ll); |
| 1197 | release_console_sem(); | 1197 | console_unlock(); |
| 1198 | } | 1198 | } |
| 1199 | break; | 1199 | break; |
| 1200 | } | 1200 | } |
| @@ -1367,7 +1367,7 @@ void vc_SAK(struct work_struct *work) | |||
| 1367 | struct vc_data *vc; | 1367 | struct vc_data *vc; |
| 1368 | struct tty_struct *tty; | 1368 | struct tty_struct *tty; |
| 1369 | 1369 | ||
| 1370 | acquire_console_sem(); | 1370 | console_lock(); |
| 1371 | vc = vc_con->d; | 1371 | vc = vc_con->d; |
| 1372 | if (vc) { | 1372 | if (vc) { |
| 1373 | tty = vc->port.tty; | 1373 | tty = vc->port.tty; |
| @@ -1379,7 +1379,7 @@ void vc_SAK(struct work_struct *work) | |||
| 1379 | __do_SAK(tty); | 1379 | __do_SAK(tty); |
| 1380 | reset_vc(vc); | 1380 | reset_vc(vc); |
| 1381 | } | 1381 | } |
| 1382 | release_console_sem(); | 1382 | console_unlock(); |
| 1383 | } | 1383 | } |
| 1384 | 1384 | ||
| 1385 | #ifdef CONFIG_COMPAT | 1385 | #ifdef CONFIG_COMPAT |
| @@ -1737,10 +1737,10 @@ int vt_move_to_console(unsigned int vt, int alloc) | |||
| 1737 | { | 1737 | { |
| 1738 | int prev; | 1738 | int prev; |
| 1739 | 1739 | ||
| 1740 | acquire_console_sem(); | 1740 | console_lock(); |
| 1741 | /* Graphics mode - up to X */ | 1741 | /* Graphics mode - up to X */ |
| 1742 | if (disable_vt_switch) { | 1742 | if (disable_vt_switch) { |
| 1743 | release_console_sem(); | 1743 | console_unlock(); |
| 1744 | return 0; | 1744 | return 0; |
| 1745 | } | 1745 | } |
| 1746 | prev = fg_console; | 1746 | prev = fg_console; |
| @@ -1748,7 +1748,7 @@ int vt_move_to_console(unsigned int vt, int alloc) | |||
| 1748 | if (alloc && vc_allocate(vt)) { | 1748 | if (alloc && vc_allocate(vt)) { |
| 1749 | /* we can't have a free VC for now. Too bad, | 1749 | /* we can't have a free VC for now. Too bad, |
| 1750 | * we don't want to mess the screen for now. */ | 1750 | * we don't want to mess the screen for now. */ |
| 1751 | release_console_sem(); | 1751 | console_unlock(); |
| 1752 | return -ENOSPC; | 1752 | return -ENOSPC; |
| 1753 | } | 1753 | } |
| 1754 | 1754 | ||
| @@ -1758,10 +1758,10 @@ int vt_move_to_console(unsigned int vt, int alloc) | |||
| 1758 | * Let the calling function know so it can decide | 1758 | * Let the calling function know so it can decide |
| 1759 | * what to do. | 1759 | * what to do. |
| 1760 | */ | 1760 | */ |
| 1761 | release_console_sem(); | 1761 | console_unlock(); |
| 1762 | return -EIO; | 1762 | return -EIO; |
| 1763 | } | 1763 | } |
| 1764 | release_console_sem(); | 1764 | console_unlock(); |
| 1765 | tty_lock(); | 1765 | tty_lock(); |
| 1766 | if (vt_waitactive(vt + 1)) { | 1766 | if (vt_waitactive(vt + 1)) { |
| 1767 | pr_debug("Suspend: Can't switch VCs."); | 1767 | pr_debug("Suspend: Can't switch VCs."); |
| @@ -1781,8 +1781,8 @@ int vt_move_to_console(unsigned int vt, int alloc) | |||
| 1781 | */ | 1781 | */ |
| 1782 | void pm_set_vt_switch(int do_switch) | 1782 | void pm_set_vt_switch(int do_switch) |
| 1783 | { | 1783 | { |
| 1784 | acquire_console_sem(); | 1784 | console_lock(); |
| 1785 | disable_vt_switch = !do_switch; | 1785 | disable_vt_switch = !do_switch; |
| 1786 | release_console_sem(); | 1786 | console_unlock(); |
| 1787 | } | 1787 | } |
| 1788 | EXPORT_SYMBOL(pm_set_vt_switch); | 1788 | EXPORT_SYMBOL(pm_set_vt_switch); |
diff --git a/drivers/usb/class/cdc-wdm.c b/drivers/usb/class/cdc-wdm.c index 6ee4451bfe2..47085e5879a 100644 --- a/drivers/usb/class/cdc-wdm.c +++ b/drivers/usb/class/cdc-wdm.c | |||
| @@ -342,7 +342,7 @@ static ssize_t wdm_write | |||
| 342 | goto outnp; | 342 | goto outnp; |
| 343 | } | 343 | } |
| 344 | 344 | ||
| 345 | if (!file->f_flags && O_NONBLOCK) | 345 | if (!(file->f_flags & O_NONBLOCK)) |
| 346 | r = wait_event_interruptible(desc->wait, !test_bit(WDM_IN_USE, | 346 | r = wait_event_interruptible(desc->wait, !test_bit(WDM_IN_USE, |
| 347 | &desc->flags)); | 347 | &desc->flags)); |
| 348 | else | 348 | else |
diff --git a/drivers/usb/core/endpoint.c b/drivers/usb/core/endpoint.c index 9da25056302..df502a98d0d 100644 --- a/drivers/usb/core/endpoint.c +++ b/drivers/usb/core/endpoint.c | |||
| @@ -192,12 +192,12 @@ int usb_create_ep_devs(struct device *parent, | |||
| 192 | ep_dev->dev.parent = parent; | 192 | ep_dev->dev.parent = parent; |
| 193 | ep_dev->dev.release = ep_device_release; | 193 | ep_dev->dev.release = ep_device_release; |
| 194 | dev_set_name(&ep_dev->dev, "ep_%02x", endpoint->desc.bEndpointAddress); | 194 | dev_set_name(&ep_dev->dev, "ep_%02x", endpoint->desc.bEndpointAddress); |
| 195 | device_enable_async_suspend(&ep_dev->dev); | ||
| 196 | 195 | ||
| 197 | retval = device_register(&ep_dev->dev); | 196 | retval = device_register(&ep_dev->dev); |
| 198 | if (retval) | 197 | if (retval) |
| 199 | goto error_register; | 198 | goto error_register; |
| 200 | 199 | ||
| 200 | device_enable_async_suspend(&ep_dev->dev); | ||
| 201 | endpoint->ep_dev = ep_dev; | 201 | endpoint->ep_dev = ep_dev; |
| 202 | return retval; | 202 | return retval; |
| 203 | 203 | ||
diff --git a/drivers/usb/core/hcd-pci.c b/drivers/usb/core/hcd-pci.c index b55d46070a2..f71e8e307e0 100644 --- a/drivers/usb/core/hcd-pci.c +++ b/drivers/usb/core/hcd-pci.c | |||
| @@ -405,7 +405,12 @@ static int suspend_common(struct device *dev, bool do_wakeup) | |||
| 405 | return retval; | 405 | return retval; |
| 406 | } | 406 | } |
| 407 | 407 | ||
| 408 | synchronize_irq(pci_dev->irq); | 408 | /* If MSI-X is enabled, the driver will have synchronized all vectors |
| 409 | * in pci_suspend(). If MSI or legacy PCI is enabled, that will be | ||
| 410 | * synchronized here. | ||
| 411 | */ | ||
| 412 | if (!hcd->msix_enabled) | ||
| 413 | synchronize_irq(pci_dev->irq); | ||
| 409 | 414 | ||
| 410 | /* Downstream ports from this root hub should already be quiesced, so | 415 | /* Downstream ports from this root hub should already be quiesced, so |
| 411 | * there will be no DMA activity. Now we can shut down the upstream | 416 | * there will be no DMA activity. Now we can shut down the upstream |
diff --git a/drivers/usb/core/hub.c b/drivers/usb/core/hub.c index b98efae6a1c..4310cc4b1cb 100644 --- a/drivers/usb/core/hub.c +++ b/drivers/usb/core/hub.c | |||
| @@ -676,6 +676,8 @@ static void hub_init_func3(struct work_struct *ws); | |||
| 676 | static void hub_activate(struct usb_hub *hub, enum hub_activation_type type) | 676 | static void hub_activate(struct usb_hub *hub, enum hub_activation_type type) |
| 677 | { | 677 | { |
| 678 | struct usb_device *hdev = hub->hdev; | 678 | struct usb_device *hdev = hub->hdev; |
| 679 | struct usb_hcd *hcd; | ||
| 680 | int ret; | ||
| 679 | int port1; | 681 | int port1; |
| 680 | int status; | 682 | int status; |
| 681 | bool need_debounce_delay = false; | 683 | bool need_debounce_delay = false; |
| @@ -714,6 +716,25 @@ static void hub_activate(struct usb_hub *hub, enum hub_activation_type type) | |||
| 714 | usb_autopm_get_interface_no_resume( | 716 | usb_autopm_get_interface_no_resume( |
| 715 | to_usb_interface(hub->intfdev)); | 717 | to_usb_interface(hub->intfdev)); |
| 716 | return; /* Continues at init2: below */ | 718 | return; /* Continues at init2: below */ |
| 719 | } else if (type == HUB_RESET_RESUME) { | ||
| 720 | /* The internal host controller state for the hub device | ||
| 721 | * may be gone after a host power loss on system resume. | ||
| 722 | * Update the device's info so the HW knows it's a hub. | ||
| 723 | */ | ||
| 724 | hcd = bus_to_hcd(hdev->bus); | ||
| 725 | if (hcd->driver->update_hub_device) { | ||
| 726 | ret = hcd->driver->update_hub_device(hcd, hdev, | ||
| 727 | &hub->tt, GFP_NOIO); | ||
| 728 | if (ret < 0) { | ||
| 729 | dev_err(hub->intfdev, "Host not " | ||
| 730 | "accepting hub info " | ||
| 731 | "update.\n"); | ||
| 732 | dev_err(hub->intfdev, "LS/FS devices " | ||
| 733 | "and hubs may not work " | ||
| 734 | "under this hub\n."); | ||
| 735 | } | ||
| 736 | } | ||
| 737 | hub_power_on(hub, true); | ||
| 717 | } else { | 738 | } else { |
| 718 | hub_power_on(hub, true); | 739 | hub_power_on(hub, true); |
| 719 | } | 740 | } |
diff --git a/drivers/usb/gadget/Kconfig b/drivers/usb/gadget/Kconfig index 1dc9739277b..06bb9d4587e 100644 --- a/drivers/usb/gadget/Kconfig +++ b/drivers/usb/gadget/Kconfig | |||
| @@ -509,7 +509,7 @@ config USB_LANGWELL | |||
| 509 | select USB_GADGET_SELECTED | 509 | select USB_GADGET_SELECTED |
| 510 | 510 | ||
| 511 | config USB_GADGET_EG20T | 511 | config USB_GADGET_EG20T |
| 512 | boolean "Intel EG20T(Topcliff) USB Device controller" | 512 | boolean "Intel EG20T PCH/OKI SEMICONDUCTOR ML7213 IOH UDC" |
| 513 | depends on PCI | 513 | depends on PCI |
| 514 | select USB_GADGET_DUALSPEED | 514 | select USB_GADGET_DUALSPEED |
| 515 | help | 515 | help |
| @@ -525,6 +525,11 @@ config USB_GADGET_EG20T | |||
| 525 | This driver dose not support interrupt transfer or isochronous | 525 | This driver dose not support interrupt transfer or isochronous |
| 526 | transfer modes. | 526 | transfer modes. |
| 527 | 527 | ||
| 528 | This driver also can be used for OKI SEMICONDUCTOR's ML7213 which is | ||
| 529 | for IVI(In-Vehicle Infotainment) use. | ||
| 530 | ML7213 is companion chip for Intel Atom E6xx series. | ||
| 531 | ML7213 is completely compatible for Intel EG20T PCH. | ||
| 532 | |||
| 528 | config USB_EG20T | 533 | config USB_EG20T |
| 529 | tristate | 534 | tristate |
| 530 | depends on USB_GADGET_EG20T | 535 | depends on USB_GADGET_EG20T |
diff --git a/drivers/usb/gadget/ci13xxx_udc.c b/drivers/usb/gadget/ci13xxx_udc.c index 31656a2b4ab..a1c67ae1572 100644 --- a/drivers/usb/gadget/ci13xxx_udc.c +++ b/drivers/usb/gadget/ci13xxx_udc.c | |||
| @@ -76,10 +76,21 @@ static DEFINE_SPINLOCK(udc_lock); | |||
| 76 | 76 | ||
| 77 | /* control endpoint description */ | 77 | /* control endpoint description */ |
| 78 | static const struct usb_endpoint_descriptor | 78 | static const struct usb_endpoint_descriptor |
| 79 | ctrl_endpt_desc = { | 79 | ctrl_endpt_out_desc = { |
| 80 | .bLength = USB_DT_ENDPOINT_SIZE, | 80 | .bLength = USB_DT_ENDPOINT_SIZE, |
| 81 | .bDescriptorType = USB_DT_ENDPOINT, | 81 | .bDescriptorType = USB_DT_ENDPOINT, |
| 82 | 82 | ||
| 83 | .bEndpointAddress = USB_DIR_OUT, | ||
| 84 | .bmAttributes = USB_ENDPOINT_XFER_CONTROL, | ||
| 85 | .wMaxPacketSize = cpu_to_le16(CTRL_PAYLOAD_MAX), | ||
| 86 | }; | ||
| 87 | |||
| 88 | static const struct usb_endpoint_descriptor | ||
| 89 | ctrl_endpt_in_desc = { | ||
| 90 | .bLength = USB_DT_ENDPOINT_SIZE, | ||
| 91 | .bDescriptorType = USB_DT_ENDPOINT, | ||
| 92 | |||
| 93 | .bEndpointAddress = USB_DIR_IN, | ||
| 83 | .bmAttributes = USB_ENDPOINT_XFER_CONTROL, | 94 | .bmAttributes = USB_ENDPOINT_XFER_CONTROL, |
| 84 | .wMaxPacketSize = cpu_to_le16(CTRL_PAYLOAD_MAX), | 95 | .wMaxPacketSize = cpu_to_le16(CTRL_PAYLOAD_MAX), |
| 85 | }; | 96 | }; |
| @@ -265,10 +276,10 @@ static int hw_device_init(void __iomem *base) | |||
| 265 | hw_bank.size /= sizeof(u32); | 276 | hw_bank.size /= sizeof(u32); |
| 266 | 277 | ||
| 267 | reg = hw_aread(ABS_DCCPARAMS, DCCPARAMS_DEN) >> ffs_nr(DCCPARAMS_DEN); | 278 | reg = hw_aread(ABS_DCCPARAMS, DCCPARAMS_DEN) >> ffs_nr(DCCPARAMS_DEN); |
| 268 | if (reg == 0 || reg > ENDPT_MAX) | 279 | hw_ep_max = reg * 2; /* cache hw ENDPT_MAX */ |
| 269 | return -ENODEV; | ||
| 270 | 280 | ||
| 271 | hw_ep_max = reg; /* cache hw ENDPT_MAX */ | 281 | if (hw_ep_max == 0 || hw_ep_max > ENDPT_MAX) |
| 282 | return -ENODEV; | ||
| 272 | 283 | ||
| 273 | /* setup lock mode ? */ | 284 | /* setup lock mode ? */ |
| 274 | 285 | ||
| @@ -1197,16 +1208,17 @@ static ssize_t show_qheads(struct device *dev, struct device_attribute *attr, | |||
| 1197 | } | 1208 | } |
| 1198 | 1209 | ||
| 1199 | spin_lock_irqsave(udc->lock, flags); | 1210 | spin_lock_irqsave(udc->lock, flags); |
| 1200 | for (i = 0; i < hw_ep_max; i++) { | 1211 | for (i = 0; i < hw_ep_max/2; i++) { |
| 1201 | struct ci13xxx_ep *mEp = &udc->ci13xxx_ep[i]; | 1212 | struct ci13xxx_ep *mEpRx = &udc->ci13xxx_ep[i]; |
| 1213 | struct ci13xxx_ep *mEpTx = &udc->ci13xxx_ep[i + hw_ep_max/2]; | ||
| 1202 | n += scnprintf(buf + n, PAGE_SIZE - n, | 1214 | n += scnprintf(buf + n, PAGE_SIZE - n, |
| 1203 | "EP=%02i: RX=%08X TX=%08X\n", | 1215 | "EP=%02i: RX=%08X TX=%08X\n", |
| 1204 | i, (u32)mEp->qh[RX].dma, (u32)mEp->qh[TX].dma); | 1216 | i, (u32)mEpRx->qh.dma, (u32)mEpTx->qh.dma); |
| 1205 | for (j = 0; j < (sizeof(struct ci13xxx_qh)/sizeof(u32)); j++) { | 1217 | for (j = 0; j < (sizeof(struct ci13xxx_qh)/sizeof(u32)); j++) { |
| 1206 | n += scnprintf(buf + n, PAGE_SIZE - n, | 1218 | n += scnprintf(buf + n, PAGE_SIZE - n, |
| 1207 | " %04X: %08X %08X\n", j, | 1219 | " %04X: %08X %08X\n", j, |
| 1208 | *((u32 *)mEp->qh[RX].ptr + j), | 1220 | *((u32 *)mEpRx->qh.ptr + j), |
| 1209 | *((u32 *)mEp->qh[TX].ptr + j)); | 1221 | *((u32 *)mEpTx->qh.ptr + j)); |
| 1210 | } | 1222 | } |
| 1211 | } | 1223 | } |
| 1212 | spin_unlock_irqrestore(udc->lock, flags); | 1224 | spin_unlock_irqrestore(udc->lock, flags); |
| @@ -1293,7 +1305,7 @@ static ssize_t show_requests(struct device *dev, struct device_attribute *attr, | |||
| 1293 | unsigned long flags; | 1305 | unsigned long flags; |
| 1294 | struct list_head *ptr = NULL; | 1306 | struct list_head *ptr = NULL; |
| 1295 | struct ci13xxx_req *req = NULL; | 1307 | struct ci13xxx_req *req = NULL; |
| 1296 | unsigned i, j, k, n = 0, qSize = sizeof(struct ci13xxx_td)/sizeof(u32); | 1308 | unsigned i, j, n = 0, qSize = sizeof(struct ci13xxx_td)/sizeof(u32); |
| 1297 | 1309 | ||
| 1298 | dbg_trace("[%s] %p\n", __func__, buf); | 1310 | dbg_trace("[%s] %p\n", __func__, buf); |
| 1299 | if (attr == NULL || buf == NULL) { | 1311 | if (attr == NULL || buf == NULL) { |
| @@ -1303,22 +1315,20 @@ static ssize_t show_requests(struct device *dev, struct device_attribute *attr, | |||
| 1303 | 1315 | ||
| 1304 | spin_lock_irqsave(udc->lock, flags); | 1316 | spin_lock_irqsave(udc->lock, flags); |
| 1305 | for (i = 0; i < hw_ep_max; i++) | 1317 | for (i = 0; i < hw_ep_max; i++) |
| 1306 | for (k = RX; k <= TX; k++) | 1318 | list_for_each(ptr, &udc->ci13xxx_ep[i].qh.queue) |
| 1307 | list_for_each(ptr, &udc->ci13xxx_ep[i].qh[k].queue) | 1319 | { |
| 1308 | { | 1320 | req = list_entry(ptr, struct ci13xxx_req, queue); |
| 1309 | req = list_entry(ptr, | 1321 | |
| 1310 | struct ci13xxx_req, queue); | 1322 | n += scnprintf(buf + n, PAGE_SIZE - n, |
| 1323 | "EP=%02i: TD=%08X %s\n", | ||
| 1324 | i % hw_ep_max/2, (u32)req->dma, | ||
| 1325 | ((i < hw_ep_max/2) ? "RX" : "TX")); | ||
| 1311 | 1326 | ||
| 1327 | for (j = 0; j < qSize; j++) | ||
| 1312 | n += scnprintf(buf + n, PAGE_SIZE - n, | 1328 | n += scnprintf(buf + n, PAGE_SIZE - n, |
| 1313 | "EP=%02i: TD=%08X %s\n", | 1329 | " %04X: %08X\n", j, |
| 1314 | i, (u32)req->dma, | 1330 | *((u32 *)req->ptr + j)); |
| 1315 | ((k == RX) ? "RX" : "TX")); | 1331 | } |
| 1316 | |||
| 1317 | for (j = 0; j < qSize; j++) | ||
| 1318 | n += scnprintf(buf + n, PAGE_SIZE - n, | ||
| 1319 | " %04X: %08X\n", j, | ||
| 1320 | *((u32 *)req->ptr + j)); | ||
| 1321 | } | ||
| 1322 | spin_unlock_irqrestore(udc->lock, flags); | 1332 | spin_unlock_irqrestore(udc->lock, flags); |
| 1323 | 1333 | ||
| 1324 | return n; | 1334 | return n; |
| @@ -1467,12 +1477,12 @@ static int _hardware_enqueue(struct ci13xxx_ep *mEp, struct ci13xxx_req *mReq) | |||
| 1467 | * At this point it's guaranteed exclusive access to qhead | 1477 | * At this point it's guaranteed exclusive access to qhead |
| 1468 | * (endpt is not primed) so it's no need to use tripwire | 1478 | * (endpt is not primed) so it's no need to use tripwire |
| 1469 | */ | 1479 | */ |
| 1470 | mEp->qh[mEp->dir].ptr->td.next = mReq->dma; /* TERMINATE = 0 */ | 1480 | mEp->qh.ptr->td.next = mReq->dma; /* TERMINATE = 0 */ |
| 1471 | mEp->qh[mEp->dir].ptr->td.token &= ~TD_STATUS; /* clear status */ | 1481 | mEp->qh.ptr->td.token &= ~TD_STATUS; /* clear status */ |
| 1472 | if (mReq->req.zero == 0) | 1482 | if (mReq->req.zero == 0) |
| 1473 | mEp->qh[mEp->dir].ptr->cap |= QH_ZLT; | 1483 | mEp->qh.ptr->cap |= QH_ZLT; |
| 1474 | else | 1484 | else |
| 1475 | mEp->qh[mEp->dir].ptr->cap &= ~QH_ZLT; | 1485 | mEp->qh.ptr->cap &= ~QH_ZLT; |
| 1476 | 1486 | ||
| 1477 | wmb(); /* synchronize before ep prime */ | 1487 | wmb(); /* synchronize before ep prime */ |
| 1478 | 1488 | ||
| @@ -1542,11 +1552,11 @@ __acquires(mEp->lock) | |||
| 1542 | 1552 | ||
| 1543 | hw_ep_flush(mEp->num, mEp->dir); | 1553 | hw_ep_flush(mEp->num, mEp->dir); |
| 1544 | 1554 | ||
| 1545 | while (!list_empty(&mEp->qh[mEp->dir].queue)) { | 1555 | while (!list_empty(&mEp->qh.queue)) { |
| 1546 | 1556 | ||
| 1547 | /* pop oldest request */ | 1557 | /* pop oldest request */ |
| 1548 | struct ci13xxx_req *mReq = \ | 1558 | struct ci13xxx_req *mReq = \ |
| 1549 | list_entry(mEp->qh[mEp->dir].queue.next, | 1559 | list_entry(mEp->qh.queue.next, |
| 1550 | struct ci13xxx_req, queue); | 1560 | struct ci13xxx_req, queue); |
| 1551 | list_del_init(&mReq->queue); | 1561 | list_del_init(&mReq->queue); |
| 1552 | mReq->req.status = -ESHUTDOWN; | 1562 | mReq->req.status = -ESHUTDOWN; |
| @@ -1571,8 +1581,6 @@ static int _gadget_stop_activity(struct usb_gadget *gadget) | |||
| 1571 | { | 1581 | { |
| 1572 | struct usb_ep *ep; | 1582 | struct usb_ep *ep; |
| 1573 | struct ci13xxx *udc = container_of(gadget, struct ci13xxx, gadget); | 1583 | struct ci13xxx *udc = container_of(gadget, struct ci13xxx, gadget); |
| 1574 | struct ci13xxx_ep *mEp = container_of(gadget->ep0, | ||
| 1575 | struct ci13xxx_ep, ep); | ||
| 1576 | 1584 | ||
| 1577 | trace("%p", gadget); | 1585 | trace("%p", gadget); |
| 1578 | 1586 | ||
| @@ -1583,7 +1591,8 @@ static int _gadget_stop_activity(struct usb_gadget *gadget) | |||
| 1583 | gadget_for_each_ep(ep, gadget) { | 1591 | gadget_for_each_ep(ep, gadget) { |
| 1584 | usb_ep_fifo_flush(ep); | 1592 | usb_ep_fifo_flush(ep); |
| 1585 | } | 1593 | } |
| 1586 | usb_ep_fifo_flush(gadget->ep0); | 1594 | usb_ep_fifo_flush(&udc->ep0out.ep); |
| 1595 | usb_ep_fifo_flush(&udc->ep0in.ep); | ||
| 1587 | 1596 | ||
| 1588 | udc->driver->disconnect(gadget); | 1597 | udc->driver->disconnect(gadget); |
| 1589 | 1598 | ||
| @@ -1591,11 +1600,12 @@ static int _gadget_stop_activity(struct usb_gadget *gadget) | |||
| 1591 | gadget_for_each_ep(ep, gadget) { | 1600 | gadget_for_each_ep(ep, gadget) { |
| 1592 | usb_ep_disable(ep); | 1601 | usb_ep_disable(ep); |
| 1593 | } | 1602 | } |
| 1594 | usb_ep_disable(gadget->ep0); | 1603 | usb_ep_disable(&udc->ep0out.ep); |
| 1604 | usb_ep_disable(&udc->ep0in.ep); | ||
| 1595 | 1605 | ||
| 1596 | if (mEp->status != NULL) { | 1606 | if (udc->status != NULL) { |
| 1597 | usb_ep_free_request(gadget->ep0, mEp->status); | 1607 | usb_ep_free_request(&udc->ep0in.ep, udc->status); |
| 1598 | mEp->status = NULL; | 1608 | udc->status = NULL; |
| 1599 | } | 1609 | } |
| 1600 | 1610 | ||
| 1601 | return 0; | 1611 | return 0; |
| @@ -1614,7 +1624,6 @@ static void isr_reset_handler(struct ci13xxx *udc) | |||
| 1614 | __releases(udc->lock) | 1624 | __releases(udc->lock) |
| 1615 | __acquires(udc->lock) | 1625 | __acquires(udc->lock) |
| 1616 | { | 1626 | { |
| 1617 | struct ci13xxx_ep *mEp = &udc->ci13xxx_ep[0]; | ||
| 1618 | int retval; | 1627 | int retval; |
| 1619 | 1628 | ||
| 1620 | trace("%p", udc); | 1629 | trace("%p", udc); |
| @@ -1635,11 +1644,15 @@ __acquires(udc->lock) | |||
| 1635 | if (retval) | 1644 | if (retval) |
| 1636 | goto done; | 1645 | goto done; |
| 1637 | 1646 | ||
| 1638 | retval = usb_ep_enable(&mEp->ep, &ctrl_endpt_desc); | 1647 | retval = usb_ep_enable(&udc->ep0out.ep, &ctrl_endpt_out_desc); |
| 1648 | if (retval) | ||
| 1649 | goto done; | ||
| 1650 | |||
| 1651 | retval = usb_ep_enable(&udc->ep0in.ep, &ctrl_endpt_in_desc); | ||
| 1639 | if (!retval) { | 1652 | if (!retval) { |
| 1640 | mEp->status = usb_ep_alloc_request(&mEp->ep, GFP_ATOMIC); | 1653 | udc->status = usb_ep_alloc_request(&udc->ep0in.ep, GFP_ATOMIC); |
| 1641 | if (mEp->status == NULL) { | 1654 | if (udc->status == NULL) { |
| 1642 | usb_ep_disable(&mEp->ep); | 1655 | usb_ep_disable(&udc->ep0out.ep); |
| 1643 | retval = -ENOMEM; | 1656 | retval = -ENOMEM; |
| 1644 | } | 1657 | } |
| 1645 | } | 1658 | } |
| @@ -1672,16 +1685,17 @@ static void isr_get_status_complete(struct usb_ep *ep, struct usb_request *req) | |||
| 1672 | 1685 | ||
| 1673 | /** | 1686 | /** |
| 1674 | * isr_get_status_response: get_status request response | 1687 | * isr_get_status_response: get_status request response |
| 1675 | * @ep: endpoint | 1688 | * @udc: udc struct |
| 1676 | * @setup: setup request packet | 1689 | * @setup: setup request packet |
| 1677 | * | 1690 | * |
| 1678 | * This function returns an error code | 1691 | * This function returns an error code |
| 1679 | */ | 1692 | */ |
| 1680 | static int isr_get_status_response(struct ci13xxx_ep *mEp, | 1693 | static int isr_get_status_response(struct ci13xxx *udc, |
| 1681 | struct usb_ctrlrequest *setup) | 1694 | struct usb_ctrlrequest *setup) |
| 1682 | __releases(mEp->lock) | 1695 | __releases(mEp->lock) |
| 1683 | __acquires(mEp->lock) | 1696 | __acquires(mEp->lock) |
| 1684 | { | 1697 | { |
| 1698 | struct ci13xxx_ep *mEp = &udc->ep0in; | ||
| 1685 | struct usb_request *req = NULL; | 1699 | struct usb_request *req = NULL; |
| 1686 | gfp_t gfp_flags = GFP_ATOMIC; | 1700 | gfp_t gfp_flags = GFP_ATOMIC; |
| 1687 | int dir, num, retval; | 1701 | int dir, num, retval; |
| @@ -1736,27 +1750,23 @@ __acquires(mEp->lock) | |||
| 1736 | 1750 | ||
| 1737 | /** | 1751 | /** |
| 1738 | * isr_setup_status_phase: queues the status phase of a setup transation | 1752 | * isr_setup_status_phase: queues the status phase of a setup transation |
| 1739 | * @mEp: endpoint | 1753 | * @udc: udc struct |
| 1740 | * | 1754 | * |
| 1741 | * This function returns an error code | 1755 | * This function returns an error code |
| 1742 | */ | 1756 | */ |
| 1743 | static int isr_setup_status_phase(struct ci13xxx_ep *mEp) | 1757 | static int isr_setup_status_phase(struct ci13xxx *udc) |
| 1744 | __releases(mEp->lock) | 1758 | __releases(mEp->lock) |
| 1745 | __acquires(mEp->lock) | 1759 | __acquires(mEp->lock) |
| 1746 | { | 1760 | { |
| 1747 | int retval; | 1761 | int retval; |
| 1762 | struct ci13xxx_ep *mEp; | ||
| 1748 | 1763 | ||
| 1749 | trace("%p", mEp); | 1764 | trace("%p", udc); |
| 1750 | |||
| 1751 | /* mEp is always valid & configured */ | ||
| 1752 | |||
| 1753 | if (mEp->type == USB_ENDPOINT_XFER_CONTROL) | ||
| 1754 | mEp->dir = (mEp->dir == TX) ? RX : TX; | ||
| 1755 | 1765 | ||
| 1756 | mEp->status->no_interrupt = 1; | 1766 | mEp = (udc->ep0_dir == TX) ? &udc->ep0out : &udc->ep0in; |
| 1757 | 1767 | ||
| 1758 | spin_unlock(mEp->lock); | 1768 | spin_unlock(mEp->lock); |
| 1759 | retval = usb_ep_queue(&mEp->ep, mEp->status, GFP_ATOMIC); | 1769 | retval = usb_ep_queue(&mEp->ep, udc->status, GFP_ATOMIC); |
| 1760 | spin_lock(mEp->lock); | 1770 | spin_lock(mEp->lock); |
| 1761 | 1771 | ||
| 1762 | return retval; | 1772 | return retval; |
| @@ -1778,11 +1788,11 @@ __acquires(mEp->lock) | |||
| 1778 | 1788 | ||
| 1779 | trace("%p", mEp); | 1789 | trace("%p", mEp); |
| 1780 | 1790 | ||
| 1781 | if (list_empty(&mEp->qh[mEp->dir].queue)) | 1791 | if (list_empty(&mEp->qh.queue)) |
| 1782 | return -EINVAL; | 1792 | return -EINVAL; |
| 1783 | 1793 | ||
| 1784 | /* pop oldest request */ | 1794 | /* pop oldest request */ |
| 1785 | mReq = list_entry(mEp->qh[mEp->dir].queue.next, | 1795 | mReq = list_entry(mEp->qh.queue.next, |
| 1786 | struct ci13xxx_req, queue); | 1796 | struct ci13xxx_req, queue); |
| 1787 | list_del_init(&mReq->queue); | 1797 | list_del_init(&mReq->queue); |
| 1788 | 1798 | ||
| @@ -1794,10 +1804,10 @@ __acquires(mEp->lock) | |||
| 1794 | 1804 | ||
| 1795 | dbg_done(_usb_addr(mEp), mReq->ptr->token, retval); | 1805 | dbg_done(_usb_addr(mEp), mReq->ptr->token, retval); |
| 1796 | 1806 | ||
| 1797 | if (!list_empty(&mEp->qh[mEp->dir].queue)) { | 1807 | if (!list_empty(&mEp->qh.queue)) { |
| 1798 | struct ci13xxx_req* mReqEnq; | 1808 | struct ci13xxx_req* mReqEnq; |
| 1799 | 1809 | ||
| 1800 | mReqEnq = list_entry(mEp->qh[mEp->dir].queue.next, | 1810 | mReqEnq = list_entry(mEp->qh.queue.next, |
| 1801 | struct ci13xxx_req, queue); | 1811 | struct ci13xxx_req, queue); |
| 1802 | _hardware_enqueue(mEp, mReqEnq); | 1812 | _hardware_enqueue(mEp, mReqEnq); |
| 1803 | } | 1813 | } |
| @@ -1836,16 +1846,14 @@ __acquires(udc->lock) | |||
| 1836 | int type, num, err = -EINVAL; | 1846 | int type, num, err = -EINVAL; |
| 1837 | struct usb_ctrlrequest req; | 1847 | struct usb_ctrlrequest req; |
| 1838 | 1848 | ||
| 1839 | |||
| 1840 | if (mEp->desc == NULL) | 1849 | if (mEp->desc == NULL) |
| 1841 | continue; /* not configured */ | 1850 | continue; /* not configured */ |
| 1842 | 1851 | ||
| 1843 | if ((mEp->dir == RX && hw_test_and_clear_complete(i)) || | 1852 | if (hw_test_and_clear_complete(i)) { |
| 1844 | (mEp->dir == TX && hw_test_and_clear_complete(i + 16))) { | ||
| 1845 | err = isr_tr_complete_low(mEp); | 1853 | err = isr_tr_complete_low(mEp); |
| 1846 | if (mEp->type == USB_ENDPOINT_XFER_CONTROL) { | 1854 | if (mEp->type == USB_ENDPOINT_XFER_CONTROL) { |
| 1847 | if (err > 0) /* needs status phase */ | 1855 | if (err > 0) /* needs status phase */ |
| 1848 | err = isr_setup_status_phase(mEp); | 1856 | err = isr_setup_status_phase(udc); |
| 1849 | if (err < 0) { | 1857 | if (err < 0) { |
| 1850 | dbg_event(_usb_addr(mEp), | 1858 | dbg_event(_usb_addr(mEp), |
| 1851 | "ERROR", err); | 1859 | "ERROR", err); |
| @@ -1866,15 +1874,22 @@ __acquires(udc->lock) | |||
| 1866 | continue; | 1874 | continue; |
| 1867 | } | 1875 | } |
| 1868 | 1876 | ||
| 1877 | /* | ||
| 1878 | * Flush data and handshake transactions of previous | ||
| 1879 | * setup packet. | ||
| 1880 | */ | ||
| 1881 | _ep_nuke(&udc->ep0out); | ||
| 1882 | _ep_nuke(&udc->ep0in); | ||
| 1883 | |||
| 1869 | /* read_setup_packet */ | 1884 | /* read_setup_packet */ |
| 1870 | do { | 1885 | do { |
| 1871 | hw_test_and_set_setup_guard(); | 1886 | hw_test_and_set_setup_guard(); |
| 1872 | memcpy(&req, &mEp->qh[RX].ptr->setup, sizeof(req)); | 1887 | memcpy(&req, &mEp->qh.ptr->setup, sizeof(req)); |
| 1873 | } while (!hw_test_and_clear_setup_guard()); | 1888 | } while (!hw_test_and_clear_setup_guard()); |
| 1874 | 1889 | ||
| 1875 | type = req.bRequestType; | 1890 | type = req.bRequestType; |
| 1876 | 1891 | ||
| 1877 | mEp->dir = (type & USB_DIR_IN) ? TX : RX; | 1892 | udc->ep0_dir = (type & USB_DIR_IN) ? TX : RX; |
| 1878 | 1893 | ||
| 1879 | dbg_setup(_usb_addr(mEp), &req); | 1894 | dbg_setup(_usb_addr(mEp), &req); |
| 1880 | 1895 | ||
| @@ -1895,7 +1910,7 @@ __acquires(udc->lock) | |||
| 1895 | if (err) | 1910 | if (err) |
| 1896 | break; | 1911 | break; |
| 1897 | } | 1912 | } |
| 1898 | err = isr_setup_status_phase(mEp); | 1913 | err = isr_setup_status_phase(udc); |
| 1899 | break; | 1914 | break; |
| 1900 | case USB_REQ_GET_STATUS: | 1915 | case USB_REQ_GET_STATUS: |
| 1901 | if (type != (USB_DIR_IN|USB_RECIP_DEVICE) && | 1916 | if (type != (USB_DIR_IN|USB_RECIP_DEVICE) && |
| @@ -1905,7 +1920,7 @@ __acquires(udc->lock) | |||
| 1905 | if (le16_to_cpu(req.wLength) != 2 || | 1920 | if (le16_to_cpu(req.wLength) != 2 || |
| 1906 | le16_to_cpu(req.wValue) != 0) | 1921 | le16_to_cpu(req.wValue) != 0) |
| 1907 | break; | 1922 | break; |
| 1908 | err = isr_get_status_response(mEp, &req); | 1923 | err = isr_get_status_response(udc, &req); |
| 1909 | break; | 1924 | break; |
| 1910 | case USB_REQ_SET_ADDRESS: | 1925 | case USB_REQ_SET_ADDRESS: |
| 1911 | if (type != (USB_DIR_OUT|USB_RECIP_DEVICE)) | 1926 | if (type != (USB_DIR_OUT|USB_RECIP_DEVICE)) |
| @@ -1916,7 +1931,7 @@ __acquires(udc->lock) | |||
| 1916 | err = hw_usb_set_address((u8)le16_to_cpu(req.wValue)); | 1931 | err = hw_usb_set_address((u8)le16_to_cpu(req.wValue)); |
| 1917 | if (err) | 1932 | if (err) |
| 1918 | break; | 1933 | break; |
| 1919 | err = isr_setup_status_phase(mEp); | 1934 | err = isr_setup_status_phase(udc); |
| 1920 | break; | 1935 | break; |
| 1921 | case USB_REQ_SET_FEATURE: | 1936 | case USB_REQ_SET_FEATURE: |
| 1922 | if (type != (USB_DIR_OUT|USB_RECIP_ENDPOINT) && | 1937 | if (type != (USB_DIR_OUT|USB_RECIP_ENDPOINT) && |
| @@ -1932,12 +1947,12 @@ __acquires(udc->lock) | |||
| 1932 | spin_lock(udc->lock); | 1947 | spin_lock(udc->lock); |
| 1933 | if (err) | 1948 | if (err) |
| 1934 | break; | 1949 | break; |
| 1935 | err = isr_setup_status_phase(mEp); | 1950 | err = isr_setup_status_phase(udc); |
| 1936 | break; | 1951 | break; |
| 1937 | default: | 1952 | default: |
| 1938 | delegate: | 1953 | delegate: |
| 1939 | if (req.wLength == 0) /* no data phase */ | 1954 | if (req.wLength == 0) /* no data phase */ |
| 1940 | mEp->dir = TX; | 1955 | udc->ep0_dir = TX; |
| 1941 | 1956 | ||
| 1942 | spin_unlock(udc->lock); | 1957 | spin_unlock(udc->lock); |
| 1943 | err = udc->driver->setup(&udc->gadget, &req); | 1958 | err = udc->driver->setup(&udc->gadget, &req); |
| @@ -1968,7 +1983,7 @@ static int ep_enable(struct usb_ep *ep, | |||
| 1968 | const struct usb_endpoint_descriptor *desc) | 1983 | const struct usb_endpoint_descriptor *desc) |
| 1969 | { | 1984 | { |
| 1970 | struct ci13xxx_ep *mEp = container_of(ep, struct ci13xxx_ep, ep); | 1985 | struct ci13xxx_ep *mEp = container_of(ep, struct ci13xxx_ep, ep); |
| 1971 | int direction, retval = 0; | 1986 | int retval = 0; |
| 1972 | unsigned long flags; | 1987 | unsigned long flags; |
| 1973 | 1988 | ||
| 1974 | trace("%p, %p", ep, desc); | 1989 | trace("%p, %p", ep, desc); |
| @@ -1982,7 +1997,7 @@ static int ep_enable(struct usb_ep *ep, | |||
| 1982 | 1997 | ||
| 1983 | mEp->desc = desc; | 1998 | mEp->desc = desc; |
| 1984 | 1999 | ||
| 1985 | if (!list_empty(&mEp->qh[mEp->dir].queue)) | 2000 | if (!list_empty(&mEp->qh.queue)) |
| 1986 | warn("enabling a non-empty endpoint!"); | 2001 | warn("enabling a non-empty endpoint!"); |
| 1987 | 2002 | ||
| 1988 | mEp->dir = usb_endpoint_dir_in(desc) ? TX : RX; | 2003 | mEp->dir = usb_endpoint_dir_in(desc) ? TX : RX; |
| @@ -1991,29 +2006,22 @@ static int ep_enable(struct usb_ep *ep, | |||
| 1991 | 2006 | ||
| 1992 | mEp->ep.maxpacket = __constant_le16_to_cpu(desc->wMaxPacketSize); | 2007 | mEp->ep.maxpacket = __constant_le16_to_cpu(desc->wMaxPacketSize); |
| 1993 | 2008 | ||
| 1994 | direction = mEp->dir; | 2009 | dbg_event(_usb_addr(mEp), "ENABLE", 0); |
| 1995 | do { | ||
| 1996 | dbg_event(_usb_addr(mEp), "ENABLE", 0); | ||
| 1997 | 2010 | ||
| 1998 | mEp->qh[mEp->dir].ptr->cap = 0; | 2011 | mEp->qh.ptr->cap = 0; |
| 1999 | 2012 | ||
| 2000 | if (mEp->type == USB_ENDPOINT_XFER_CONTROL) | 2013 | if (mEp->type == USB_ENDPOINT_XFER_CONTROL) |
| 2001 | mEp->qh[mEp->dir].ptr->cap |= QH_IOS; | 2014 | mEp->qh.ptr->cap |= QH_IOS; |
| 2002 | else if (mEp->type == USB_ENDPOINT_XFER_ISOC) | 2015 | else if (mEp->type == USB_ENDPOINT_XFER_ISOC) |
| 2003 | mEp->qh[mEp->dir].ptr->cap &= ~QH_MULT; | 2016 | mEp->qh.ptr->cap &= ~QH_MULT; |
| 2004 | else | 2017 | else |
| 2005 | mEp->qh[mEp->dir].ptr->cap &= ~QH_ZLT; | 2018 | mEp->qh.ptr->cap &= ~QH_ZLT; |
| 2006 | |||
| 2007 | mEp->qh[mEp->dir].ptr->cap |= | ||
| 2008 | (mEp->ep.maxpacket << ffs_nr(QH_MAX_PKT)) & QH_MAX_PKT; | ||
| 2009 | mEp->qh[mEp->dir].ptr->td.next |= TD_TERMINATE; /* needed? */ | ||
| 2010 | |||
| 2011 | retval |= hw_ep_enable(mEp->num, mEp->dir, mEp->type); | ||
| 2012 | 2019 | ||
| 2013 | if (mEp->type == USB_ENDPOINT_XFER_CONTROL) | 2020 | mEp->qh.ptr->cap |= |
| 2014 | mEp->dir = (mEp->dir == TX) ? RX : TX; | 2021 | (mEp->ep.maxpacket << ffs_nr(QH_MAX_PKT)) & QH_MAX_PKT; |
| 2022 | mEp->qh.ptr->td.next |= TD_TERMINATE; /* needed? */ | ||
| 2015 | 2023 | ||
| 2016 | } while (mEp->dir != direction); | 2024 | retval |= hw_ep_enable(mEp->num, mEp->dir, mEp->type); |
| 2017 | 2025 | ||
| 2018 | spin_unlock_irqrestore(mEp->lock, flags); | 2026 | spin_unlock_irqrestore(mEp->lock, flags); |
| 2019 | return retval; | 2027 | return retval; |
| @@ -2146,7 +2154,7 @@ static int ep_queue(struct usb_ep *ep, struct usb_request *req, | |||
| 2146 | spin_lock_irqsave(mEp->lock, flags); | 2154 | spin_lock_irqsave(mEp->lock, flags); |
| 2147 | 2155 | ||
| 2148 | if (mEp->type == USB_ENDPOINT_XFER_CONTROL && | 2156 | if (mEp->type == USB_ENDPOINT_XFER_CONTROL && |
| 2149 | !list_empty(&mEp->qh[mEp->dir].queue)) { | 2157 | !list_empty(&mEp->qh.queue)) { |
| 2150 | _ep_nuke(mEp); | 2158 | _ep_nuke(mEp); |
| 2151 | retval = -EOVERFLOW; | 2159 | retval = -EOVERFLOW; |
| 2152 | warn("endpoint ctrl %X nuked", _usb_addr(mEp)); | 2160 | warn("endpoint ctrl %X nuked", _usb_addr(mEp)); |
| @@ -2170,9 +2178,9 @@ static int ep_queue(struct usb_ep *ep, struct usb_request *req, | |||
| 2170 | /* push request */ | 2178 | /* push request */ |
| 2171 | mReq->req.status = -EINPROGRESS; | 2179 | mReq->req.status = -EINPROGRESS; |
| 2172 | mReq->req.actual = 0; | 2180 | mReq->req.actual = 0; |
| 2173 | list_add_tail(&mReq->queue, &mEp->qh[mEp->dir].queue); | 2181 | list_add_tail(&mReq->queue, &mEp->qh.queue); |
| 2174 | 2182 | ||
| 2175 | if (list_is_singular(&mEp->qh[mEp->dir].queue)) | 2183 | if (list_is_singular(&mEp->qh.queue)) |
| 2176 | retval = _hardware_enqueue(mEp, mReq); | 2184 | retval = _hardware_enqueue(mEp, mReq); |
| 2177 | 2185 | ||
| 2178 | if (retval == -EALREADY) { | 2186 | if (retval == -EALREADY) { |
| @@ -2199,7 +2207,7 @@ static int ep_dequeue(struct usb_ep *ep, struct usb_request *req) | |||
| 2199 | trace("%p, %p", ep, req); | 2207 | trace("%p, %p", ep, req); |
| 2200 | 2208 | ||
| 2201 | if (ep == NULL || req == NULL || mEp->desc == NULL || | 2209 | if (ep == NULL || req == NULL || mEp->desc == NULL || |
| 2202 | list_empty(&mReq->queue) || list_empty(&mEp->qh[mEp->dir].queue)) | 2210 | list_empty(&mReq->queue) || list_empty(&mEp->qh.queue)) |
| 2203 | return -EINVAL; | 2211 | return -EINVAL; |
| 2204 | 2212 | ||
| 2205 | spin_lock_irqsave(mEp->lock, flags); | 2213 | spin_lock_irqsave(mEp->lock, flags); |
| @@ -2244,7 +2252,7 @@ static int ep_set_halt(struct usb_ep *ep, int value) | |||
| 2244 | #ifndef STALL_IN | 2252 | #ifndef STALL_IN |
| 2245 | /* g_file_storage MS compliant but g_zero fails chapter 9 compliance */ | 2253 | /* g_file_storage MS compliant but g_zero fails chapter 9 compliance */ |
| 2246 | if (value && mEp->type == USB_ENDPOINT_XFER_BULK && mEp->dir == TX && | 2254 | if (value && mEp->type == USB_ENDPOINT_XFER_BULK && mEp->dir == TX && |
| 2247 | !list_empty(&mEp->qh[mEp->dir].queue)) { | 2255 | !list_empty(&mEp->qh.queue)) { |
| 2248 | spin_unlock_irqrestore(mEp->lock, flags); | 2256 | spin_unlock_irqrestore(mEp->lock, flags); |
| 2249 | return -EAGAIN; | 2257 | return -EAGAIN; |
| 2250 | } | 2258 | } |
| @@ -2355,7 +2363,7 @@ static int ci13xxx_vbus_session(struct usb_gadget *_gadget, int is_active) | |||
| 2355 | if (is_active) { | 2363 | if (is_active) { |
| 2356 | pm_runtime_get_sync(&_gadget->dev); | 2364 | pm_runtime_get_sync(&_gadget->dev); |
| 2357 | hw_device_reset(udc); | 2365 | hw_device_reset(udc); |
| 2358 | hw_device_state(udc->ci13xxx_ep[0].qh[RX].dma); | 2366 | hw_device_state(udc->ep0out.qh.dma); |
| 2359 | } else { | 2367 | } else { |
| 2360 | hw_device_state(0); | 2368 | hw_device_state(0); |
| 2361 | if (udc->udc_driver->notify_event) | 2369 | if (udc->udc_driver->notify_event) |
| @@ -2390,7 +2398,8 @@ int usb_gadget_probe_driver(struct usb_gadget_driver *driver, | |||
| 2390 | int (*bind)(struct usb_gadget *)) | 2398 | int (*bind)(struct usb_gadget *)) |
| 2391 | { | 2399 | { |
| 2392 | struct ci13xxx *udc = _udc; | 2400 | struct ci13xxx *udc = _udc; |
| 2393 | unsigned long i, k, flags; | 2401 | unsigned long flags; |
| 2402 | int i, j; | ||
| 2394 | int retval = -ENOMEM; | 2403 | int retval = -ENOMEM; |
| 2395 | 2404 | ||
| 2396 | trace("%p", driver); | 2405 | trace("%p", driver); |
| @@ -2427,45 +2436,46 @@ int usb_gadget_probe_driver(struct usb_gadget_driver *driver, | |||
| 2427 | 2436 | ||
| 2428 | info("hw_ep_max = %d", hw_ep_max); | 2437 | info("hw_ep_max = %d", hw_ep_max); |
| 2429 | 2438 | ||
| 2430 | udc->driver = driver; | ||
| 2431 | udc->gadget.dev.driver = NULL; | 2439 | udc->gadget.dev.driver = NULL; |
| 2432 | 2440 | ||
| 2433 | retval = 0; | 2441 | retval = 0; |
| 2434 | for (i = 0; i < hw_ep_max; i++) { | 2442 | for (i = 0; i < hw_ep_max/2; i++) { |
| 2435 | struct ci13xxx_ep *mEp = &udc->ci13xxx_ep[i]; | 2443 | for (j = RX; j <= TX; j++) { |
| 2444 | int k = i + j * hw_ep_max/2; | ||
| 2445 | struct ci13xxx_ep *mEp = &udc->ci13xxx_ep[k]; | ||
| 2436 | 2446 | ||
| 2437 | scnprintf(mEp->name, sizeof(mEp->name), "ep%i", (int)i); | 2447 | scnprintf(mEp->name, sizeof(mEp->name), "ep%i%s", i, |
| 2448 | (j == TX) ? "in" : "out"); | ||
| 2438 | 2449 | ||
| 2439 | mEp->lock = udc->lock; | 2450 | mEp->lock = udc->lock; |
| 2440 | mEp->device = &udc->gadget.dev; | 2451 | mEp->device = &udc->gadget.dev; |
| 2441 | mEp->td_pool = udc->td_pool; | 2452 | mEp->td_pool = udc->td_pool; |
| 2442 | 2453 | ||
| 2443 | mEp->ep.name = mEp->name; | 2454 | mEp->ep.name = mEp->name; |
| 2444 | mEp->ep.ops = &usb_ep_ops; | 2455 | mEp->ep.ops = &usb_ep_ops; |
| 2445 | mEp->ep.maxpacket = CTRL_PAYLOAD_MAX; | 2456 | mEp->ep.maxpacket = CTRL_PAYLOAD_MAX; |
| 2446 | 2457 | ||
| 2447 | /* this allocation cannot be random */ | 2458 | INIT_LIST_HEAD(&mEp->qh.queue); |
| 2448 | for (k = RX; k <= TX; k++) { | ||
| 2449 | INIT_LIST_HEAD(&mEp->qh[k].queue); | ||
| 2450 | spin_unlock_irqrestore(udc->lock, flags); | 2459 | spin_unlock_irqrestore(udc->lock, flags); |
| 2451 | mEp->qh[k].ptr = dma_pool_alloc(udc->qh_pool, | 2460 | mEp->qh.ptr = dma_pool_alloc(udc->qh_pool, GFP_KERNEL, |
| 2452 | GFP_KERNEL, | 2461 | &mEp->qh.dma); |
| 2453 | &mEp->qh[k].dma); | ||
| 2454 | spin_lock_irqsave(udc->lock, flags); | 2462 | spin_lock_irqsave(udc->lock, flags); |
| 2455 | if (mEp->qh[k].ptr == NULL) | 2463 | if (mEp->qh.ptr == NULL) |
| 2456 | retval = -ENOMEM; | 2464 | retval = -ENOMEM; |
| 2457 | else | 2465 | else |
| 2458 | memset(mEp->qh[k].ptr, 0, | 2466 | memset(mEp->qh.ptr, 0, sizeof(*mEp->qh.ptr)); |
| 2459 | sizeof(*mEp->qh[k].ptr)); | 2467 | |
| 2460 | } | 2468 | /* skip ep0 out and in endpoints */ |
| 2461 | if (i == 0) | 2469 | if (i == 0) |
| 2462 | udc->gadget.ep0 = &mEp->ep; | 2470 | continue; |
| 2463 | else | 2471 | |
| 2464 | list_add_tail(&mEp->ep.ep_list, &udc->gadget.ep_list); | 2472 | list_add_tail(&mEp->ep.ep_list, &udc->gadget.ep_list); |
| 2473 | } | ||
| 2465 | } | 2474 | } |
| 2466 | if (retval) | 2475 | if (retval) |
| 2467 | goto done; | 2476 | goto done; |
| 2468 | 2477 | ||
| 2478 | udc->gadget.ep0 = &udc->ep0in.ep; | ||
| 2469 | /* bind gadget */ | 2479 | /* bind gadget */ |
| 2470 | driver->driver.bus = NULL; | 2480 | driver->driver.bus = NULL; |
| 2471 | udc->gadget.dev.driver = &driver->driver; | 2481 | udc->gadget.dev.driver = &driver->driver; |
| @@ -2479,6 +2489,7 @@ int usb_gadget_probe_driver(struct usb_gadget_driver *driver, | |||
| 2479 | goto done; | 2489 | goto done; |
| 2480 | } | 2490 | } |
| 2481 | 2491 | ||
| 2492 | udc->driver = driver; | ||
| 2482 | pm_runtime_get_sync(&udc->gadget.dev); | 2493 | pm_runtime_get_sync(&udc->gadget.dev); |
| 2483 | if (udc->udc_driver->flags & CI13XXX_PULLUP_ON_VBUS) { | 2494 | if (udc->udc_driver->flags & CI13XXX_PULLUP_ON_VBUS) { |
| 2484 | if (udc->vbus_active) { | 2495 | if (udc->vbus_active) { |
| @@ -2490,14 +2501,12 @@ int usb_gadget_probe_driver(struct usb_gadget_driver *driver, | |||
| 2490 | } | 2501 | } |
| 2491 | } | 2502 | } |
| 2492 | 2503 | ||
| 2493 | retval = hw_device_state(udc->ci13xxx_ep[0].qh[RX].dma); | 2504 | retval = hw_device_state(udc->ep0out.qh.dma); |
| 2494 | if (retval) | 2505 | if (retval) |
| 2495 | pm_runtime_put_sync(&udc->gadget.dev); | 2506 | pm_runtime_put_sync(&udc->gadget.dev); |
| 2496 | 2507 | ||
| 2497 | done: | 2508 | done: |
| 2498 | spin_unlock_irqrestore(udc->lock, flags); | 2509 | spin_unlock_irqrestore(udc->lock, flags); |
| 2499 | if (retval) | ||
| 2500 | usb_gadget_unregister_driver(driver); | ||
| 2501 | return retval; | 2510 | return retval; |
| 2502 | } | 2511 | } |
| 2503 | EXPORT_SYMBOL(usb_gadget_probe_driver); | 2512 | EXPORT_SYMBOL(usb_gadget_probe_driver); |
| @@ -2510,7 +2519,7 @@ EXPORT_SYMBOL(usb_gadget_probe_driver); | |||
| 2510 | int usb_gadget_unregister_driver(struct usb_gadget_driver *driver) | 2519 | int usb_gadget_unregister_driver(struct usb_gadget_driver *driver) |
| 2511 | { | 2520 | { |
| 2512 | struct ci13xxx *udc = _udc; | 2521 | struct ci13xxx *udc = _udc; |
| 2513 | unsigned long i, k, flags; | 2522 | unsigned long i, flags; |
| 2514 | 2523 | ||
| 2515 | trace("%p", driver); | 2524 | trace("%p", driver); |
| 2516 | 2525 | ||
| @@ -2546,17 +2555,14 @@ int usb_gadget_unregister_driver(struct usb_gadget_driver *driver) | |||
| 2546 | for (i = 0; i < hw_ep_max; i++) { | 2555 | for (i = 0; i < hw_ep_max; i++) { |
| 2547 | struct ci13xxx_ep *mEp = &udc->ci13xxx_ep[i]; | 2556 | struct ci13xxx_ep *mEp = &udc->ci13xxx_ep[i]; |
| 2548 | 2557 | ||
| 2549 | if (i == 0) | 2558 | if (!list_empty(&mEp->ep.ep_list)) |
| 2550 | udc->gadget.ep0 = NULL; | ||
| 2551 | else if (!list_empty(&mEp->ep.ep_list)) | ||
| 2552 | list_del_init(&mEp->ep.ep_list); | 2559 | list_del_init(&mEp->ep.ep_list); |
| 2553 | 2560 | ||
| 2554 | for (k = RX; k <= TX; k++) | 2561 | if (mEp->qh.ptr != NULL) |
| 2555 | if (mEp->qh[k].ptr != NULL) | 2562 | dma_pool_free(udc->qh_pool, mEp->qh.ptr, mEp->qh.dma); |
| 2556 | dma_pool_free(udc->qh_pool, | ||
| 2557 | mEp->qh[k].ptr, mEp->qh[k].dma); | ||
| 2558 | } | 2563 | } |
| 2559 | 2564 | ||
| 2565 | udc->gadget.ep0 = NULL; | ||
| 2560 | udc->driver = NULL; | 2566 | udc->driver = NULL; |
| 2561 | 2567 | ||
| 2562 | spin_unlock_irqrestore(udc->lock, flags); | 2568 | spin_unlock_irqrestore(udc->lock, flags); |
diff --git a/drivers/usb/gadget/ci13xxx_udc.h b/drivers/usb/gadget/ci13xxx_udc.h index f61fed07f76..a2492b65f98 100644 --- a/drivers/usb/gadget/ci13xxx_udc.h +++ b/drivers/usb/gadget/ci13xxx_udc.h | |||
| @@ -20,7 +20,7 @@ | |||
| 20 | * DEFINE | 20 | * DEFINE |
| 21 | *****************************************************************************/ | 21 | *****************************************************************************/ |
| 22 | #define CI13XXX_PAGE_SIZE 4096ul /* page size for TD's */ | 22 | #define CI13XXX_PAGE_SIZE 4096ul /* page size for TD's */ |
| 23 | #define ENDPT_MAX (16) | 23 | #define ENDPT_MAX (32) |
| 24 | #define CTRL_PAYLOAD_MAX (64) | 24 | #define CTRL_PAYLOAD_MAX (64) |
| 25 | #define RX (0) /* similar to USB_DIR_OUT but can be used as an index */ | 25 | #define RX (0) /* similar to USB_DIR_OUT but can be used as an index */ |
| 26 | #define TX (1) /* similar to USB_DIR_IN but can be used as an index */ | 26 | #define TX (1) /* similar to USB_DIR_IN but can be used as an index */ |
| @@ -88,8 +88,7 @@ struct ci13xxx_ep { | |||
| 88 | struct list_head queue; | 88 | struct list_head queue; |
| 89 | struct ci13xxx_qh *ptr; | 89 | struct ci13xxx_qh *ptr; |
| 90 | dma_addr_t dma; | 90 | dma_addr_t dma; |
| 91 | } qh[2]; | 91 | } qh; |
| 92 | struct usb_request *status; | ||
| 93 | int wedge; | 92 | int wedge; |
| 94 | 93 | ||
| 95 | /* global resources */ | 94 | /* global resources */ |
| @@ -119,9 +118,13 @@ struct ci13xxx { | |||
| 119 | 118 | ||
| 120 | struct dma_pool *qh_pool; /* DMA pool for queue heads */ | 119 | struct dma_pool *qh_pool; /* DMA pool for queue heads */ |
| 121 | struct dma_pool *td_pool; /* DMA pool for transfer descs */ | 120 | struct dma_pool *td_pool; /* DMA pool for transfer descs */ |
| 121 | struct usb_request *status; /* ep0 status request */ | ||
| 122 | 122 | ||
| 123 | struct usb_gadget gadget; /* USB slave device */ | 123 | struct usb_gadget gadget; /* USB slave device */ |
| 124 | struct ci13xxx_ep ci13xxx_ep[ENDPT_MAX]; /* extended endpts */ | 124 | struct ci13xxx_ep ci13xxx_ep[ENDPT_MAX]; /* extended endpts */ |
| 125 | u32 ep0_dir; /* ep0 direction */ | ||
| 126 | #define ep0out ci13xxx_ep[0] | ||
| 127 | #define ep0in ci13xxx_ep[16] | ||
| 125 | 128 | ||
| 126 | struct usb_gadget_driver *driver; /* 3rd party gadget driver */ | 129 | struct usb_gadget_driver *driver; /* 3rd party gadget driver */ |
| 127 | struct ci13xxx_udc_driver *udc_driver; /* device controller driver */ | 130 | struct ci13xxx_udc_driver *udc_driver; /* device controller driver */ |
diff --git a/drivers/usb/gadget/composite.c b/drivers/usb/gadget/composite.c index f6ff8456d52..1ba4befe336 100644 --- a/drivers/usb/gadget/composite.c +++ b/drivers/usb/gadget/composite.c | |||
| @@ -928,8 +928,9 @@ unknown: | |||
| 928 | */ | 928 | */ |
| 929 | switch (ctrl->bRequestType & USB_RECIP_MASK) { | 929 | switch (ctrl->bRequestType & USB_RECIP_MASK) { |
| 930 | case USB_RECIP_INTERFACE: | 930 | case USB_RECIP_INTERFACE: |
| 931 | if (cdev->config) | 931 | if (!cdev->config || w_index >= MAX_CONFIG_INTERFACES) |
| 932 | f = cdev->config->interface[intf]; | 932 | break; |
| 933 | f = cdev->config->interface[intf]; | ||
| 933 | break; | 934 | break; |
| 934 | 935 | ||
| 935 | case USB_RECIP_ENDPOINT: | 936 | case USB_RECIP_ENDPOINT: |
diff --git a/drivers/usb/gadget/pch_udc.c b/drivers/usb/gadget/pch_udc.c index 0c8dd81dddc..b120dbb64d0 100644 --- a/drivers/usb/gadget/pch_udc.c +++ b/drivers/usb/gadget/pch_udc.c | |||
| @@ -198,10 +198,10 @@ | |||
| 198 | #define PCH_UDC_BRLEN 0x0F /* Burst length */ | 198 | #define PCH_UDC_BRLEN 0x0F /* Burst length */ |
| 199 | #define PCH_UDC_THLEN 0x1F /* Threshold length */ | 199 | #define PCH_UDC_THLEN 0x1F /* Threshold length */ |
| 200 | /* Value of EP Buffer Size */ | 200 | /* Value of EP Buffer Size */ |
| 201 | #define UDC_EP0IN_BUFF_SIZE 64 | 201 | #define UDC_EP0IN_BUFF_SIZE 16 |
| 202 | #define UDC_EPIN_BUFF_SIZE 512 | 202 | #define UDC_EPIN_BUFF_SIZE 256 |
| 203 | #define UDC_EP0OUT_BUFF_SIZE 64 | 203 | #define UDC_EP0OUT_BUFF_SIZE 16 |
| 204 | #define UDC_EPOUT_BUFF_SIZE 512 | 204 | #define UDC_EPOUT_BUFF_SIZE 256 |
| 205 | /* Value of EP maximum packet size */ | 205 | /* Value of EP maximum packet size */ |
| 206 | #define UDC_EP0IN_MAX_PKT_SIZE 64 | 206 | #define UDC_EP0IN_MAX_PKT_SIZE 64 |
| 207 | #define UDC_EP0OUT_MAX_PKT_SIZE 64 | 207 | #define UDC_EP0OUT_MAX_PKT_SIZE 64 |
| @@ -351,7 +351,7 @@ struct pch_udc_dev { | |||
| 351 | struct pci_pool *data_requests; | 351 | struct pci_pool *data_requests; |
| 352 | struct pci_pool *stp_requests; | 352 | struct pci_pool *stp_requests; |
| 353 | dma_addr_t dma_addr; | 353 | dma_addr_t dma_addr; |
| 354 | unsigned long ep0out_buf[64]; | 354 | void *ep0out_buf; |
| 355 | struct usb_ctrlrequest setup_data; | 355 | struct usb_ctrlrequest setup_data; |
| 356 | unsigned long phys_addr; | 356 | unsigned long phys_addr; |
| 357 | void __iomem *base_addr; | 357 | void __iomem *base_addr; |
| @@ -361,6 +361,8 @@ struct pch_udc_dev { | |||
| 361 | 361 | ||
| 362 | #define PCH_UDC_PCI_BAR 1 | 362 | #define PCH_UDC_PCI_BAR 1 |
| 363 | #define PCI_DEVICE_ID_INTEL_EG20T_UDC 0x8808 | 363 | #define PCI_DEVICE_ID_INTEL_EG20T_UDC 0x8808 |
| 364 | #define PCI_VENDOR_ID_ROHM 0x10DB | ||
| 365 | #define PCI_DEVICE_ID_ML7213_IOH_UDC 0x801D | ||
| 364 | 366 | ||
| 365 | static const char ep0_string[] = "ep0in"; | 367 | static const char ep0_string[] = "ep0in"; |
| 366 | static DEFINE_SPINLOCK(udc_stall_spinlock); /* stall spin lock */ | 368 | static DEFINE_SPINLOCK(udc_stall_spinlock); /* stall spin lock */ |
| @@ -1219,11 +1221,11 @@ static void complete_req(struct pch_udc_ep *ep, struct pch_udc_request *req, | |||
| 1219 | dev = ep->dev; | 1221 | dev = ep->dev; |
| 1220 | if (req->dma_mapped) { | 1222 | if (req->dma_mapped) { |
| 1221 | if (ep->in) | 1223 | if (ep->in) |
| 1222 | pci_unmap_single(dev->pdev, req->req.dma, | 1224 | dma_unmap_single(&dev->pdev->dev, req->req.dma, |
| 1223 | req->req.length, PCI_DMA_TODEVICE); | 1225 | req->req.length, DMA_TO_DEVICE); |
| 1224 | else | 1226 | else |
| 1225 | pci_unmap_single(dev->pdev, req->req.dma, | 1227 | dma_unmap_single(&dev->pdev->dev, req->req.dma, |
| 1226 | req->req.length, PCI_DMA_FROMDEVICE); | 1228 | req->req.length, DMA_FROM_DEVICE); |
| 1227 | req->dma_mapped = 0; | 1229 | req->dma_mapped = 0; |
| 1228 | req->req.dma = DMA_ADDR_INVALID; | 1230 | req->req.dma = DMA_ADDR_INVALID; |
| 1229 | } | 1231 | } |
| @@ -1414,7 +1416,6 @@ static void pch_udc_start_rxrequest(struct pch_udc_ep *ep, | |||
| 1414 | 1416 | ||
| 1415 | pch_udc_clear_dma(ep->dev, DMA_DIR_RX); | 1417 | pch_udc_clear_dma(ep->dev, DMA_DIR_RX); |
| 1416 | td_data = req->td_data; | 1418 | td_data = req->td_data; |
| 1417 | ep->td_data = req->td_data; | ||
| 1418 | /* Set the status bits for all descriptors */ | 1419 | /* Set the status bits for all descriptors */ |
| 1419 | while (1) { | 1420 | while (1) { |
| 1420 | td_data->status = (td_data->status & ~PCH_UDC_BUFF_STS) | | 1421 | td_data->status = (td_data->status & ~PCH_UDC_BUFF_STS) | |
| @@ -1613,15 +1614,19 @@ static int pch_udc_pcd_queue(struct usb_ep *usbep, struct usb_request *usbreq, | |||
| 1613 | if (usbreq->length && | 1614 | if (usbreq->length && |
| 1614 | ((usbreq->dma == DMA_ADDR_INVALID) || !usbreq->dma)) { | 1615 | ((usbreq->dma == DMA_ADDR_INVALID) || !usbreq->dma)) { |
| 1615 | if (ep->in) | 1616 | if (ep->in) |
| 1616 | usbreq->dma = pci_map_single(dev->pdev, usbreq->buf, | 1617 | usbreq->dma = dma_map_single(&dev->pdev->dev, |
| 1617 | usbreq->length, PCI_DMA_TODEVICE); | 1618 | usbreq->buf, |
| 1619 | usbreq->length, | ||
| 1620 | DMA_TO_DEVICE); | ||
| 1618 | else | 1621 | else |
| 1619 | usbreq->dma = pci_map_single(dev->pdev, usbreq->buf, | 1622 | usbreq->dma = dma_map_single(&dev->pdev->dev, |
| 1620 | usbreq->length, PCI_DMA_FROMDEVICE); | 1623 | usbreq->buf, |
| 1624 | usbreq->length, | ||
| 1625 | DMA_FROM_DEVICE); | ||
| 1621 | req->dma_mapped = 1; | 1626 | req->dma_mapped = 1; |
| 1622 | } | 1627 | } |
| 1623 | if (usbreq->length > 0) { | 1628 | if (usbreq->length > 0) { |
| 1624 | retval = prepare_dma(ep, req, gfp); | 1629 | retval = prepare_dma(ep, req, GFP_ATOMIC); |
| 1625 | if (retval) | 1630 | if (retval) |
| 1626 | goto probe_end; | 1631 | goto probe_end; |
| 1627 | } | 1632 | } |
| @@ -1646,7 +1651,6 @@ static int pch_udc_pcd_queue(struct usb_ep *usbep, struct usb_request *usbreq, | |||
| 1646 | pch_udc_wait_ep_stall(ep); | 1651 | pch_udc_wait_ep_stall(ep); |
| 1647 | pch_udc_ep_clear_nak(ep); | 1652 | pch_udc_ep_clear_nak(ep); |
| 1648 | pch_udc_enable_ep_interrupts(ep->dev, (1 << ep->num)); | 1653 | pch_udc_enable_ep_interrupts(ep->dev, (1 << ep->num)); |
| 1649 | pch_udc_set_dma(dev, DMA_DIR_TX); | ||
| 1650 | } | 1654 | } |
| 1651 | } | 1655 | } |
| 1652 | /* Now add this request to the ep's pending requests */ | 1656 | /* Now add this request to the ep's pending requests */ |
| @@ -1926,6 +1930,7 @@ static void pch_udc_complete_receiver(struct pch_udc_ep *ep) | |||
| 1926 | PCH_UDC_BS_DMA_DONE) | 1930 | PCH_UDC_BS_DMA_DONE) |
| 1927 | return; | 1931 | return; |
| 1928 | pch_udc_clear_dma(ep->dev, DMA_DIR_RX); | 1932 | pch_udc_clear_dma(ep->dev, DMA_DIR_RX); |
| 1933 | pch_udc_ep_set_ddptr(ep, 0); | ||
| 1929 | if ((req->td_data_last->status & PCH_UDC_RXTX_STS) != | 1934 | if ((req->td_data_last->status & PCH_UDC_RXTX_STS) != |
| 1930 | PCH_UDC_RTS_SUCC) { | 1935 | PCH_UDC_RTS_SUCC) { |
| 1931 | dev_err(&dev->pdev->dev, "Invalid RXTX status (0x%08x) " | 1936 | dev_err(&dev->pdev->dev, "Invalid RXTX status (0x%08x) " |
| @@ -1963,7 +1968,7 @@ static void pch_udc_svc_data_in(struct pch_udc_dev *dev, int ep_num) | |||
| 1963 | u32 epsts; | 1968 | u32 epsts; |
| 1964 | struct pch_udc_ep *ep; | 1969 | struct pch_udc_ep *ep; |
| 1965 | 1970 | ||
| 1966 | ep = &dev->ep[2*ep_num]; | 1971 | ep = &dev->ep[UDC_EPIN_IDX(ep_num)]; |
| 1967 | epsts = ep->epsts; | 1972 | epsts = ep->epsts; |
| 1968 | ep->epsts = 0; | 1973 | ep->epsts = 0; |
| 1969 | 1974 | ||
| @@ -2008,7 +2013,7 @@ static void pch_udc_svc_data_out(struct pch_udc_dev *dev, int ep_num) | |||
| 2008 | struct pch_udc_ep *ep; | 2013 | struct pch_udc_ep *ep; |
| 2009 | struct pch_udc_request *req = NULL; | 2014 | struct pch_udc_request *req = NULL; |
| 2010 | 2015 | ||
| 2011 | ep = &dev->ep[2*ep_num + 1]; | 2016 | ep = &dev->ep[UDC_EPOUT_IDX(ep_num)]; |
| 2012 | epsts = ep->epsts; | 2017 | epsts = ep->epsts; |
| 2013 | ep->epsts = 0; | 2018 | ep->epsts = 0; |
| 2014 | 2019 | ||
| @@ -2025,10 +2030,11 @@ static void pch_udc_svc_data_out(struct pch_udc_dev *dev, int ep_num) | |||
| 2025 | } | 2030 | } |
| 2026 | if (epsts & UDC_EPSTS_HE) | 2031 | if (epsts & UDC_EPSTS_HE) |
| 2027 | return; | 2032 | return; |
| 2028 | if (epsts & UDC_EPSTS_RSS) | 2033 | if (epsts & UDC_EPSTS_RSS) { |
| 2029 | pch_udc_ep_set_stall(ep); | 2034 | pch_udc_ep_set_stall(ep); |
| 2030 | pch_udc_enable_ep_interrupts(ep->dev, | 2035 | pch_udc_enable_ep_interrupts(ep->dev, |
| 2031 | PCH_UDC_EPINT(ep->in, ep->num)); | 2036 | PCH_UDC_EPINT(ep->in, ep->num)); |
| 2037 | } | ||
| 2032 | if (epsts & UDC_EPSTS_RCS) { | 2038 | if (epsts & UDC_EPSTS_RCS) { |
| 2033 | if (!dev->prot_stall) { | 2039 | if (!dev->prot_stall) { |
| 2034 | pch_udc_ep_clear_stall(ep); | 2040 | pch_udc_ep_clear_stall(ep); |
| @@ -2060,8 +2066,10 @@ static void pch_udc_svc_control_in(struct pch_udc_dev *dev) | |||
| 2060 | { | 2066 | { |
| 2061 | u32 epsts; | 2067 | u32 epsts; |
| 2062 | struct pch_udc_ep *ep; | 2068 | struct pch_udc_ep *ep; |
| 2069 | struct pch_udc_ep *ep_out; | ||
| 2063 | 2070 | ||
| 2064 | ep = &dev->ep[UDC_EP0IN_IDX]; | 2071 | ep = &dev->ep[UDC_EP0IN_IDX]; |
| 2072 | ep_out = &dev->ep[UDC_EP0OUT_IDX]; | ||
| 2065 | epsts = ep->epsts; | 2073 | epsts = ep->epsts; |
| 2066 | ep->epsts = 0; | 2074 | ep->epsts = 0; |
| 2067 | 2075 | ||
| @@ -2073,8 +2081,16 @@ static void pch_udc_svc_control_in(struct pch_udc_dev *dev) | |||
| 2073 | return; | 2081 | return; |
| 2074 | if (epsts & UDC_EPSTS_HE) | 2082 | if (epsts & UDC_EPSTS_HE) |
| 2075 | return; | 2083 | return; |
| 2076 | if ((epsts & UDC_EPSTS_TDC) && (!dev->stall)) | 2084 | if ((epsts & UDC_EPSTS_TDC) && (!dev->stall)) { |
| 2077 | pch_udc_complete_transfer(ep); | 2085 | pch_udc_complete_transfer(ep); |
| 2086 | pch_udc_clear_dma(dev, DMA_DIR_RX); | ||
| 2087 | ep_out->td_data->status = (ep_out->td_data->status & | ||
| 2088 | ~PCH_UDC_BUFF_STS) | | ||
| 2089 | PCH_UDC_BS_HST_RDY; | ||
| 2090 | pch_udc_ep_clear_nak(ep_out); | ||
| 2091 | pch_udc_set_dma(dev, DMA_DIR_RX); | ||
| 2092 | pch_udc_ep_set_rrdy(ep_out); | ||
| 2093 | } | ||
| 2078 | /* On IN interrupt, provide data if we have any */ | 2094 | /* On IN interrupt, provide data if we have any */ |
| 2079 | if ((epsts & UDC_EPSTS_IN) && !(epsts & UDC_EPSTS_TDC) && | 2095 | if ((epsts & UDC_EPSTS_IN) && !(epsts & UDC_EPSTS_TDC) && |
| 2080 | !(epsts & UDC_EPSTS_TXEMPTY)) | 2096 | !(epsts & UDC_EPSTS_TXEMPTY)) |
| @@ -2102,11 +2118,9 @@ static void pch_udc_svc_control_out(struct pch_udc_dev *dev) | |||
| 2102 | dev->stall = 0; | 2118 | dev->stall = 0; |
| 2103 | dev->ep[UDC_EP0IN_IDX].halted = 0; | 2119 | dev->ep[UDC_EP0IN_IDX].halted = 0; |
| 2104 | dev->ep[UDC_EP0OUT_IDX].halted = 0; | 2120 | dev->ep[UDC_EP0OUT_IDX].halted = 0; |
| 2105 | /* In data not ready */ | ||
| 2106 | pch_udc_ep_set_nak(&(dev->ep[UDC_EP0IN_IDX])); | ||
| 2107 | dev->setup_data = ep->td_stp->request; | 2121 | dev->setup_data = ep->td_stp->request; |
| 2108 | pch_udc_init_setup_buff(ep->td_stp); | 2122 | pch_udc_init_setup_buff(ep->td_stp); |
| 2109 | pch_udc_clear_dma(dev, DMA_DIR_TX); | 2123 | pch_udc_clear_dma(dev, DMA_DIR_RX); |
| 2110 | pch_udc_ep_fifo_flush(&(dev->ep[UDC_EP0IN_IDX]), | 2124 | pch_udc_ep_fifo_flush(&(dev->ep[UDC_EP0IN_IDX]), |
| 2111 | dev->ep[UDC_EP0IN_IDX].in); | 2125 | dev->ep[UDC_EP0IN_IDX].in); |
| 2112 | if ((dev->setup_data.bRequestType & USB_DIR_IN)) | 2126 | if ((dev->setup_data.bRequestType & USB_DIR_IN)) |
| @@ -2122,14 +2136,23 @@ static void pch_udc_svc_control_out(struct pch_udc_dev *dev) | |||
| 2122 | setup_supported = dev->driver->setup(&dev->gadget, | 2136 | setup_supported = dev->driver->setup(&dev->gadget, |
| 2123 | &dev->setup_data); | 2137 | &dev->setup_data); |
| 2124 | spin_lock(&dev->lock); | 2138 | spin_lock(&dev->lock); |
| 2139 | |||
| 2140 | if (dev->setup_data.bRequestType & USB_DIR_IN) { | ||
| 2141 | ep->td_data->status = (ep->td_data->status & | ||
| 2142 | ~PCH_UDC_BUFF_STS) | | ||
| 2143 | PCH_UDC_BS_HST_RDY; | ||
| 2144 | pch_udc_ep_set_ddptr(ep, ep->td_data_phys); | ||
| 2145 | } | ||
| 2125 | /* ep0 in returns data on IN phase */ | 2146 | /* ep0 in returns data on IN phase */ |
| 2126 | if (setup_supported >= 0 && setup_supported < | 2147 | if (setup_supported >= 0 && setup_supported < |
| 2127 | UDC_EP0IN_MAX_PKT_SIZE) { | 2148 | UDC_EP0IN_MAX_PKT_SIZE) { |
| 2128 | pch_udc_ep_clear_nak(&(dev->ep[UDC_EP0IN_IDX])); | 2149 | pch_udc_ep_clear_nak(&(dev->ep[UDC_EP0IN_IDX])); |
| 2129 | /* Gadget would have queued a request when | 2150 | /* Gadget would have queued a request when |
| 2130 | * we called the setup */ | 2151 | * we called the setup */ |
| 2131 | pch_udc_set_dma(dev, DMA_DIR_RX); | 2152 | if (!(dev->setup_data.bRequestType & USB_DIR_IN)) { |
| 2132 | pch_udc_ep_clear_nak(ep); | 2153 | pch_udc_set_dma(dev, DMA_DIR_RX); |
| 2154 | pch_udc_ep_clear_nak(ep); | ||
| 2155 | } | ||
| 2133 | } else if (setup_supported < 0) { | 2156 | } else if (setup_supported < 0) { |
| 2134 | /* if unsupported request, then stall */ | 2157 | /* if unsupported request, then stall */ |
| 2135 | pch_udc_ep_set_stall(&(dev->ep[UDC_EP0IN_IDX])); | 2158 | pch_udc_ep_set_stall(&(dev->ep[UDC_EP0IN_IDX])); |
| @@ -2142,22 +2165,13 @@ static void pch_udc_svc_control_out(struct pch_udc_dev *dev) | |||
| 2142 | } | 2165 | } |
| 2143 | } else if ((((stat & UDC_EPSTS_OUT_MASK) >> UDC_EPSTS_OUT_SHIFT) == | 2166 | } else if ((((stat & UDC_EPSTS_OUT_MASK) >> UDC_EPSTS_OUT_SHIFT) == |
| 2144 | UDC_EPSTS_OUT_DATA) && !dev->stall) { | 2167 | UDC_EPSTS_OUT_DATA) && !dev->stall) { |
| 2145 | if (list_empty(&ep->queue)) { | 2168 | pch_udc_clear_dma(dev, DMA_DIR_RX); |
| 2146 | dev_err(&dev->pdev->dev, "%s: No request\n", __func__); | 2169 | pch_udc_ep_set_ddptr(ep, 0); |
| 2147 | ep->td_data->status = (ep->td_data->status & | 2170 | if (!list_empty(&ep->queue)) { |
| 2148 | ~PCH_UDC_BUFF_STS) | | ||
| 2149 | PCH_UDC_BS_HST_RDY; | ||
| 2150 | pch_udc_set_dma(dev, DMA_DIR_RX); | ||
| 2151 | } else { | ||
| 2152 | /* control write */ | ||
| 2153 | /* next function will pickuo an clear the status */ | ||
| 2154 | ep->epsts = stat; | 2171 | ep->epsts = stat; |
| 2155 | 2172 | pch_udc_svc_data_out(dev, PCH_UDC_EP0); | |
| 2156 | pch_udc_svc_data_out(dev, 0); | ||
| 2157 | /* re-program desc. pointer for possible ZLPs */ | ||
| 2158 | pch_udc_ep_set_ddptr(ep, ep->td_data_phys); | ||
| 2159 | pch_udc_set_dma(dev, DMA_DIR_RX); | ||
| 2160 | } | 2173 | } |
| 2174 | pch_udc_set_dma(dev, DMA_DIR_RX); | ||
| 2161 | } | 2175 | } |
| 2162 | pch_udc_ep_set_rrdy(ep); | 2176 | pch_udc_ep_set_rrdy(ep); |
| 2163 | } | 2177 | } |
| @@ -2174,7 +2188,7 @@ static void pch_udc_postsvc_epinters(struct pch_udc_dev *dev, int ep_num) | |||
| 2174 | struct pch_udc_ep *ep; | 2188 | struct pch_udc_ep *ep; |
| 2175 | struct pch_udc_request *req; | 2189 | struct pch_udc_request *req; |
| 2176 | 2190 | ||
| 2177 | ep = &dev->ep[2*ep_num]; | 2191 | ep = &dev->ep[UDC_EPIN_IDX(ep_num)]; |
| 2178 | if (!list_empty(&ep->queue)) { | 2192 | if (!list_empty(&ep->queue)) { |
| 2179 | req = list_entry(ep->queue.next, struct pch_udc_request, queue); | 2193 | req = list_entry(ep->queue.next, struct pch_udc_request, queue); |
| 2180 | pch_udc_enable_ep_interrupts(ep->dev, | 2194 | pch_udc_enable_ep_interrupts(ep->dev, |
| @@ -2196,13 +2210,13 @@ static void pch_udc_read_all_epstatus(struct pch_udc_dev *dev, u32 ep_intr) | |||
| 2196 | for (i = 0; i < PCH_UDC_USED_EP_NUM; i++) { | 2210 | for (i = 0; i < PCH_UDC_USED_EP_NUM; i++) { |
| 2197 | /* IN */ | 2211 | /* IN */ |
| 2198 | if (ep_intr & (0x1 << i)) { | 2212 | if (ep_intr & (0x1 << i)) { |
| 2199 | ep = &dev->ep[2*i]; | 2213 | ep = &dev->ep[UDC_EPIN_IDX(i)]; |
| 2200 | ep->epsts = pch_udc_read_ep_status(ep); | 2214 | ep->epsts = pch_udc_read_ep_status(ep); |
| 2201 | pch_udc_clear_ep_status(ep, ep->epsts); | 2215 | pch_udc_clear_ep_status(ep, ep->epsts); |
| 2202 | } | 2216 | } |
| 2203 | /* OUT */ | 2217 | /* OUT */ |
| 2204 | if (ep_intr & (0x10000 << i)) { | 2218 | if (ep_intr & (0x10000 << i)) { |
| 2205 | ep = &dev->ep[2*i+1]; | 2219 | ep = &dev->ep[UDC_EPOUT_IDX(i)]; |
| 2206 | ep->epsts = pch_udc_read_ep_status(ep); | 2220 | ep->epsts = pch_udc_read_ep_status(ep); |
| 2207 | pch_udc_clear_ep_status(ep, ep->epsts); | 2221 | pch_udc_clear_ep_status(ep, ep->epsts); |
| 2208 | } | 2222 | } |
| @@ -2563,9 +2577,6 @@ static void pch_udc_pcd_reinit(struct pch_udc_dev *dev) | |||
| 2563 | dev->ep[UDC_EP0IN_IDX].ep.maxpacket = UDC_EP0IN_MAX_PKT_SIZE; | 2577 | dev->ep[UDC_EP0IN_IDX].ep.maxpacket = UDC_EP0IN_MAX_PKT_SIZE; |
| 2564 | dev->ep[UDC_EP0OUT_IDX].ep.maxpacket = UDC_EP0OUT_MAX_PKT_SIZE; | 2578 | dev->ep[UDC_EP0OUT_IDX].ep.maxpacket = UDC_EP0OUT_MAX_PKT_SIZE; |
| 2565 | 2579 | ||
| 2566 | dev->dma_addr = pci_map_single(dev->pdev, dev->ep0out_buf, 256, | ||
| 2567 | PCI_DMA_FROMDEVICE); | ||
| 2568 | |||
| 2569 | /* remove ep0 in and out from the list. They have own pointer */ | 2580 | /* remove ep0 in and out from the list. They have own pointer */ |
| 2570 | list_del_init(&dev->ep[UDC_EP0IN_IDX].ep.ep_list); | 2581 | list_del_init(&dev->ep[UDC_EP0IN_IDX].ep.ep_list); |
| 2571 | list_del_init(&dev->ep[UDC_EP0OUT_IDX].ep.ep_list); | 2582 | list_del_init(&dev->ep[UDC_EP0OUT_IDX].ep.ep_list); |
| @@ -2637,6 +2648,13 @@ static int init_dma_pools(struct pch_udc_dev *dev) | |||
| 2637 | dev->ep[UDC_EP0IN_IDX].td_stp_phys = 0; | 2648 | dev->ep[UDC_EP0IN_IDX].td_stp_phys = 0; |
| 2638 | dev->ep[UDC_EP0IN_IDX].td_data = NULL; | 2649 | dev->ep[UDC_EP0IN_IDX].td_data = NULL; |
| 2639 | dev->ep[UDC_EP0IN_IDX].td_data_phys = 0; | 2650 | dev->ep[UDC_EP0IN_IDX].td_data_phys = 0; |
| 2651 | |||
| 2652 | dev->ep0out_buf = kzalloc(UDC_EP0OUT_BUFF_SIZE * 4, GFP_KERNEL); | ||
| 2653 | if (!dev->ep0out_buf) | ||
| 2654 | return -ENOMEM; | ||
| 2655 | dev->dma_addr = dma_map_single(&dev->pdev->dev, dev->ep0out_buf, | ||
| 2656 | UDC_EP0OUT_BUFF_SIZE * 4, | ||
| 2657 | DMA_FROM_DEVICE); | ||
| 2640 | return 0; | 2658 | return 0; |
| 2641 | } | 2659 | } |
| 2642 | 2660 | ||
| @@ -2700,7 +2718,8 @@ int usb_gadget_unregister_driver(struct usb_gadget_driver *driver) | |||
| 2700 | 2718 | ||
| 2701 | pch_udc_disable_interrupts(dev, UDC_DEVINT_MSK); | 2719 | pch_udc_disable_interrupts(dev, UDC_DEVINT_MSK); |
| 2702 | 2720 | ||
| 2703 | /* Assues that there are no pending requets with this driver */ | 2721 | /* Assures that there are no pending requests with this driver */ |
| 2722 | driver->disconnect(&dev->gadget); | ||
| 2704 | driver->unbind(&dev->gadget); | 2723 | driver->unbind(&dev->gadget); |
| 2705 | dev->gadget.dev.driver = NULL; | 2724 | dev->gadget.dev.driver = NULL; |
| 2706 | dev->driver = NULL; | 2725 | dev->driver = NULL; |
| @@ -2750,6 +2769,11 @@ static void pch_udc_remove(struct pci_dev *pdev) | |||
| 2750 | pci_pool_destroy(dev->stp_requests); | 2769 | pci_pool_destroy(dev->stp_requests); |
| 2751 | } | 2770 | } |
| 2752 | 2771 | ||
| 2772 | if (dev->dma_addr) | ||
| 2773 | dma_unmap_single(&dev->pdev->dev, dev->dma_addr, | ||
| 2774 | UDC_EP0OUT_BUFF_SIZE * 4, DMA_FROM_DEVICE); | ||
| 2775 | kfree(dev->ep0out_buf); | ||
| 2776 | |||
| 2753 | pch_udc_exit(dev); | 2777 | pch_udc_exit(dev); |
| 2754 | 2778 | ||
| 2755 | if (dev->irq_registered) | 2779 | if (dev->irq_registered) |
| @@ -2792,11 +2816,7 @@ static int pch_udc_resume(struct pci_dev *pdev) | |||
| 2792 | int ret; | 2816 | int ret; |
| 2793 | 2817 | ||
| 2794 | pci_set_power_state(pdev, PCI_D0); | 2818 | pci_set_power_state(pdev, PCI_D0); |
| 2795 | ret = pci_restore_state(pdev); | 2819 | pci_restore_state(pdev); |
| 2796 | if (ret) { | ||
| 2797 | dev_err(&pdev->dev, "%s: pci_restore_state failed\n", __func__); | ||
| 2798 | return ret; | ||
| 2799 | } | ||
| 2800 | ret = pci_enable_device(pdev); | 2820 | ret = pci_enable_device(pdev); |
| 2801 | if (ret) { | 2821 | if (ret) { |
| 2802 | dev_err(&pdev->dev, "%s: pci_enable_device failed\n", __func__); | 2822 | dev_err(&pdev->dev, "%s: pci_enable_device failed\n", __func__); |
| @@ -2914,6 +2934,11 @@ static DEFINE_PCI_DEVICE_TABLE(pch_udc_pcidev_id) = { | |||
| 2914 | .class = (PCI_CLASS_SERIAL_USB << 8) | 0xfe, | 2934 | .class = (PCI_CLASS_SERIAL_USB << 8) | 0xfe, |
| 2915 | .class_mask = 0xffffffff, | 2935 | .class_mask = 0xffffffff, |
| 2916 | }, | 2936 | }, |
| 2937 | { | ||
| 2938 | PCI_DEVICE(PCI_VENDOR_ID_ROHM, PCI_DEVICE_ID_ML7213_IOH_UDC), | ||
| 2939 | .class = (PCI_CLASS_SERIAL_USB << 8) | 0xfe, | ||
| 2940 | .class_mask = 0xffffffff, | ||
| 2941 | }, | ||
| 2917 | { 0 }, | 2942 | { 0 }, |
| 2918 | }; | 2943 | }; |
| 2919 | 2944 | ||
diff --git a/drivers/usb/gadget/printer.c b/drivers/usb/gadget/printer.c index 2fc8636316c..12ff6cffedc 100644 --- a/drivers/usb/gadget/printer.c +++ b/drivers/usb/gadget/printer.c | |||
| @@ -131,31 +131,31 @@ static struct printer_dev usb_printer_gadget; | |||
| 131 | * parameters are in UTF-8 (superset of ASCII's 7 bit characters). | 131 | * parameters are in UTF-8 (superset of ASCII's 7 bit characters). |
| 132 | */ | 132 | */ |
| 133 | 133 | ||
| 134 | static ushort __initdata idVendor; | 134 | static ushort idVendor; |
| 135 | module_param(idVendor, ushort, S_IRUGO); | 135 | module_param(idVendor, ushort, S_IRUGO); |
| 136 | MODULE_PARM_DESC(idVendor, "USB Vendor ID"); | 136 | MODULE_PARM_DESC(idVendor, "USB Vendor ID"); |
| 137 | 137 | ||
| 138 | static ushort __initdata idProduct; | 138 | static ushort idProduct; |
| 139 | module_param(idProduct, ushort, S_IRUGO); | 139 | module_param(idProduct, ushort, S_IRUGO); |
| 140 | MODULE_PARM_DESC(idProduct, "USB Product ID"); | 140 | MODULE_PARM_DESC(idProduct, "USB Product ID"); |
| 141 | 141 | ||
| 142 | static ushort __initdata bcdDevice; | 142 | static ushort bcdDevice; |
| 143 | module_param(bcdDevice, ushort, S_IRUGO); | 143 | module_param(bcdDevice, ushort, S_IRUGO); |
| 144 | MODULE_PARM_DESC(bcdDevice, "USB Device version (BCD)"); | 144 | MODULE_PARM_DESC(bcdDevice, "USB Device version (BCD)"); |
| 145 | 145 | ||
| 146 | static char *__initdata iManufacturer; | 146 | static char *iManufacturer; |
| 147 | module_param(iManufacturer, charp, S_IRUGO); | 147 | module_param(iManufacturer, charp, S_IRUGO); |
| 148 | MODULE_PARM_DESC(iManufacturer, "USB Manufacturer string"); | 148 | MODULE_PARM_DESC(iManufacturer, "USB Manufacturer string"); |
| 149 | 149 | ||
| 150 | static char *__initdata iProduct; | 150 | static char *iProduct; |
| 151 | module_param(iProduct, charp, S_IRUGO); | 151 | module_param(iProduct, charp, S_IRUGO); |
| 152 | MODULE_PARM_DESC(iProduct, "USB Product string"); | 152 | MODULE_PARM_DESC(iProduct, "USB Product string"); |
| 153 | 153 | ||
| 154 | static char *__initdata iSerialNum; | 154 | static char *iSerialNum; |
| 155 | module_param(iSerialNum, charp, S_IRUGO); | 155 | module_param(iSerialNum, charp, S_IRUGO); |
| 156 | MODULE_PARM_DESC(iSerialNum, "1"); | 156 | MODULE_PARM_DESC(iSerialNum, "1"); |
| 157 | 157 | ||
| 158 | static char *__initdata iPNPstring; | 158 | static char *iPNPstring; |
| 159 | module_param(iPNPstring, charp, S_IRUGO); | 159 | module_param(iPNPstring, charp, S_IRUGO); |
| 160 | MODULE_PARM_DESC(iPNPstring, "MFG:linux;MDL:g_printer;CLS:PRINTER;SN:1;"); | 160 | MODULE_PARM_DESC(iPNPstring, "MFG:linux;MDL:g_printer;CLS:PRINTER;SN:1;"); |
| 161 | 161 | ||
| @@ -1596,13 +1596,12 @@ cleanup(void) | |||
| 1596 | int status; | 1596 | int status; |
| 1597 | 1597 | ||
| 1598 | mutex_lock(&usb_printer_gadget.lock_printer_io); | 1598 | mutex_lock(&usb_printer_gadget.lock_printer_io); |
| 1599 | class_destroy(usb_gadget_class); | ||
| 1600 | unregister_chrdev_region(g_printer_devno, 2); | ||
| 1601 | |||
| 1602 | status = usb_gadget_unregister_driver(&printer_driver); | 1599 | status = usb_gadget_unregister_driver(&printer_driver); |
| 1603 | if (status) | 1600 | if (status) |
| 1604 | ERROR(dev, "usb_gadget_unregister_driver %x\n", status); | 1601 | ERROR(dev, "usb_gadget_unregister_driver %x\n", status); |
| 1605 | 1602 | ||
| 1603 | unregister_chrdev_region(g_printer_devno, 2); | ||
| 1604 | class_destroy(usb_gadget_class); | ||
| 1606 | mutex_unlock(&usb_printer_gadget.lock_printer_io); | 1605 | mutex_unlock(&usb_printer_gadget.lock_printer_io); |
| 1607 | } | 1606 | } |
| 1608 | module_exit(cleanup); | 1607 | module_exit(cleanup); |
diff --git a/drivers/usb/host/ehci-fsl.c b/drivers/usb/host/ehci-fsl.c index 86e42892016..5c761df7fa8 100644 --- a/drivers/usb/host/ehci-fsl.c +++ b/drivers/usb/host/ehci-fsl.c | |||
| @@ -52,7 +52,6 @@ static int usb_hcd_fsl_probe(const struct hc_driver *driver, | |||
| 52 | struct resource *res; | 52 | struct resource *res; |
| 53 | int irq; | 53 | int irq; |
| 54 | int retval; | 54 | int retval; |
| 55 | unsigned int temp; | ||
| 56 | 55 | ||
| 57 | pr_debug("initializing FSL-SOC USB Controller\n"); | 56 | pr_debug("initializing FSL-SOC USB Controller\n"); |
| 58 | 57 | ||
| @@ -126,18 +125,6 @@ static int usb_hcd_fsl_probe(const struct hc_driver *driver, | |||
| 126 | goto err3; | 125 | goto err3; |
| 127 | } | 126 | } |
| 128 | 127 | ||
| 129 | /* | ||
| 130 | * Check if it is MPC5121 SoC, otherwise set pdata->have_sysif_regs | ||
| 131 | * flag for 83xx or 8536 system interface registers. | ||
| 132 | */ | ||
| 133 | if (pdata->big_endian_mmio) | ||
| 134 | temp = in_be32(hcd->regs + FSL_SOC_USB_ID); | ||
| 135 | else | ||
| 136 | temp = in_le32(hcd->regs + FSL_SOC_USB_ID); | ||
| 137 | |||
| 138 | if ((temp & ID_MSK) != (~((temp & NID_MSK) >> 8) & ID_MSK)) | ||
| 139 | pdata->have_sysif_regs = 1; | ||
| 140 | |||
| 141 | /* Enable USB controller, 83xx or 8536 */ | 128 | /* Enable USB controller, 83xx or 8536 */ |
| 142 | if (pdata->have_sysif_regs) | 129 | if (pdata->have_sysif_regs) |
| 143 | setbits32(hcd->regs + FSL_SOC_USB_CTRL, 0x4); | 130 | setbits32(hcd->regs + FSL_SOC_USB_CTRL, 0x4); |
diff --git a/drivers/usb/host/ehci-fsl.h b/drivers/usb/host/ehci-fsl.h index 2c835379522..3fabed33d94 100644 --- a/drivers/usb/host/ehci-fsl.h +++ b/drivers/usb/host/ehci-fsl.h | |||
| @@ -19,9 +19,6 @@ | |||
| 19 | #define _EHCI_FSL_H | 19 | #define _EHCI_FSL_H |
| 20 | 20 | ||
| 21 | /* offsets for the non-ehci registers in the FSL SOC USB controller */ | 21 | /* offsets for the non-ehci registers in the FSL SOC USB controller */ |
| 22 | #define FSL_SOC_USB_ID 0x0 | ||
| 23 | #define ID_MSK 0x3f | ||
| 24 | #define NID_MSK 0x3f00 | ||
| 25 | #define FSL_SOC_USB_ULPIVP 0x170 | 22 | #define FSL_SOC_USB_ULPIVP 0x170 |
| 26 | #define FSL_SOC_USB_PORTSC1 0x184 | 23 | #define FSL_SOC_USB_PORTSC1 0x184 |
| 27 | #define PORT_PTS_MSK (3<<30) | 24 | #define PORT_PTS_MSK (3<<30) |
diff --git a/drivers/usb/host/ehci-hcd.c b/drivers/usb/host/ehci-hcd.c index 6fee3cd58ef..74dcf49bd01 100644 --- a/drivers/usb/host/ehci-hcd.c +++ b/drivers/usb/host/ehci-hcd.c | |||
| @@ -572,6 +572,8 @@ static int ehci_init(struct usb_hcd *hcd) | |||
| 572 | ehci->iaa_watchdog.function = ehci_iaa_watchdog; | 572 | ehci->iaa_watchdog.function = ehci_iaa_watchdog; |
| 573 | ehci->iaa_watchdog.data = (unsigned long) ehci; | 573 | ehci->iaa_watchdog.data = (unsigned long) ehci; |
| 574 | 574 | ||
| 575 | hcc_params = ehci_readl(ehci, &ehci->caps->hcc_params); | ||
| 576 | |||
| 575 | /* | 577 | /* |
| 576 | * hw default: 1K periodic list heads, one per frame. | 578 | * hw default: 1K periodic list heads, one per frame. |
| 577 | * periodic_size can shrink by USBCMD update if hcc_params allows. | 579 | * periodic_size can shrink by USBCMD update if hcc_params allows. |
| @@ -579,11 +581,20 @@ static int ehci_init(struct usb_hcd *hcd) | |||
| 579 | ehci->periodic_size = DEFAULT_I_TDPS; | 581 | ehci->periodic_size = DEFAULT_I_TDPS; |
| 580 | INIT_LIST_HEAD(&ehci->cached_itd_list); | 582 | INIT_LIST_HEAD(&ehci->cached_itd_list); |
| 581 | INIT_LIST_HEAD(&ehci->cached_sitd_list); | 583 | INIT_LIST_HEAD(&ehci->cached_sitd_list); |
| 584 | |||
| 585 | if (HCC_PGM_FRAMELISTLEN(hcc_params)) { | ||
| 586 | /* periodic schedule size can be smaller than default */ | ||
| 587 | switch (EHCI_TUNE_FLS) { | ||
| 588 | case 0: ehci->periodic_size = 1024; break; | ||
| 589 | case 1: ehci->periodic_size = 512; break; | ||
| 590 | case 2: ehci->periodic_size = 256; break; | ||
| 591 | default: BUG(); | ||
| 592 | } | ||
| 593 | } | ||
| 582 | if ((retval = ehci_mem_init(ehci, GFP_KERNEL)) < 0) | 594 | if ((retval = ehci_mem_init(ehci, GFP_KERNEL)) < 0) |
| 583 | return retval; | 595 | return retval; |
| 584 | 596 | ||
| 585 | /* controllers may cache some of the periodic schedule ... */ | 597 | /* controllers may cache some of the periodic schedule ... */ |
| 586 | hcc_params = ehci_readl(ehci, &ehci->caps->hcc_params); | ||
| 587 | if (HCC_ISOC_CACHE(hcc_params)) // full frame cache | 598 | if (HCC_ISOC_CACHE(hcc_params)) // full frame cache |
| 588 | ehci->i_thresh = 2 + 8; | 599 | ehci->i_thresh = 2 + 8; |
| 589 | else // N microframes cached | 600 | else // N microframes cached |
| @@ -637,12 +648,6 @@ static int ehci_init(struct usb_hcd *hcd) | |||
| 637 | /* periodic schedule size can be smaller than default */ | 648 | /* periodic schedule size can be smaller than default */ |
| 638 | temp &= ~(3 << 2); | 649 | temp &= ~(3 << 2); |
| 639 | temp |= (EHCI_TUNE_FLS << 2); | 650 | temp |= (EHCI_TUNE_FLS << 2); |
| 640 | switch (EHCI_TUNE_FLS) { | ||
| 641 | case 0: ehci->periodic_size = 1024; break; | ||
| 642 | case 1: ehci->periodic_size = 512; break; | ||
| 643 | case 2: ehci->periodic_size = 256; break; | ||
| 644 | default: BUG(); | ||
| 645 | } | ||
| 646 | } | 651 | } |
| 647 | if (HCC_LPM(hcc_params)) { | 652 | if (HCC_LPM(hcc_params)) { |
| 648 | /* support link power management EHCI 1.1 addendum */ | 653 | /* support link power management EHCI 1.1 addendum */ |
diff --git a/drivers/usb/host/ehci-mxc.c b/drivers/usb/host/ehci-mxc.c index fa59b26fc5b..c8e360d7d97 100644 --- a/drivers/usb/host/ehci-mxc.c +++ b/drivers/usb/host/ehci-mxc.c | |||
| @@ -21,10 +21,13 @@ | |||
| 21 | #include <linux/clk.h> | 21 | #include <linux/clk.h> |
| 22 | #include <linux/delay.h> | 22 | #include <linux/delay.h> |
| 23 | #include <linux/usb/otg.h> | 23 | #include <linux/usb/otg.h> |
| 24 | #include <linux/usb/ulpi.h> | ||
| 24 | #include <linux/slab.h> | 25 | #include <linux/slab.h> |
| 25 | 26 | ||
| 26 | #include <mach/mxc_ehci.h> | 27 | #include <mach/mxc_ehci.h> |
| 27 | 28 | ||
| 29 | #include <asm/mach-types.h> | ||
| 30 | |||
| 28 | #define ULPI_VIEWPORT_OFFSET 0x170 | 31 | #define ULPI_VIEWPORT_OFFSET 0x170 |
| 29 | 32 | ||
| 30 | struct ehci_mxc_priv { | 33 | struct ehci_mxc_priv { |
| @@ -114,6 +117,7 @@ static int ehci_mxc_drv_probe(struct platform_device *pdev) | |||
| 114 | struct usb_hcd *hcd; | 117 | struct usb_hcd *hcd; |
| 115 | struct resource *res; | 118 | struct resource *res; |
| 116 | int irq, ret; | 119 | int irq, ret; |
| 120 | unsigned int flags; | ||
| 117 | struct ehci_mxc_priv *priv; | 121 | struct ehci_mxc_priv *priv; |
| 118 | struct device *dev = &pdev->dev; | 122 | struct device *dev = &pdev->dev; |
| 119 | struct ehci_hcd *ehci; | 123 | struct ehci_hcd *ehci; |
| @@ -177,8 +181,8 @@ static int ehci_mxc_drv_probe(struct platform_device *pdev) | |||
| 177 | clk_enable(priv->ahbclk); | 181 | clk_enable(priv->ahbclk); |
| 178 | } | 182 | } |
| 179 | 183 | ||
| 180 | /* "dr" device has its own clock */ | 184 | /* "dr" device has its own clock on i.MX51 */ |
| 181 | if (pdev->id == 0) { | 185 | if (cpu_is_mx51() && (pdev->id == 0)) { |
| 182 | priv->phy1clk = clk_get(dev, "usb_phy1"); | 186 | priv->phy1clk = clk_get(dev, "usb_phy1"); |
| 183 | if (IS_ERR(priv->phy1clk)) { | 187 | if (IS_ERR(priv->phy1clk)) { |
| 184 | ret = PTR_ERR(priv->phy1clk); | 188 | ret = PTR_ERR(priv->phy1clk); |
| @@ -240,6 +244,23 @@ static int ehci_mxc_drv_probe(struct platform_device *pdev) | |||
| 240 | if (ret) | 244 | if (ret) |
| 241 | goto err_add; | 245 | goto err_add; |
| 242 | 246 | ||
| 247 | if (pdata->otg) { | ||
| 248 | /* | ||
| 249 | * efikamx and efikasb have some hardware bug which is | ||
| 250 | * preventing usb to work unless CHRGVBUS is set. | ||
| 251 | * It's in violation of USB specs | ||
| 252 | */ | ||
| 253 | if (machine_is_mx51_efikamx() || machine_is_mx51_efikasb()) { | ||
| 254 | flags = otg_io_read(pdata->otg, ULPI_OTG_CTRL); | ||
| 255 | flags |= ULPI_OTG_CTRL_CHRGVBUS; | ||
| 256 | ret = otg_io_write(pdata->otg, flags, ULPI_OTG_CTRL); | ||
| 257 | if (ret) { | ||
| 258 | dev_err(dev, "unable to set CHRVBUS\n"); | ||
| 259 | goto err_add; | ||
| 260 | } | ||
| 261 | } | ||
| 262 | } | ||
| 263 | |||
| 243 | return 0; | 264 | return 0; |
| 244 | 265 | ||
| 245 | err_add: | 266 | err_add: |
diff --git a/drivers/usb/host/ehci-pci.c b/drivers/usb/host/ehci-pci.c index 76179c39c0e..bed07d4aab0 100644 --- a/drivers/usb/host/ehci-pci.c +++ b/drivers/usb/host/ehci-pci.c | |||
| @@ -44,28 +44,35 @@ static int ehci_pci_reinit(struct ehci_hcd *ehci, struct pci_dev *pdev) | |||
| 44 | return 0; | 44 | return 0; |
| 45 | } | 45 | } |
| 46 | 46 | ||
| 47 | static int ehci_quirk_amd_SB800(struct ehci_hcd *ehci) | 47 | static int ehci_quirk_amd_hudson(struct ehci_hcd *ehci) |
| 48 | { | 48 | { |
| 49 | struct pci_dev *amd_smbus_dev; | 49 | struct pci_dev *amd_smbus_dev; |
| 50 | u8 rev = 0; | 50 | u8 rev = 0; |
| 51 | 51 | ||
| 52 | amd_smbus_dev = pci_get_device(PCI_VENDOR_ID_ATI, 0x4385, NULL); | 52 | amd_smbus_dev = pci_get_device(PCI_VENDOR_ID_ATI, 0x4385, NULL); |
| 53 | if (!amd_smbus_dev) | 53 | if (amd_smbus_dev) { |
| 54 | return 0; | 54 | pci_read_config_byte(amd_smbus_dev, PCI_REVISION_ID, &rev); |
| 55 | 55 | if (rev < 0x40) { | |
| 56 | pci_read_config_byte(amd_smbus_dev, PCI_REVISION_ID, &rev); | 56 | pci_dev_put(amd_smbus_dev); |
| 57 | if (rev < 0x40) { | 57 | amd_smbus_dev = NULL; |
| 58 | pci_dev_put(amd_smbus_dev); | 58 | return 0; |
| 59 | amd_smbus_dev = NULL; | 59 | } |
| 60 | return 0; | 60 | } else { |
| 61 | amd_smbus_dev = pci_get_device(PCI_VENDOR_ID_AMD, 0x780b, NULL); | ||
| 62 | if (!amd_smbus_dev) | ||
| 63 | return 0; | ||
| 64 | pci_read_config_byte(amd_smbus_dev, PCI_REVISION_ID, &rev); | ||
| 65 | if (rev < 0x11 || rev > 0x18) { | ||
| 66 | pci_dev_put(amd_smbus_dev); | ||
| 67 | amd_smbus_dev = NULL; | ||
| 68 | return 0; | ||
| 69 | } | ||
| 61 | } | 70 | } |
| 62 | 71 | ||
| 63 | if (!amd_nb_dev) | 72 | if (!amd_nb_dev) |
| 64 | amd_nb_dev = pci_get_device(PCI_VENDOR_ID_AMD, 0x1510, NULL); | 73 | amd_nb_dev = pci_get_device(PCI_VENDOR_ID_AMD, 0x1510, NULL); |
| 65 | if (!amd_nb_dev) | ||
| 66 | ehci_err(ehci, "QUIRK: unable to get AMD NB device\n"); | ||
| 67 | 74 | ||
| 68 | ehci_info(ehci, "QUIRK: Enable AMD SB800 L1 fix\n"); | 75 | ehci_info(ehci, "QUIRK: Enable exception for AMD Hudson ASPM\n"); |
| 69 | 76 | ||
| 70 | pci_dev_put(amd_smbus_dev); | 77 | pci_dev_put(amd_smbus_dev); |
| 71 | amd_smbus_dev = NULL; | 78 | amd_smbus_dev = NULL; |
| @@ -131,7 +138,7 @@ static int ehci_pci_setup(struct usb_hcd *hcd) | |||
| 131 | /* cache this readonly data; minimize chip reads */ | 138 | /* cache this readonly data; minimize chip reads */ |
| 132 | ehci->hcs_params = ehci_readl(ehci, &ehci->caps->hcs_params); | 139 | ehci->hcs_params = ehci_readl(ehci, &ehci->caps->hcs_params); |
| 133 | 140 | ||
| 134 | if (ehci_quirk_amd_SB800(ehci)) | 141 | if (ehci_quirk_amd_hudson(ehci)) |
| 135 | ehci->amd_l1_fix = 1; | 142 | ehci->amd_l1_fix = 1; |
| 136 | 143 | ||
| 137 | retval = ehci_halt(ehci); | 144 | retval = ehci_halt(ehci); |
diff --git a/drivers/usb/host/fsl-mph-dr-of.c b/drivers/usb/host/fsl-mph-dr-of.c index 574b99ea070..79a66d622f9 100644 --- a/drivers/usb/host/fsl-mph-dr-of.c +++ b/drivers/usb/host/fsl-mph-dr-of.c | |||
| @@ -262,19 +262,24 @@ static void fsl_usb2_mpc5121_exit(struct platform_device *pdev) | |||
| 262 | } | 262 | } |
| 263 | } | 263 | } |
| 264 | 264 | ||
| 265 | struct fsl_usb2_platform_data fsl_usb2_mpc5121_pd = { | 265 | static struct fsl_usb2_platform_data fsl_usb2_mpc5121_pd = { |
| 266 | .big_endian_desc = 1, | 266 | .big_endian_desc = 1, |
| 267 | .big_endian_mmio = 1, | 267 | .big_endian_mmio = 1, |
| 268 | .es = 1, | 268 | .es = 1, |
| 269 | .have_sysif_regs = 0, | ||
| 269 | .le_setup_buf = 1, | 270 | .le_setup_buf = 1, |
| 270 | .init = fsl_usb2_mpc5121_init, | 271 | .init = fsl_usb2_mpc5121_init, |
| 271 | .exit = fsl_usb2_mpc5121_exit, | 272 | .exit = fsl_usb2_mpc5121_exit, |
| 272 | }; | 273 | }; |
| 273 | #endif /* CONFIG_PPC_MPC512x */ | 274 | #endif /* CONFIG_PPC_MPC512x */ |
| 274 | 275 | ||
| 276 | static struct fsl_usb2_platform_data fsl_usb2_mpc8xxx_pd = { | ||
| 277 | .have_sysif_regs = 1, | ||
| 278 | }; | ||
| 279 | |||
| 275 | static const struct of_device_id fsl_usb2_mph_dr_of_match[] = { | 280 | static const struct of_device_id fsl_usb2_mph_dr_of_match[] = { |
| 276 | { .compatible = "fsl-usb2-mph", }, | 281 | { .compatible = "fsl-usb2-mph", .data = &fsl_usb2_mpc8xxx_pd, }, |
| 277 | { .compatible = "fsl-usb2-dr", }, | 282 | { .compatible = "fsl-usb2-dr", .data = &fsl_usb2_mpc8xxx_pd, }, |
| 278 | #ifdef CONFIG_PPC_MPC512x | 283 | #ifdef CONFIG_PPC_MPC512x |
| 279 | { .compatible = "fsl,mpc5121-usb2-dr", .data = &fsl_usb2_mpc5121_pd, }, | 284 | { .compatible = "fsl,mpc5121-usb2-dr", .data = &fsl_usb2_mpc5121_pd, }, |
| 280 | #endif | 285 | #endif |
diff --git a/drivers/usb/host/xhci-ring.c b/drivers/usb/host/xhci-ring.c index df558f6f84e..3e8211c1ce5 100644 --- a/drivers/usb/host/xhci-ring.c +++ b/drivers/usb/host/xhci-ring.c | |||
| @@ -308,11 +308,8 @@ static int room_on_ring(struct xhci_hcd *xhci, struct xhci_ring *ring, | |||
| 308 | /* Ring the host controller doorbell after placing a command on the ring */ | 308 | /* Ring the host controller doorbell after placing a command on the ring */ |
| 309 | void xhci_ring_cmd_db(struct xhci_hcd *xhci) | 309 | void xhci_ring_cmd_db(struct xhci_hcd *xhci) |
| 310 | { | 310 | { |
| 311 | u32 temp; | ||
| 312 | |||
| 313 | xhci_dbg(xhci, "// Ding dong!\n"); | 311 | xhci_dbg(xhci, "// Ding dong!\n"); |
| 314 | temp = xhci_readl(xhci, &xhci->dba->doorbell[0]) & DB_MASK; | 312 | xhci_writel(xhci, DB_VALUE_HOST, &xhci->dba->doorbell[0]); |
| 315 | xhci_writel(xhci, temp | DB_TARGET_HOST, &xhci->dba->doorbell[0]); | ||
| 316 | /* Flush PCI posted writes */ | 313 | /* Flush PCI posted writes */ |
| 317 | xhci_readl(xhci, &xhci->dba->doorbell[0]); | 314 | xhci_readl(xhci, &xhci->dba->doorbell[0]); |
| 318 | } | 315 | } |
| @@ -322,26 +319,24 @@ void xhci_ring_ep_doorbell(struct xhci_hcd *xhci, | |||
| 322 | unsigned int ep_index, | 319 | unsigned int ep_index, |
| 323 | unsigned int stream_id) | 320 | unsigned int stream_id) |
| 324 | { | 321 | { |
| 325 | struct xhci_virt_ep *ep; | ||
| 326 | unsigned int ep_state; | ||
| 327 | u32 field; | ||
| 328 | __u32 __iomem *db_addr = &xhci->dba->doorbell[slot_id]; | 322 | __u32 __iomem *db_addr = &xhci->dba->doorbell[slot_id]; |
| 323 | struct xhci_virt_ep *ep = &xhci->devs[slot_id]->eps[ep_index]; | ||
| 324 | unsigned int ep_state = ep->ep_state; | ||
| 329 | 325 | ||
| 330 | ep = &xhci->devs[slot_id]->eps[ep_index]; | ||
| 331 | ep_state = ep->ep_state; | ||
| 332 | /* Don't ring the doorbell for this endpoint if there are pending | 326 | /* Don't ring the doorbell for this endpoint if there are pending |
| 333 | * cancellations because the we don't want to interrupt processing. | 327 | * cancellations because we don't want to interrupt processing. |
| 334 | * We don't want to restart any stream rings if there's a set dequeue | 328 | * We don't want to restart any stream rings if there's a set dequeue |
| 335 | * pointer command pending because the device can choose to start any | 329 | * pointer command pending because the device can choose to start any |
| 336 | * stream once the endpoint is on the HW schedule. | 330 | * stream once the endpoint is on the HW schedule. |
| 337 | * FIXME - check all the stream rings for pending cancellations. | 331 | * FIXME - check all the stream rings for pending cancellations. |
| 338 | */ | 332 | */ |
| 339 | if (!(ep_state & EP_HALT_PENDING) && !(ep_state & SET_DEQ_PENDING) | 333 | if ((ep_state & EP_HALT_PENDING) || (ep_state & SET_DEQ_PENDING) || |
| 340 | && !(ep_state & EP_HALTED)) { | 334 | (ep_state & EP_HALTED)) |
| 341 | field = xhci_readl(xhci, db_addr) & DB_MASK; | 335 | return; |
| 342 | field |= EPI_TO_DB(ep_index) | STREAM_ID_TO_DB(stream_id); | 336 | xhci_writel(xhci, DB_VALUE(ep_index, stream_id), db_addr); |
| 343 | xhci_writel(xhci, field, db_addr); | 337 | /* The CPU has better things to do at this point than wait for a |
| 344 | } | 338 | * write-posting flush. It'll get there soon enough. |
| 339 | */ | ||
| 345 | } | 340 | } |
| 346 | 341 | ||
| 347 | /* Ring the doorbell for any rings with pending URBs */ | 342 | /* Ring the doorbell for any rings with pending URBs */ |
| @@ -1188,7 +1183,7 @@ static void handle_port_status(struct xhci_hcd *xhci, | |||
| 1188 | 1183 | ||
| 1189 | addr = &xhci->op_regs->port_status_base + NUM_PORT_REGS * (port_id - 1); | 1184 | addr = &xhci->op_regs->port_status_base + NUM_PORT_REGS * (port_id - 1); |
| 1190 | temp = xhci_readl(xhci, addr); | 1185 | temp = xhci_readl(xhci, addr); |
| 1191 | if ((temp & PORT_CONNECT) && (hcd->state == HC_STATE_SUSPENDED)) { | 1186 | if (hcd->state == HC_STATE_SUSPENDED) { |
| 1192 | xhci_dbg(xhci, "resume root hub\n"); | 1187 | xhci_dbg(xhci, "resume root hub\n"); |
| 1193 | usb_hcd_resume_root_hub(hcd); | 1188 | usb_hcd_resume_root_hub(hcd); |
| 1194 | } | 1189 | } |
| @@ -1710,8 +1705,7 @@ static int process_bulk_intr_td(struct xhci_hcd *xhci, struct xhci_td *td, | |||
| 1710 | /* Others already handled above */ | 1705 | /* Others already handled above */ |
| 1711 | break; | 1706 | break; |
| 1712 | } | 1707 | } |
| 1713 | dev_dbg(&td->urb->dev->dev, | 1708 | xhci_dbg(xhci, "ep %#x - asked for %d bytes, " |
| 1714 | "ep %#x - asked for %d bytes, " | ||
| 1715 | "%d bytes untransferred\n", | 1709 | "%d bytes untransferred\n", |
| 1716 | td->urb->ep->desc.bEndpointAddress, | 1710 | td->urb->ep->desc.bEndpointAddress, |
| 1717 | td->urb->transfer_buffer_length, | 1711 | td->urb->transfer_buffer_length, |
| @@ -2389,7 +2383,8 @@ static unsigned int count_sg_trbs_needed(struct xhci_hcd *xhci, struct urb *urb) | |||
| 2389 | } | 2383 | } |
| 2390 | xhci_dbg(xhci, "\n"); | 2384 | xhci_dbg(xhci, "\n"); |
| 2391 | if (!in_interrupt()) | 2385 | if (!in_interrupt()) |
| 2392 | dev_dbg(&urb->dev->dev, "ep %#x - urb len = %d, sglist used, num_trbs = %d\n", | 2386 | xhci_dbg(xhci, "ep %#x - urb len = %d, sglist used, " |
| 2387 | "num_trbs = %d\n", | ||
| 2393 | urb->ep->desc.bEndpointAddress, | 2388 | urb->ep->desc.bEndpointAddress, |
| 2394 | urb->transfer_buffer_length, | 2389 | urb->transfer_buffer_length, |
| 2395 | num_trbs); | 2390 | num_trbs); |
| @@ -2414,14 +2409,17 @@ static void check_trb_math(struct urb *urb, int num_trbs, int running_total) | |||
| 2414 | 2409 | ||
| 2415 | static void giveback_first_trb(struct xhci_hcd *xhci, int slot_id, | 2410 | static void giveback_first_trb(struct xhci_hcd *xhci, int slot_id, |
| 2416 | unsigned int ep_index, unsigned int stream_id, int start_cycle, | 2411 | unsigned int ep_index, unsigned int stream_id, int start_cycle, |
| 2417 | struct xhci_generic_trb *start_trb, struct xhci_td *td) | 2412 | struct xhci_generic_trb *start_trb) |
| 2418 | { | 2413 | { |
| 2419 | /* | 2414 | /* |
| 2420 | * Pass all the TRBs to the hardware at once and make sure this write | 2415 | * Pass all the TRBs to the hardware at once and make sure this write |
| 2421 | * isn't reordered. | 2416 | * isn't reordered. |
| 2422 | */ | 2417 | */ |
| 2423 | wmb(); | 2418 | wmb(); |
| 2424 | start_trb->field[3] |= start_cycle; | 2419 | if (start_cycle) |
| 2420 | start_trb->field[3] |= start_cycle; | ||
| 2421 | else | ||
| 2422 | start_trb->field[3] &= ~0x1; | ||
| 2425 | xhci_ring_ep_doorbell(xhci, slot_id, ep_index, stream_id); | 2423 | xhci_ring_ep_doorbell(xhci, slot_id, ep_index, stream_id); |
| 2426 | } | 2424 | } |
| 2427 | 2425 | ||
| @@ -2449,7 +2447,7 @@ int xhci_queue_intr_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2449 | * to set the polling interval (once the API is added). | 2447 | * to set the polling interval (once the API is added). |
| 2450 | */ | 2448 | */ |
| 2451 | if (xhci_interval != ep_interval) { | 2449 | if (xhci_interval != ep_interval) { |
| 2452 | if (!printk_ratelimit()) | 2450 | if (printk_ratelimit()) |
| 2453 | dev_dbg(&urb->dev->dev, "Driver uses different interval" | 2451 | dev_dbg(&urb->dev->dev, "Driver uses different interval" |
| 2454 | " (%d microframe%s) than xHCI " | 2452 | " (%d microframe%s) than xHCI " |
| 2455 | "(%d microframe%s)\n", | 2453 | "(%d microframe%s)\n", |
| @@ -2551,9 +2549,11 @@ static int queue_bulk_sg_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2551 | u32 remainder = 0; | 2549 | u32 remainder = 0; |
| 2552 | 2550 | ||
| 2553 | /* Don't change the cycle bit of the first TRB until later */ | 2551 | /* Don't change the cycle bit of the first TRB until later */ |
| 2554 | if (first_trb) | 2552 | if (first_trb) { |
| 2555 | first_trb = false; | 2553 | first_trb = false; |
| 2556 | else | 2554 | if (start_cycle == 0) |
| 2555 | field |= 0x1; | ||
| 2556 | } else | ||
| 2557 | field |= ep_ring->cycle_state; | 2557 | field |= ep_ring->cycle_state; |
| 2558 | 2558 | ||
| 2559 | /* Chain all the TRBs together; clear the chain bit in the last | 2559 | /* Chain all the TRBs together; clear the chain bit in the last |
| @@ -2625,7 +2625,7 @@ static int queue_bulk_sg_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2625 | 2625 | ||
| 2626 | check_trb_math(urb, num_trbs, running_total); | 2626 | check_trb_math(urb, num_trbs, running_total); |
| 2627 | giveback_first_trb(xhci, slot_id, ep_index, urb->stream_id, | 2627 | giveback_first_trb(xhci, slot_id, ep_index, urb->stream_id, |
| 2628 | start_cycle, start_trb, td); | 2628 | start_cycle, start_trb); |
| 2629 | return 0; | 2629 | return 0; |
| 2630 | } | 2630 | } |
| 2631 | 2631 | ||
| @@ -2671,7 +2671,8 @@ int xhci_queue_bulk_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2671 | /* FIXME: this doesn't deal with URB_ZERO_PACKET - need one more */ | 2671 | /* FIXME: this doesn't deal with URB_ZERO_PACKET - need one more */ |
| 2672 | 2672 | ||
| 2673 | if (!in_interrupt()) | 2673 | if (!in_interrupt()) |
| 2674 | dev_dbg(&urb->dev->dev, "ep %#x - urb len = %#x (%d), addr = %#llx, num_trbs = %d\n", | 2674 | xhci_dbg(xhci, "ep %#x - urb len = %#x (%d), " |
| 2675 | "addr = %#llx, num_trbs = %d\n", | ||
| 2675 | urb->ep->desc.bEndpointAddress, | 2676 | urb->ep->desc.bEndpointAddress, |
| 2676 | urb->transfer_buffer_length, | 2677 | urb->transfer_buffer_length, |
| 2677 | urb->transfer_buffer_length, | 2678 | urb->transfer_buffer_length, |
| @@ -2711,9 +2712,11 @@ int xhci_queue_bulk_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2711 | field = 0; | 2712 | field = 0; |
| 2712 | 2713 | ||
| 2713 | /* Don't change the cycle bit of the first TRB until later */ | 2714 | /* Don't change the cycle bit of the first TRB until later */ |
| 2714 | if (first_trb) | 2715 | if (first_trb) { |
| 2715 | first_trb = false; | 2716 | first_trb = false; |
| 2716 | else | 2717 | if (start_cycle == 0) |
| 2718 | field |= 0x1; | ||
| 2719 | } else | ||
| 2717 | field |= ep_ring->cycle_state; | 2720 | field |= ep_ring->cycle_state; |
| 2718 | 2721 | ||
| 2719 | /* Chain all the TRBs together; clear the chain bit in the last | 2722 | /* Chain all the TRBs together; clear the chain bit in the last |
| @@ -2757,7 +2760,7 @@ int xhci_queue_bulk_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2757 | 2760 | ||
| 2758 | check_trb_math(urb, num_trbs, running_total); | 2761 | check_trb_math(urb, num_trbs, running_total); |
| 2759 | giveback_first_trb(xhci, slot_id, ep_index, urb->stream_id, | 2762 | giveback_first_trb(xhci, slot_id, ep_index, urb->stream_id, |
| 2760 | start_cycle, start_trb, td); | 2763 | start_cycle, start_trb); |
| 2761 | return 0; | 2764 | return 0; |
| 2762 | } | 2765 | } |
| 2763 | 2766 | ||
| @@ -2818,13 +2821,17 @@ int xhci_queue_ctrl_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2818 | /* Queue setup TRB - see section 6.4.1.2.1 */ | 2821 | /* Queue setup TRB - see section 6.4.1.2.1 */ |
| 2819 | /* FIXME better way to translate setup_packet into two u32 fields? */ | 2822 | /* FIXME better way to translate setup_packet into two u32 fields? */ |
| 2820 | setup = (struct usb_ctrlrequest *) urb->setup_packet; | 2823 | setup = (struct usb_ctrlrequest *) urb->setup_packet; |
| 2824 | field = 0; | ||
| 2825 | field |= TRB_IDT | TRB_TYPE(TRB_SETUP); | ||
| 2826 | if (start_cycle == 0) | ||
| 2827 | field |= 0x1; | ||
| 2821 | queue_trb(xhci, ep_ring, false, true, | 2828 | queue_trb(xhci, ep_ring, false, true, |
| 2822 | /* FIXME endianness is probably going to bite my ass here. */ | 2829 | /* FIXME endianness is probably going to bite my ass here. */ |
| 2823 | setup->bRequestType | setup->bRequest << 8 | setup->wValue << 16, | 2830 | setup->bRequestType | setup->bRequest << 8 | setup->wValue << 16, |
| 2824 | setup->wIndex | setup->wLength << 16, | 2831 | setup->wIndex | setup->wLength << 16, |
| 2825 | TRB_LEN(8) | TRB_INTR_TARGET(0), | 2832 | TRB_LEN(8) | TRB_INTR_TARGET(0), |
| 2826 | /* Immediate data in pointer */ | 2833 | /* Immediate data in pointer */ |
| 2827 | TRB_IDT | TRB_TYPE(TRB_SETUP)); | 2834 | field); |
| 2828 | 2835 | ||
| 2829 | /* If there's data, queue data TRBs */ | 2836 | /* If there's data, queue data TRBs */ |
| 2830 | field = 0; | 2837 | field = 0; |
| @@ -2859,7 +2866,7 @@ int xhci_queue_ctrl_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2859 | field | TRB_IOC | TRB_TYPE(TRB_STATUS) | ep_ring->cycle_state); | 2866 | field | TRB_IOC | TRB_TYPE(TRB_STATUS) | ep_ring->cycle_state); |
| 2860 | 2867 | ||
| 2861 | giveback_first_trb(xhci, slot_id, ep_index, 0, | 2868 | giveback_first_trb(xhci, slot_id, ep_index, 0, |
| 2862 | start_cycle, start_trb, td); | 2869 | start_cycle, start_trb); |
| 2863 | return 0; | 2870 | return 0; |
| 2864 | } | 2871 | } |
| 2865 | 2872 | ||
| @@ -2900,6 +2907,7 @@ static int xhci_queue_isoc_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2900 | int running_total, trb_buff_len, td_len, td_remain_len, ret; | 2907 | int running_total, trb_buff_len, td_len, td_remain_len, ret; |
| 2901 | u64 start_addr, addr; | 2908 | u64 start_addr, addr; |
| 2902 | int i, j; | 2909 | int i, j; |
| 2910 | bool more_trbs_coming; | ||
| 2903 | 2911 | ||
| 2904 | ep_ring = xhci->devs[slot_id]->eps[ep_index].ring; | 2912 | ep_ring = xhci->devs[slot_id]->eps[ep_index].ring; |
| 2905 | 2913 | ||
| @@ -2910,7 +2918,7 @@ static int xhci_queue_isoc_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2910 | } | 2918 | } |
| 2911 | 2919 | ||
| 2912 | if (!in_interrupt()) | 2920 | if (!in_interrupt()) |
| 2913 | dev_dbg(&urb->dev->dev, "ep %#x - urb len = %#x (%d)," | 2921 | xhci_dbg(xhci, "ep %#x - urb len = %#x (%d)," |
| 2914 | " addr = %#llx, num_tds = %d\n", | 2922 | " addr = %#llx, num_tds = %d\n", |
| 2915 | urb->ep->desc.bEndpointAddress, | 2923 | urb->ep->desc.bEndpointAddress, |
| 2916 | urb->transfer_buffer_length, | 2924 | urb->transfer_buffer_length, |
| @@ -2950,7 +2958,10 @@ static int xhci_queue_isoc_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2950 | field |= TRB_TYPE(TRB_ISOC); | 2958 | field |= TRB_TYPE(TRB_ISOC); |
| 2951 | /* Assume URB_ISO_ASAP is set */ | 2959 | /* Assume URB_ISO_ASAP is set */ |
| 2952 | field |= TRB_SIA; | 2960 | field |= TRB_SIA; |
| 2953 | if (i > 0) | 2961 | if (i == 0) { |
| 2962 | if (start_cycle == 0) | ||
| 2963 | field |= 0x1; | ||
| 2964 | } else | ||
| 2954 | field |= ep_ring->cycle_state; | 2965 | field |= ep_ring->cycle_state; |
| 2955 | first_trb = false; | 2966 | first_trb = false; |
| 2956 | } else { | 2967 | } else { |
| @@ -2965,9 +2976,11 @@ static int xhci_queue_isoc_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2965 | */ | 2976 | */ |
| 2966 | if (j < trbs_per_td - 1) { | 2977 | if (j < trbs_per_td - 1) { |
| 2967 | field |= TRB_CHAIN; | 2978 | field |= TRB_CHAIN; |
| 2979 | more_trbs_coming = true; | ||
| 2968 | } else { | 2980 | } else { |
| 2969 | td->last_trb = ep_ring->enqueue; | 2981 | td->last_trb = ep_ring->enqueue; |
| 2970 | field |= TRB_IOC; | 2982 | field |= TRB_IOC; |
| 2983 | more_trbs_coming = false; | ||
| 2971 | } | 2984 | } |
| 2972 | 2985 | ||
| 2973 | /* Calculate TRB length */ | 2986 | /* Calculate TRB length */ |
| @@ -2980,7 +2993,7 @@ static int xhci_queue_isoc_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 2980 | length_field = TRB_LEN(trb_buff_len) | | 2993 | length_field = TRB_LEN(trb_buff_len) | |
| 2981 | remainder | | 2994 | remainder | |
| 2982 | TRB_INTR_TARGET(0); | 2995 | TRB_INTR_TARGET(0); |
| 2983 | queue_trb(xhci, ep_ring, false, false, | 2996 | queue_trb(xhci, ep_ring, false, more_trbs_coming, |
| 2984 | lower_32_bits(addr), | 2997 | lower_32_bits(addr), |
| 2985 | upper_32_bits(addr), | 2998 | upper_32_bits(addr), |
| 2986 | length_field, | 2999 | length_field, |
| @@ -3003,10 +3016,8 @@ static int xhci_queue_isoc_tx(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 3003 | } | 3016 | } |
| 3004 | } | 3017 | } |
| 3005 | 3018 | ||
| 3006 | wmb(); | 3019 | giveback_first_trb(xhci, slot_id, ep_index, urb->stream_id, |
| 3007 | start_trb->field[3] |= start_cycle; | 3020 | start_cycle, start_trb); |
| 3008 | |||
| 3009 | xhci_ring_ep_doorbell(xhci, slot_id, ep_index, urb->stream_id); | ||
| 3010 | return 0; | 3021 | return 0; |
| 3011 | } | 3022 | } |
| 3012 | 3023 | ||
| @@ -3064,7 +3075,7 @@ int xhci_queue_isoc_tx_prepare(struct xhci_hcd *xhci, gfp_t mem_flags, | |||
| 3064 | * to set the polling interval (once the API is added). | 3075 | * to set the polling interval (once the API is added). |
| 3065 | */ | 3076 | */ |
| 3066 | if (xhci_interval != ep_interval) { | 3077 | if (xhci_interval != ep_interval) { |
| 3067 | if (!printk_ratelimit()) | 3078 | if (printk_ratelimit()) |
| 3068 | dev_dbg(&urb->dev->dev, "Driver uses different interval" | 3079 | dev_dbg(&urb->dev->dev, "Driver uses different interval" |
| 3069 | " (%d microframe%s) than xHCI " | 3080 | " (%d microframe%s) than xHCI " |
| 3070 | "(%d microframe%s)\n", | 3081 | "(%d microframe%s)\n", |
diff --git a/drivers/usb/host/xhci.c b/drivers/usb/host/xhci.c index 45e4a3108cc..34cf4e16587 100644 --- a/drivers/usb/host/xhci.c +++ b/drivers/usb/host/xhci.c | |||
| @@ -226,7 +226,8 @@ static int xhci_setup_msi(struct xhci_hcd *xhci) | |||
| 226 | static int xhci_setup_msix(struct xhci_hcd *xhci) | 226 | static int xhci_setup_msix(struct xhci_hcd *xhci) |
| 227 | { | 227 | { |
| 228 | int i, ret = 0; | 228 | int i, ret = 0; |
| 229 | struct pci_dev *pdev = to_pci_dev(xhci_to_hcd(xhci)->self.controller); | 229 | struct usb_hcd *hcd = xhci_to_hcd(xhci); |
| 230 | struct pci_dev *pdev = to_pci_dev(hcd->self.controller); | ||
| 230 | 231 | ||
| 231 | /* | 232 | /* |
| 232 | * calculate number of msi-x vectors supported. | 233 | * calculate number of msi-x vectors supported. |
| @@ -265,6 +266,7 @@ static int xhci_setup_msix(struct xhci_hcd *xhci) | |||
| 265 | goto disable_msix; | 266 | goto disable_msix; |
| 266 | } | 267 | } |
| 267 | 268 | ||
| 269 | hcd->msix_enabled = 1; | ||
| 268 | return ret; | 270 | return ret; |
| 269 | 271 | ||
| 270 | disable_msix: | 272 | disable_msix: |
| @@ -280,7 +282,8 @@ free_entries: | |||
| 280 | /* Free any IRQs and disable MSI-X */ | 282 | /* Free any IRQs and disable MSI-X */ |
| 281 | static void xhci_cleanup_msix(struct xhci_hcd *xhci) | 283 | static void xhci_cleanup_msix(struct xhci_hcd *xhci) |
| 282 | { | 284 | { |
| 283 | struct pci_dev *pdev = to_pci_dev(xhci_to_hcd(xhci)->self.controller); | 285 | struct usb_hcd *hcd = xhci_to_hcd(xhci); |
| 286 | struct pci_dev *pdev = to_pci_dev(hcd->self.controller); | ||
| 284 | 287 | ||
| 285 | xhci_free_irq(xhci); | 288 | xhci_free_irq(xhci); |
| 286 | 289 | ||
| @@ -292,6 +295,7 @@ static void xhci_cleanup_msix(struct xhci_hcd *xhci) | |||
| 292 | pci_disable_msi(pdev); | 295 | pci_disable_msi(pdev); |
| 293 | } | 296 | } |
| 294 | 297 | ||
| 298 | hcd->msix_enabled = 0; | ||
| 295 | return; | 299 | return; |
| 296 | } | 300 | } |
| 297 | 301 | ||
| @@ -508,9 +512,10 @@ void xhci_stop(struct usb_hcd *hcd) | |||
| 508 | spin_lock_irq(&xhci->lock); | 512 | spin_lock_irq(&xhci->lock); |
| 509 | xhci_halt(xhci); | 513 | xhci_halt(xhci); |
| 510 | xhci_reset(xhci); | 514 | xhci_reset(xhci); |
| 511 | xhci_cleanup_msix(xhci); | ||
| 512 | spin_unlock_irq(&xhci->lock); | 515 | spin_unlock_irq(&xhci->lock); |
| 513 | 516 | ||
| 517 | xhci_cleanup_msix(xhci); | ||
| 518 | |||
| 514 | #ifdef CONFIG_USB_XHCI_HCD_DEBUGGING | 519 | #ifdef CONFIG_USB_XHCI_HCD_DEBUGGING |
| 515 | /* Tell the event ring poll function not to reschedule */ | 520 | /* Tell the event ring poll function not to reschedule */ |
| 516 | xhci->zombie = 1; | 521 | xhci->zombie = 1; |
| @@ -544,9 +549,10 @@ void xhci_shutdown(struct usb_hcd *hcd) | |||
| 544 | 549 | ||
| 545 | spin_lock_irq(&xhci->lock); | 550 | spin_lock_irq(&xhci->lock); |
| 546 | xhci_halt(xhci); | 551 | xhci_halt(xhci); |
| 547 | xhci_cleanup_msix(xhci); | ||
| 548 | spin_unlock_irq(&xhci->lock); | 552 | spin_unlock_irq(&xhci->lock); |
| 549 | 553 | ||
| 554 | xhci_cleanup_msix(xhci); | ||
| 555 | |||
| 550 | xhci_dbg(xhci, "xhci_shutdown completed - status = %x\n", | 556 | xhci_dbg(xhci, "xhci_shutdown completed - status = %x\n", |
| 551 | xhci_readl(xhci, &xhci->op_regs->status)); | 557 | xhci_readl(xhci, &xhci->op_regs->status)); |
| 552 | } | 558 | } |
| @@ -647,6 +653,7 @@ int xhci_suspend(struct xhci_hcd *xhci) | |||
| 647 | int rc = 0; | 653 | int rc = 0; |
| 648 | struct usb_hcd *hcd = xhci_to_hcd(xhci); | 654 | struct usb_hcd *hcd = xhci_to_hcd(xhci); |
| 649 | u32 command; | 655 | u32 command; |
| 656 | int i; | ||
| 650 | 657 | ||
| 651 | spin_lock_irq(&xhci->lock); | 658 | spin_lock_irq(&xhci->lock); |
| 652 | clear_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags); | 659 | clear_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags); |
| @@ -677,10 +684,15 @@ int xhci_suspend(struct xhci_hcd *xhci) | |||
| 677 | spin_unlock_irq(&xhci->lock); | 684 | spin_unlock_irq(&xhci->lock); |
| 678 | return -ETIMEDOUT; | 685 | return -ETIMEDOUT; |
| 679 | } | 686 | } |
| 680 | /* step 5: remove core well power */ | ||
| 681 | xhci_cleanup_msix(xhci); | ||
| 682 | spin_unlock_irq(&xhci->lock); | 687 | spin_unlock_irq(&xhci->lock); |
| 683 | 688 | ||
| 689 | /* step 5: remove core well power */ | ||
| 690 | /* synchronize irq when using MSI-X */ | ||
| 691 | if (xhci->msix_entries) { | ||
| 692 | for (i = 0; i < xhci->msix_count; i++) | ||
| 693 | synchronize_irq(xhci->msix_entries[i].vector); | ||
| 694 | } | ||
| 695 | |||
| 684 | return rc; | 696 | return rc; |
| 685 | } | 697 | } |
| 686 | 698 | ||
| @@ -694,7 +706,6 @@ int xhci_resume(struct xhci_hcd *xhci, bool hibernated) | |||
| 694 | { | 706 | { |
| 695 | u32 command, temp = 0; | 707 | u32 command, temp = 0; |
| 696 | struct usb_hcd *hcd = xhci_to_hcd(xhci); | 708 | struct usb_hcd *hcd = xhci_to_hcd(xhci); |
| 697 | struct pci_dev *pdev = to_pci_dev(hcd->self.controller); | ||
| 698 | int old_state, retval; | 709 | int old_state, retval; |
| 699 | 710 | ||
| 700 | old_state = hcd->state; | 711 | old_state = hcd->state; |
| @@ -729,9 +740,8 @@ int xhci_resume(struct xhci_hcd *xhci, bool hibernated) | |||
| 729 | xhci_dbg(xhci, "Stop HCD\n"); | 740 | xhci_dbg(xhci, "Stop HCD\n"); |
| 730 | xhci_halt(xhci); | 741 | xhci_halt(xhci); |
| 731 | xhci_reset(xhci); | 742 | xhci_reset(xhci); |
| 732 | if (hibernated) | ||
| 733 | xhci_cleanup_msix(xhci); | ||
| 734 | spin_unlock_irq(&xhci->lock); | 743 | spin_unlock_irq(&xhci->lock); |
| 744 | xhci_cleanup_msix(xhci); | ||
| 735 | 745 | ||
| 736 | #ifdef CONFIG_USB_XHCI_HCD_DEBUGGING | 746 | #ifdef CONFIG_USB_XHCI_HCD_DEBUGGING |
| 737 | /* Tell the event ring poll function not to reschedule */ | 747 | /* Tell the event ring poll function not to reschedule */ |
| @@ -765,30 +775,6 @@ int xhci_resume(struct xhci_hcd *xhci, bool hibernated) | |||
| 765 | return retval; | 775 | return retval; |
| 766 | } | 776 | } |
| 767 | 777 | ||
| 768 | spin_unlock_irq(&xhci->lock); | ||
| 769 | /* Re-setup MSI-X */ | ||
| 770 | if (hcd->irq) | ||
| 771 | free_irq(hcd->irq, hcd); | ||
| 772 | hcd->irq = -1; | ||
| 773 | |||
| 774 | retval = xhci_setup_msix(xhci); | ||
| 775 | if (retval) | ||
| 776 | /* fall back to msi*/ | ||
| 777 | retval = xhci_setup_msi(xhci); | ||
| 778 | |||
| 779 | if (retval) { | ||
| 780 | /* fall back to legacy interrupt*/ | ||
| 781 | retval = request_irq(pdev->irq, &usb_hcd_irq, IRQF_SHARED, | ||
| 782 | hcd->irq_descr, hcd); | ||
| 783 | if (retval) { | ||
| 784 | xhci_err(xhci, "request interrupt %d failed\n", | ||
| 785 | pdev->irq); | ||
| 786 | return retval; | ||
| 787 | } | ||
| 788 | hcd->irq = pdev->irq; | ||
| 789 | } | ||
| 790 | |||
| 791 | spin_lock_irq(&xhci->lock); | ||
| 792 | /* step 4: set Run/Stop bit */ | 778 | /* step 4: set Run/Stop bit */ |
| 793 | command = xhci_readl(xhci, &xhci->op_regs->command); | 779 | command = xhci_readl(xhci, &xhci->op_regs->command); |
| 794 | command |= CMD_RUN; | 780 | command |= CMD_RUN; |
| @@ -2445,8 +2431,12 @@ int xhci_alloc_dev(struct usb_hcd *hcd, struct usb_device *udev) | |||
| 2445 | xhci_err(xhci, "Error while assigning device slot ID\n"); | 2431 | xhci_err(xhci, "Error while assigning device slot ID\n"); |
| 2446 | return 0; | 2432 | return 0; |
| 2447 | } | 2433 | } |
| 2448 | /* xhci_alloc_virt_device() does not touch rings; no need to lock */ | 2434 | /* xhci_alloc_virt_device() does not touch rings; no need to lock. |
| 2449 | if (!xhci_alloc_virt_device(xhci, xhci->slot_id, udev, GFP_KERNEL)) { | 2435 | * Use GFP_NOIO, since this function can be called from |
| 2436 | * xhci_discover_or_reset_device(), which may be called as part of | ||
| 2437 | * mass storage driver error handling. | ||
| 2438 | */ | ||
| 2439 | if (!xhci_alloc_virt_device(xhci, xhci->slot_id, udev, GFP_NOIO)) { | ||
| 2450 | /* Disable slot, if we can do it without mem alloc */ | 2440 | /* Disable slot, if we can do it without mem alloc */ |
| 2451 | xhci_warn(xhci, "Could not allocate xHCI USB device data structures\n"); | 2441 | xhci_warn(xhci, "Could not allocate xHCI USB device data structures\n"); |
| 2452 | spin_lock_irqsave(&xhci->lock, flags); | 2442 | spin_lock_irqsave(&xhci->lock, flags); |
diff --git a/drivers/usb/host/xhci.h b/drivers/usb/host/xhci.h index 170c367112d..7f236fd2201 100644 --- a/drivers/usb/host/xhci.h +++ b/drivers/usb/host/xhci.h | |||
| @@ -436,22 +436,18 @@ struct xhci_run_regs { | |||
| 436 | /** | 436 | /** |
| 437 | * struct doorbell_array | 437 | * struct doorbell_array |
| 438 | * | 438 | * |
| 439 | * Bits 0 - 7: Endpoint target | ||
| 440 | * Bits 8 - 15: RsvdZ | ||
| 441 | * Bits 16 - 31: Stream ID | ||
| 442 | * | ||
| 439 | * Section 5.6 | 443 | * Section 5.6 |
| 440 | */ | 444 | */ |
| 441 | struct xhci_doorbell_array { | 445 | struct xhci_doorbell_array { |
| 442 | u32 doorbell[256]; | 446 | u32 doorbell[256]; |
| 443 | }; | 447 | }; |
| 444 | 448 | ||
| 445 | #define DB_TARGET_MASK 0xFFFFFF00 | 449 | #define DB_VALUE(ep, stream) ((((ep) + 1) & 0xff) | ((stream) << 16)) |
| 446 | #define DB_STREAM_ID_MASK 0x0000FFFF | 450 | #define DB_VALUE_HOST 0x00000000 |
| 447 | #define DB_TARGET_HOST 0x0 | ||
| 448 | #define DB_STREAM_ID_HOST 0x0 | ||
| 449 | #define DB_MASK (0xff << 8) | ||
| 450 | |||
| 451 | /* Endpoint Target - bits 0:7 */ | ||
| 452 | #define EPI_TO_DB(p) (((p) + 1) & 0xff) | ||
| 453 | #define STREAM_ID_TO_DB(p) (((p) & 0xffff) << 16) | ||
| 454 | |||
| 455 | 451 | ||
| 456 | /** | 452 | /** |
| 457 | * struct xhci_protocol_caps | 453 | * struct xhci_protocol_caps |
diff --git a/drivers/usb/misc/usbled.c b/drivers/usb/misc/usbled.c index 1732d9bc097..1616ad1793a 100644 --- a/drivers/usb/misc/usbled.c +++ b/drivers/usb/misc/usbled.c | |||
| @@ -45,7 +45,7 @@ struct usb_led { | |||
| 45 | 45 | ||
| 46 | static void change_color(struct usb_led *led) | 46 | static void change_color(struct usb_led *led) |
| 47 | { | 47 | { |
| 48 | int retval; | 48 | int retval = 0; |
| 49 | unsigned char *buffer; | 49 | unsigned char *buffer; |
| 50 | 50 | ||
| 51 | buffer = kmalloc(8, GFP_KERNEL); | 51 | buffer = kmalloc(8, GFP_KERNEL); |
diff --git a/drivers/usb/misc/uss720.c b/drivers/usb/misc/uss720.c index 4ff21587ab0..f7a20573803 100644 --- a/drivers/usb/misc/uss720.c +++ b/drivers/usb/misc/uss720.c | |||
| @@ -776,7 +776,6 @@ static const struct usb_device_id uss720_table[] = { | |||
| 776 | { USB_DEVICE(0x0557, 0x2001) }, | 776 | { USB_DEVICE(0x0557, 0x2001) }, |
| 777 | { USB_DEVICE(0x0729, 0x1284) }, | 777 | { USB_DEVICE(0x0729, 0x1284) }, |
| 778 | { USB_DEVICE(0x1293, 0x0002) }, | 778 | { USB_DEVICE(0x1293, 0x0002) }, |
| 779 | { USB_DEVICE(0x1293, 0x0002) }, | ||
| 780 | { USB_DEVICE(0x050d, 0x0002) }, | 779 | { USB_DEVICE(0x050d, 0x0002) }, |
| 781 | { } /* Terminating entry */ | 780 | { } /* Terminating entry */ |
| 782 | }; | 781 | }; |
diff --git a/drivers/usb/otg/nop-usb-xceiv.c b/drivers/usb/otg/nop-usb-xceiv.c index e70014ab097..8acf165fe13 100644 --- a/drivers/usb/otg/nop-usb-xceiv.c +++ b/drivers/usb/otg/nop-usb-xceiv.c | |||
| @@ -132,6 +132,8 @@ static int __devinit nop_usb_xceiv_probe(struct platform_device *pdev) | |||
| 132 | 132 | ||
| 133 | platform_set_drvdata(pdev, nop); | 133 | platform_set_drvdata(pdev, nop); |
| 134 | 134 | ||
| 135 | BLOCKING_INIT_NOTIFIER_HEAD(&nop->otg.notifier); | ||
| 136 | |||
| 135 | return 0; | 137 | return 0; |
| 136 | exit: | 138 | exit: |
| 137 | kfree(nop); | 139 | kfree(nop); |
diff --git a/drivers/usb/otg/ulpi.c b/drivers/usb/otg/ulpi.c index 059d9ac0ab5..770d799d5af 100644 --- a/drivers/usb/otg/ulpi.c +++ b/drivers/usb/otg/ulpi.c | |||
| @@ -45,7 +45,7 @@ struct ulpi_info { | |||
| 45 | /* ULPI hardcoded IDs, used for probing */ | 45 | /* ULPI hardcoded IDs, used for probing */ |
| 46 | static struct ulpi_info ulpi_ids[] = { | 46 | static struct ulpi_info ulpi_ids[] = { |
| 47 | ULPI_INFO(ULPI_ID(0x04cc, 0x1504), "NXP ISP1504"), | 47 | ULPI_INFO(ULPI_ID(0x04cc, 0x1504), "NXP ISP1504"), |
| 48 | ULPI_INFO(ULPI_ID(0x0424, 0x0006), "SMSC USB3319"), | 48 | ULPI_INFO(ULPI_ID(0x0424, 0x0006), "SMSC USB331x"), |
| 49 | }; | 49 | }; |
| 50 | 50 | ||
| 51 | static int ulpi_set_otg_flags(struct otg_transceiver *otg) | 51 | static int ulpi_set_otg_flags(struct otg_transceiver *otg) |
diff --git a/drivers/usb/serial/ch341.c b/drivers/usb/serial/ch341.c index 63f7cc45bca..7b8815ddf36 100644 --- a/drivers/usb/serial/ch341.c +++ b/drivers/usb/serial/ch341.c | |||
| @@ -486,12 +486,22 @@ static void ch341_read_int_callback(struct urb *urb) | |||
| 486 | if (actual_length >= 4) { | 486 | if (actual_length >= 4) { |
| 487 | struct ch341_private *priv = usb_get_serial_port_data(port); | 487 | struct ch341_private *priv = usb_get_serial_port_data(port); |
| 488 | unsigned long flags; | 488 | unsigned long flags; |
| 489 | u8 prev_line_status = priv->line_status; | ||
| 489 | 490 | ||
| 490 | spin_lock_irqsave(&priv->lock, flags); | 491 | spin_lock_irqsave(&priv->lock, flags); |
| 491 | priv->line_status = (~(data[2])) & CH341_BITS_MODEM_STAT; | 492 | priv->line_status = (~(data[2])) & CH341_BITS_MODEM_STAT; |
| 492 | if ((data[1] & CH341_MULT_STAT)) | 493 | if ((data[1] & CH341_MULT_STAT)) |
| 493 | priv->multi_status_change = 1; | 494 | priv->multi_status_change = 1; |
| 494 | spin_unlock_irqrestore(&priv->lock, flags); | 495 | spin_unlock_irqrestore(&priv->lock, flags); |
| 496 | |||
| 497 | if ((priv->line_status ^ prev_line_status) & CH341_BIT_DCD) { | ||
| 498 | struct tty_struct *tty = tty_port_tty_get(&port->port); | ||
| 499 | if (tty) | ||
| 500 | usb_serial_handle_dcd_change(port, tty, | ||
| 501 | priv->line_status & CH341_BIT_DCD); | ||
| 502 | tty_kref_put(tty); | ||
| 503 | } | ||
| 504 | |||
| 495 | wake_up_interruptible(&priv->delta_msr_wait); | 505 | wake_up_interruptible(&priv->delta_msr_wait); |
| 496 | } | 506 | } |
| 497 | 507 | ||
diff --git a/drivers/usb/serial/cp210x.c b/drivers/usb/serial/cp210x.c index 8d7731dbf47..735ea03157a 100644 --- a/drivers/usb/serial/cp210x.c +++ b/drivers/usb/serial/cp210x.c | |||
| @@ -49,7 +49,6 @@ static int cp210x_tiocmset_port(struct usb_serial_port *port, struct file *, | |||
| 49 | static void cp210x_break_ctl(struct tty_struct *, int); | 49 | static void cp210x_break_ctl(struct tty_struct *, int); |
| 50 | static int cp210x_startup(struct usb_serial *); | 50 | static int cp210x_startup(struct usb_serial *); |
| 51 | static void cp210x_dtr_rts(struct usb_serial_port *p, int on); | 51 | static void cp210x_dtr_rts(struct usb_serial_port *p, int on); |
| 52 | static int cp210x_carrier_raised(struct usb_serial_port *p); | ||
| 53 | 52 | ||
| 54 | static int debug; | 53 | static int debug; |
| 55 | 54 | ||
| @@ -87,7 +86,6 @@ static const struct usb_device_id id_table[] = { | |||
| 87 | { USB_DEVICE(0x10C4, 0x8115) }, /* Arygon NFC/Mifare Reader */ | 86 | { USB_DEVICE(0x10C4, 0x8115) }, /* Arygon NFC/Mifare Reader */ |
| 88 | { USB_DEVICE(0x10C4, 0x813D) }, /* Burnside Telecom Deskmobile */ | 87 | { USB_DEVICE(0x10C4, 0x813D) }, /* Burnside Telecom Deskmobile */ |
| 89 | { USB_DEVICE(0x10C4, 0x813F) }, /* Tams Master Easy Control */ | 88 | { USB_DEVICE(0x10C4, 0x813F) }, /* Tams Master Easy Control */ |
| 90 | { USB_DEVICE(0x10C4, 0x8149) }, /* West Mountain Radio Computerized Battery Analyzer */ | ||
| 91 | { USB_DEVICE(0x10C4, 0x814A) }, /* West Mountain Radio RIGblaster P&P */ | 89 | { USB_DEVICE(0x10C4, 0x814A) }, /* West Mountain Radio RIGblaster P&P */ |
| 92 | { USB_DEVICE(0x10C4, 0x814B) }, /* West Mountain Radio RIGtalk */ | 90 | { USB_DEVICE(0x10C4, 0x814B) }, /* West Mountain Radio RIGtalk */ |
| 93 | { USB_DEVICE(0x10C4, 0x8156) }, /* B&G H3000 link cable */ | 91 | { USB_DEVICE(0x10C4, 0x8156) }, /* B&G H3000 link cable */ |
| @@ -110,7 +108,9 @@ static const struct usb_device_id id_table[] = { | |||
| 110 | { USB_DEVICE(0x10C4, 0x8341) }, /* Siemens MC35PU GPRS Modem */ | 108 | { USB_DEVICE(0x10C4, 0x8341) }, /* Siemens MC35PU GPRS Modem */ |
| 111 | { USB_DEVICE(0x10C4, 0x8382) }, /* Cygnal Integrated Products, Inc. */ | 109 | { USB_DEVICE(0x10C4, 0x8382) }, /* Cygnal Integrated Products, Inc. */ |
| 112 | { USB_DEVICE(0x10C4, 0x83A8) }, /* Amber Wireless AMB2560 */ | 110 | { USB_DEVICE(0x10C4, 0x83A8) }, /* Amber Wireless AMB2560 */ |
| 111 | { USB_DEVICE(0x10C4, 0x83D8) }, /* DekTec DTA Plus VHF/UHF Booster/Attenuator */ | ||
| 113 | { USB_DEVICE(0x10C4, 0x8411) }, /* Kyocera GPS Module */ | 112 | { USB_DEVICE(0x10C4, 0x8411) }, /* Kyocera GPS Module */ |
| 113 | { USB_DEVICE(0x10C4, 0x8418) }, /* IRZ Automation Teleport SG-10 GSM/GPRS Modem */ | ||
| 114 | { USB_DEVICE(0x10C4, 0x846E) }, /* BEI USB Sensor Interface (VCP) */ | 114 | { USB_DEVICE(0x10C4, 0x846E) }, /* BEI USB Sensor Interface (VCP) */ |
| 115 | { USB_DEVICE(0x10C4, 0x8477) }, /* Balluff RFID */ | 115 | { USB_DEVICE(0x10C4, 0x8477) }, /* Balluff RFID */ |
| 116 | { USB_DEVICE(0x10C4, 0xEA60) }, /* Silicon Labs factory default */ | 116 | { USB_DEVICE(0x10C4, 0xEA60) }, /* Silicon Labs factory default */ |
| @@ -165,8 +165,7 @@ static struct usb_serial_driver cp210x_device = { | |||
| 165 | .tiocmget = cp210x_tiocmget, | 165 | .tiocmget = cp210x_tiocmget, |
| 166 | .tiocmset = cp210x_tiocmset, | 166 | .tiocmset = cp210x_tiocmset, |
| 167 | .attach = cp210x_startup, | 167 | .attach = cp210x_startup, |
| 168 | .dtr_rts = cp210x_dtr_rts, | 168 | .dtr_rts = cp210x_dtr_rts |
| 169 | .carrier_raised = cp210x_carrier_raised | ||
| 170 | }; | 169 | }; |
| 171 | 170 | ||
| 172 | /* Config request types */ | 171 | /* Config request types */ |
| @@ -765,15 +764,6 @@ static int cp210x_tiocmget (struct tty_struct *tty, struct file *file) | |||
| 765 | return result; | 764 | return result; |
| 766 | } | 765 | } |
| 767 | 766 | ||
| 768 | static int cp210x_carrier_raised(struct usb_serial_port *p) | ||
| 769 | { | ||
| 770 | unsigned int control; | ||
| 771 | cp210x_get_config(p, CP210X_GET_MDMSTS, &control, 1); | ||
| 772 | if (control & CONTROL_DCD) | ||
| 773 | return 1; | ||
| 774 | return 0; | ||
| 775 | } | ||
| 776 | |||
| 777 | static void cp210x_break_ctl (struct tty_struct *tty, int break_state) | 767 | static void cp210x_break_ctl (struct tty_struct *tty, int break_state) |
| 778 | { | 768 | { |
| 779 | struct usb_serial_port *port = tty->driver_data; | 769 | struct usb_serial_port *port = tty->driver_data; |
diff --git a/drivers/usb/serial/digi_acceleport.c b/drivers/usb/serial/digi_acceleport.c index b92070c103c..666e5a6edd8 100644 --- a/drivers/usb/serial/digi_acceleport.c +++ b/drivers/usb/serial/digi_acceleport.c | |||
| @@ -455,7 +455,6 @@ static int digi_write_room(struct tty_struct *tty); | |||
| 455 | static int digi_chars_in_buffer(struct tty_struct *tty); | 455 | static int digi_chars_in_buffer(struct tty_struct *tty); |
| 456 | static int digi_open(struct tty_struct *tty, struct usb_serial_port *port); | 456 | static int digi_open(struct tty_struct *tty, struct usb_serial_port *port); |
| 457 | static void digi_close(struct usb_serial_port *port); | 457 | static void digi_close(struct usb_serial_port *port); |
| 458 | static int digi_carrier_raised(struct usb_serial_port *port); | ||
| 459 | static void digi_dtr_rts(struct usb_serial_port *port, int on); | 458 | static void digi_dtr_rts(struct usb_serial_port *port, int on); |
| 460 | static int digi_startup_device(struct usb_serial *serial); | 459 | static int digi_startup_device(struct usb_serial *serial); |
| 461 | static int digi_startup(struct usb_serial *serial); | 460 | static int digi_startup(struct usb_serial *serial); |
| @@ -511,7 +510,6 @@ static struct usb_serial_driver digi_acceleport_2_device = { | |||
| 511 | .open = digi_open, | 510 | .open = digi_open, |
| 512 | .close = digi_close, | 511 | .close = digi_close, |
| 513 | .dtr_rts = digi_dtr_rts, | 512 | .dtr_rts = digi_dtr_rts, |
| 514 | .carrier_raised = digi_carrier_raised, | ||
| 515 | .write = digi_write, | 513 | .write = digi_write, |
| 516 | .write_room = digi_write_room, | 514 | .write_room = digi_write_room, |
| 517 | .write_bulk_callback = digi_write_bulk_callback, | 515 | .write_bulk_callback = digi_write_bulk_callback, |
| @@ -1339,14 +1337,6 @@ static void digi_dtr_rts(struct usb_serial_port *port, int on) | |||
| 1339 | digi_set_modem_signals(port, on * (TIOCM_DTR|TIOCM_RTS), 1); | 1337 | digi_set_modem_signals(port, on * (TIOCM_DTR|TIOCM_RTS), 1); |
| 1340 | } | 1338 | } |
| 1341 | 1339 | ||
| 1342 | static int digi_carrier_raised(struct usb_serial_port *port) | ||
| 1343 | { | ||
| 1344 | struct digi_port *priv = usb_get_serial_port_data(port); | ||
| 1345 | if (priv->dp_modem_signals & TIOCM_CD) | ||
| 1346 | return 1; | ||
| 1347 | return 0; | ||
| 1348 | } | ||
| 1349 | |||
| 1350 | static int digi_open(struct tty_struct *tty, struct usb_serial_port *port) | 1340 | static int digi_open(struct tty_struct *tty, struct usb_serial_port *port) |
| 1351 | { | 1341 | { |
| 1352 | int ret; | 1342 | int ret; |
diff --git a/drivers/usb/serial/ftdi_sio.c b/drivers/usb/serial/ftdi_sio.c index a2668d08926..4787c0cd063 100644 --- a/drivers/usb/serial/ftdi_sio.c +++ b/drivers/usb/serial/ftdi_sio.c | |||
| @@ -676,7 +676,17 @@ static struct usb_device_id id_table_combined [] = { | |||
| 676 | { USB_DEVICE(FTDI_VID, FTDI_PCDJ_DAC2_PID) }, | 676 | { USB_DEVICE(FTDI_VID, FTDI_PCDJ_DAC2_PID) }, |
| 677 | { USB_DEVICE(FTDI_VID, FTDI_RRCIRKITS_LOCOBUFFER_PID) }, | 677 | { USB_DEVICE(FTDI_VID, FTDI_RRCIRKITS_LOCOBUFFER_PID) }, |
| 678 | { USB_DEVICE(FTDI_VID, FTDI_ASK_RDR400_PID) }, | 678 | { USB_DEVICE(FTDI_VID, FTDI_ASK_RDR400_PID) }, |
| 679 | { USB_DEVICE(ICOM_ID1_VID, ICOM_ID1_PID) }, | 679 | { USB_DEVICE(ICOM_VID, ICOM_ID_1_PID) }, |
| 680 | { USB_DEVICE(ICOM_VID, ICOM_OPC_U_UC_PID) }, | ||
| 681 | { USB_DEVICE(ICOM_VID, ICOM_ID_RP2C1_PID) }, | ||
| 682 | { USB_DEVICE(ICOM_VID, ICOM_ID_RP2C2_PID) }, | ||
| 683 | { USB_DEVICE(ICOM_VID, ICOM_ID_RP2D_PID) }, | ||
| 684 | { USB_DEVICE(ICOM_VID, ICOM_ID_RP2VT_PID) }, | ||
| 685 | { USB_DEVICE(ICOM_VID, ICOM_ID_RP2VR_PID) }, | ||
| 686 | { USB_DEVICE(ICOM_VID, ICOM_ID_RP4KVT_PID) }, | ||
| 687 | { USB_DEVICE(ICOM_VID, ICOM_ID_RP4KVR_PID) }, | ||
| 688 | { USB_DEVICE(ICOM_VID, ICOM_ID_RP2KVT_PID) }, | ||
| 689 | { USB_DEVICE(ICOM_VID, ICOM_ID_RP2KVR_PID) }, | ||
| 680 | { USB_DEVICE(FTDI_VID, FTDI_ACG_HFDUAL_PID) }, | 690 | { USB_DEVICE(FTDI_VID, FTDI_ACG_HFDUAL_PID) }, |
| 681 | { USB_DEVICE(FTDI_VID, FTDI_YEI_SERVOCENTER31_PID) }, | 691 | { USB_DEVICE(FTDI_VID, FTDI_YEI_SERVOCENTER31_PID) }, |
| 682 | { USB_DEVICE(FTDI_VID, FTDI_THORLABS_PID) }, | 692 | { USB_DEVICE(FTDI_VID, FTDI_THORLABS_PID) }, |
diff --git a/drivers/usb/serial/ftdi_sio_ids.h b/drivers/usb/serial/ftdi_sio_ids.h index bf086728548..ed160def858 100644 --- a/drivers/usb/serial/ftdi_sio_ids.h +++ b/drivers/usb/serial/ftdi_sio_ids.h | |||
| @@ -569,11 +569,23 @@ | |||
| 569 | #define OCT_US101_PID 0x0421 /* OCT US101 USB to RS-232 */ | 569 | #define OCT_US101_PID 0x0421 /* OCT US101 USB to RS-232 */ |
| 570 | 570 | ||
| 571 | /* | 571 | /* |
| 572 | * Icom ID-1 digital transceiver | 572 | * Definitions for Icom Inc. devices |
| 573 | */ | 573 | */ |
| 574 | 574 | #define ICOM_VID 0x0C26 /* Icom vendor ID */ | |
| 575 | #define ICOM_ID1_VID 0x0C26 | 575 | /* Note: ID-1 is a communications tranceiver for HAM-radio operators */ |
| 576 | #define ICOM_ID1_PID 0x0004 | 576 | #define ICOM_ID_1_PID 0x0004 /* ID-1 USB to RS-232 */ |
| 577 | /* Note: OPC is an Optional cable to connect an Icom Tranceiver */ | ||
| 578 | #define ICOM_OPC_U_UC_PID 0x0018 /* OPC-478UC, OPC-1122U cloning cable */ | ||
| 579 | /* Note: ID-RP* devices are Icom Repeater Devices for HAM-radio */ | ||
| 580 | #define ICOM_ID_RP2C1_PID 0x0009 /* ID-RP2C Asset 1 to RS-232 */ | ||
| 581 | #define ICOM_ID_RP2C2_PID 0x000A /* ID-RP2C Asset 2 to RS-232 */ | ||
| 582 | #define ICOM_ID_RP2D_PID 0x000B /* ID-RP2D configuration port*/ | ||
| 583 | #define ICOM_ID_RP2VT_PID 0x000C /* ID-RP2V Transmit config port */ | ||
| 584 | #define ICOM_ID_RP2VR_PID 0x000D /* ID-RP2V Receive config port */ | ||
| 585 | #define ICOM_ID_RP4KVT_PID 0x0010 /* ID-RP4000V Transmit config port */ | ||
| 586 | #define ICOM_ID_RP4KVR_PID 0x0011 /* ID-RP4000V Receive config port */ | ||
| 587 | #define ICOM_ID_RP2KVT_PID 0x0012 /* ID-RP2000V Transmit config port */ | ||
| 588 | #define ICOM_ID_RP2KVR_PID 0x0013 /* ID-RP2000V Receive config port */ | ||
| 577 | 589 | ||
| 578 | /* | 590 | /* |
| 579 | * GN Otometrics (http://www.otometrics.com) | 591 | * GN Otometrics (http://www.otometrics.com) |
diff --git a/drivers/usb/serial/generic.c b/drivers/usb/serial/generic.c index e6833e216fc..e4db5ad2bc5 100644 --- a/drivers/usb/serial/generic.c +++ b/drivers/usb/serial/generic.c | |||
| @@ -479,6 +479,26 @@ int usb_serial_handle_break(struct usb_serial_port *port) | |||
| 479 | } | 479 | } |
| 480 | EXPORT_SYMBOL_GPL(usb_serial_handle_break); | 480 | EXPORT_SYMBOL_GPL(usb_serial_handle_break); |
| 481 | 481 | ||
| 482 | /** | ||
| 483 | * usb_serial_handle_dcd_change - handle a change of carrier detect state | ||
| 484 | * @port: usb_serial_port structure for the open port | ||
| 485 | * @tty: tty_struct structure for the port | ||
| 486 | * @status: new carrier detect status, nonzero if active | ||
| 487 | */ | ||
| 488 | void usb_serial_handle_dcd_change(struct usb_serial_port *usb_port, | ||
| 489 | struct tty_struct *tty, unsigned int status) | ||
| 490 | { | ||
| 491 | struct tty_port *port = &usb_port->port; | ||
| 492 | |||
| 493 | dbg("%s - port %d, status %d", __func__, usb_port->number, status); | ||
| 494 | |||
| 495 | if (status) | ||
| 496 | wake_up_interruptible(&port->open_wait); | ||
| 497 | else if (tty && !C_CLOCAL(tty)) | ||
| 498 | tty_hangup(tty); | ||
| 499 | } | ||
| 500 | EXPORT_SYMBOL_GPL(usb_serial_handle_dcd_change); | ||
| 501 | |||
| 482 | int usb_serial_generic_resume(struct usb_serial *serial) | 502 | int usb_serial_generic_resume(struct usb_serial *serial) |
| 483 | { | 503 | { |
| 484 | struct usb_serial_port *port; | 504 | struct usb_serial_port *port; |
diff --git a/drivers/usb/serial/io_tables.h b/drivers/usb/serial/io_tables.h index 6ab2a3f97fe..178b22eb32b 100644 --- a/drivers/usb/serial/io_tables.h +++ b/drivers/usb/serial/io_tables.h | |||
| @@ -199,6 +199,7 @@ static struct usb_serial_driver epic_device = { | |||
| 199 | .name = "epic", | 199 | .name = "epic", |
| 200 | }, | 200 | }, |
| 201 | .description = "EPiC device", | 201 | .description = "EPiC device", |
| 202 | .usb_driver = &io_driver, | ||
| 202 | .id_table = Epic_port_id_table, | 203 | .id_table = Epic_port_id_table, |
| 203 | .num_ports = 1, | 204 | .num_ports = 1, |
| 204 | .open = edge_open, | 205 | .open = edge_open, |
diff --git a/drivers/usb/serial/iuu_phoenix.c b/drivers/usb/serial/iuu_phoenix.c index 12ed594f5f8..99b97c04896 100644 --- a/drivers/usb/serial/iuu_phoenix.c +++ b/drivers/usb/serial/iuu_phoenix.c | |||
| @@ -1275,6 +1275,7 @@ static struct usb_serial_driver iuu_device = { | |||
| 1275 | .name = "iuu_phoenix", | 1275 | .name = "iuu_phoenix", |
| 1276 | }, | 1276 | }, |
| 1277 | .id_table = id_table, | 1277 | .id_table = id_table, |
| 1278 | .usb_driver = &iuu_driver, | ||
| 1278 | .num_ports = 1, | 1279 | .num_ports = 1, |
| 1279 | .bulk_in_size = 512, | 1280 | .bulk_in_size = 512, |
| 1280 | .bulk_out_size = 512, | 1281 | .bulk_out_size = 512, |
diff --git a/drivers/usb/serial/keyspan.h b/drivers/usb/serial/keyspan.h index 2d8baf6ac47..ce134dc28dd 100644 --- a/drivers/usb/serial/keyspan.h +++ b/drivers/usb/serial/keyspan.h | |||
| @@ -546,6 +546,7 @@ static struct usb_serial_driver keyspan_pre_device = { | |||
| 546 | .name = "keyspan_no_firm", | 546 | .name = "keyspan_no_firm", |
| 547 | }, | 547 | }, |
| 548 | .description = "Keyspan - (without firmware)", | 548 | .description = "Keyspan - (without firmware)", |
| 549 | .usb_driver = &keyspan_driver, | ||
| 549 | .id_table = keyspan_pre_ids, | 550 | .id_table = keyspan_pre_ids, |
| 550 | .num_ports = 1, | 551 | .num_ports = 1, |
| 551 | .attach = keyspan_fake_startup, | 552 | .attach = keyspan_fake_startup, |
| @@ -557,6 +558,7 @@ static struct usb_serial_driver keyspan_1port_device = { | |||
| 557 | .name = "keyspan_1", | 558 | .name = "keyspan_1", |
| 558 | }, | 559 | }, |
| 559 | .description = "Keyspan 1 port adapter", | 560 | .description = "Keyspan 1 port adapter", |
| 561 | .usb_driver = &keyspan_driver, | ||
| 560 | .id_table = keyspan_1port_ids, | 562 | .id_table = keyspan_1port_ids, |
| 561 | .num_ports = 1, | 563 | .num_ports = 1, |
| 562 | .open = keyspan_open, | 564 | .open = keyspan_open, |
| @@ -579,6 +581,7 @@ static struct usb_serial_driver keyspan_2port_device = { | |||
| 579 | .name = "keyspan_2", | 581 | .name = "keyspan_2", |
| 580 | }, | 582 | }, |
| 581 | .description = "Keyspan 2 port adapter", | 583 | .description = "Keyspan 2 port adapter", |
| 584 | .usb_driver = &keyspan_driver, | ||
| 582 | .id_table = keyspan_2port_ids, | 585 | .id_table = keyspan_2port_ids, |
| 583 | .num_ports = 2, | 586 | .num_ports = 2, |
| 584 | .open = keyspan_open, | 587 | .open = keyspan_open, |
| @@ -601,6 +604,7 @@ static struct usb_serial_driver keyspan_4port_device = { | |||
| 601 | .name = "keyspan_4", | 604 | .name = "keyspan_4", |
| 602 | }, | 605 | }, |
| 603 | .description = "Keyspan 4 port adapter", | 606 | .description = "Keyspan 4 port adapter", |
| 607 | .usb_driver = &keyspan_driver, | ||
| 604 | .id_table = keyspan_4port_ids, | 608 | .id_table = keyspan_4port_ids, |
| 605 | .num_ports = 4, | 609 | .num_ports = 4, |
| 606 | .open = keyspan_open, | 610 | .open = keyspan_open, |
diff --git a/drivers/usb/serial/keyspan_pda.c b/drivers/usb/serial/keyspan_pda.c index a10dd5676cc..554a8693a46 100644 --- a/drivers/usb/serial/keyspan_pda.c +++ b/drivers/usb/serial/keyspan_pda.c | |||
| @@ -679,22 +679,6 @@ static void keyspan_pda_dtr_rts(struct usb_serial_port *port, int on) | |||
| 679 | } | 679 | } |
| 680 | } | 680 | } |
| 681 | 681 | ||
| 682 | static int keyspan_pda_carrier_raised(struct usb_serial_port *port) | ||
| 683 | { | ||
| 684 | struct usb_serial *serial = port->serial; | ||
| 685 | unsigned char modembits; | ||
| 686 | |||
| 687 | /* If we can read the modem status and the DCD is low then | ||
| 688 | carrier is not raised yet */ | ||
| 689 | if (keyspan_pda_get_modem_info(serial, &modembits) >= 0) { | ||
| 690 | if (!(modembits & (1>>6))) | ||
| 691 | return 0; | ||
| 692 | } | ||
| 693 | /* Carrier raised, or we failed (eg disconnected) so | ||
| 694 | progress accordingly */ | ||
| 695 | return 1; | ||
| 696 | } | ||
| 697 | |||
| 698 | 682 | ||
| 699 | static int keyspan_pda_open(struct tty_struct *tty, | 683 | static int keyspan_pda_open(struct tty_struct *tty, |
| 700 | struct usb_serial_port *port) | 684 | struct usb_serial_port *port) |
| @@ -881,7 +865,6 @@ static struct usb_serial_driver keyspan_pda_device = { | |||
| 881 | .id_table = id_table_std, | 865 | .id_table = id_table_std, |
| 882 | .num_ports = 1, | 866 | .num_ports = 1, |
| 883 | .dtr_rts = keyspan_pda_dtr_rts, | 867 | .dtr_rts = keyspan_pda_dtr_rts, |
| 884 | .carrier_raised = keyspan_pda_carrier_raised, | ||
| 885 | .open = keyspan_pda_open, | 868 | .open = keyspan_pda_open, |
| 886 | .close = keyspan_pda_close, | 869 | .close = keyspan_pda_close, |
| 887 | .write = keyspan_pda_write, | 870 | .write = keyspan_pda_write, |
diff --git a/drivers/usb/serial/moto_modem.c b/drivers/usb/serial/moto_modem.c index cf1718394e1..653465f61d4 100644 --- a/drivers/usb/serial/moto_modem.c +++ b/drivers/usb/serial/moto_modem.c | |||
| @@ -44,6 +44,7 @@ static struct usb_serial_driver moto_device = { | |||
| 44 | .name = "moto-modem", | 44 | .name = "moto-modem", |
| 45 | }, | 45 | }, |
| 46 | .id_table = id_table, | 46 | .id_table = id_table, |
| 47 | .usb_driver = &moto_driver, | ||
| 47 | .num_ports = 1, | 48 | .num_ports = 1, |
| 48 | }; | 49 | }; |
| 49 | 50 | ||
diff --git a/drivers/usb/serial/option.c b/drivers/usb/serial/option.c index 748778288d9..5f46838dfee 100644 --- a/drivers/usb/serial/option.c +++ b/drivers/usb/serial/option.c | |||
| @@ -382,7 +382,16 @@ static void option_instat_callback(struct urb *urb); | |||
| 382 | #define HAIER_VENDOR_ID 0x201e | 382 | #define HAIER_VENDOR_ID 0x201e |
| 383 | #define HAIER_PRODUCT_CE100 0x2009 | 383 | #define HAIER_PRODUCT_CE100 0x2009 |
| 384 | 384 | ||
| 385 | #define CINTERION_VENDOR_ID 0x0681 | 385 | /* Cinterion (formerly Siemens) products */ |
| 386 | #define SIEMENS_VENDOR_ID 0x0681 | ||
| 387 | #define CINTERION_VENDOR_ID 0x1e2d | ||
| 388 | #define CINTERION_PRODUCT_HC25_MDM 0x0047 | ||
| 389 | #define CINTERION_PRODUCT_HC25_MDMNET 0x0040 | ||
| 390 | #define CINTERION_PRODUCT_HC28_MDM 0x004C | ||
| 391 | #define CINTERION_PRODUCT_HC28_MDMNET 0x004A /* same for HC28J */ | ||
| 392 | #define CINTERION_PRODUCT_EU3_E 0x0051 | ||
| 393 | #define CINTERION_PRODUCT_EU3_P 0x0052 | ||
| 394 | #define CINTERION_PRODUCT_PH8 0x0053 | ||
| 386 | 395 | ||
| 387 | /* Olivetti products */ | 396 | /* Olivetti products */ |
| 388 | #define OLIVETTI_VENDOR_ID 0x0b3c | 397 | #define OLIVETTI_VENDOR_ID 0x0b3c |
| @@ -944,7 +953,17 @@ static const struct usb_device_id option_ids[] = { | |||
| 944 | { USB_DEVICE(PIRELLI_VENDOR_ID, PIRELLI_PRODUCT_100F) }, | 953 | { USB_DEVICE(PIRELLI_VENDOR_ID, PIRELLI_PRODUCT_100F) }, |
| 945 | { USB_DEVICE(PIRELLI_VENDOR_ID, PIRELLI_PRODUCT_1011)}, | 954 | { USB_DEVICE(PIRELLI_VENDOR_ID, PIRELLI_PRODUCT_1011)}, |
| 946 | { USB_DEVICE(PIRELLI_VENDOR_ID, PIRELLI_PRODUCT_1012)}, | 955 | { USB_DEVICE(PIRELLI_VENDOR_ID, PIRELLI_PRODUCT_1012)}, |
| 947 | { USB_DEVICE(CINTERION_VENDOR_ID, 0x0047) }, | 956 | /* Cinterion */ |
| 957 | { USB_DEVICE(CINTERION_VENDOR_ID, CINTERION_PRODUCT_EU3_E) }, | ||
| 958 | { USB_DEVICE(CINTERION_VENDOR_ID, CINTERION_PRODUCT_EU3_P) }, | ||
| 959 | { USB_DEVICE(CINTERION_VENDOR_ID, CINTERION_PRODUCT_PH8) }, | ||
| 960 | { USB_DEVICE(CINTERION_VENDOR_ID, CINTERION_PRODUCT_HC28_MDM) }, | ||
| 961 | { USB_DEVICE(CINTERION_VENDOR_ID, CINTERION_PRODUCT_HC28_MDMNET) }, | ||
| 962 | { USB_DEVICE(SIEMENS_VENDOR_ID, CINTERION_PRODUCT_HC25_MDM) }, | ||
| 963 | { USB_DEVICE(SIEMENS_VENDOR_ID, CINTERION_PRODUCT_HC25_MDMNET) }, | ||
| 964 | { USB_DEVICE(SIEMENS_VENDOR_ID, CINTERION_PRODUCT_HC28_MDM) }, /* HC28 enumerates with Siemens or Cinterion VID depending on FW revision */ | ||
| 965 | { USB_DEVICE(SIEMENS_VENDOR_ID, CINTERION_PRODUCT_HC28_MDMNET) }, | ||
| 966 | |||
| 948 | { USB_DEVICE(OLIVETTI_VENDOR_ID, OLIVETTI_PRODUCT_OLICARD100) }, | 967 | { USB_DEVICE(OLIVETTI_VENDOR_ID, OLIVETTI_PRODUCT_OLICARD100) }, |
| 949 | { USB_DEVICE(CELOT_VENDOR_ID, CELOT_PRODUCT_CT680M) }, /* CT-650 CDMA 450 1xEVDO modem */ | 968 | { USB_DEVICE(CELOT_VENDOR_ID, CELOT_PRODUCT_CT680M) }, /* CT-650 CDMA 450 1xEVDO modem */ |
| 950 | { USB_DEVICE(ONDA_VENDOR_ID, ONDA_MT825UP) }, /* ONDA MT825UP modem */ | 969 | { USB_DEVICE(ONDA_VENDOR_ID, ONDA_MT825UP) }, /* ONDA MT825UP modem */ |
diff --git a/drivers/usb/serial/oti6858.c b/drivers/usb/serial/oti6858.c index 5be866bb7a4..73613205be7 100644 --- a/drivers/usb/serial/oti6858.c +++ b/drivers/usb/serial/oti6858.c | |||
| @@ -157,6 +157,7 @@ static struct usb_serial_driver oti6858_device = { | |||
| 157 | .name = "oti6858", | 157 | .name = "oti6858", |
| 158 | }, | 158 | }, |
| 159 | .id_table = id_table, | 159 | .id_table = id_table, |
| 160 | .usb_driver = &oti6858_driver, | ||
| 160 | .num_ports = 1, | 161 | .num_ports = 1, |
| 161 | .open = oti6858_open, | 162 | .open = oti6858_open, |
| 162 | .close = oti6858_close, | 163 | .close = oti6858_close, |
diff --git a/drivers/usb/serial/pl2303.c b/drivers/usb/serial/pl2303.c index 8ae4c6cbc38..08c9181b8e4 100644 --- a/drivers/usb/serial/pl2303.c +++ b/drivers/usb/serial/pl2303.c | |||
| @@ -50,6 +50,7 @@ static const struct usb_device_id id_table[] = { | |||
| 50 | { USB_DEVICE(PL2303_VENDOR_ID, PL2303_PRODUCT_ID_MMX) }, | 50 | { USB_DEVICE(PL2303_VENDOR_ID, PL2303_PRODUCT_ID_MMX) }, |
| 51 | { USB_DEVICE(PL2303_VENDOR_ID, PL2303_PRODUCT_ID_GPRS) }, | 51 | { USB_DEVICE(PL2303_VENDOR_ID, PL2303_PRODUCT_ID_GPRS) }, |
| 52 | { USB_DEVICE(PL2303_VENDOR_ID, PL2303_PRODUCT_ID_HCR331) }, | 52 | { USB_DEVICE(PL2303_VENDOR_ID, PL2303_PRODUCT_ID_HCR331) }, |
| 53 | { USB_DEVICE(PL2303_VENDOR_ID, PL2303_PRODUCT_ID_MOTOROLA) }, | ||
| 53 | { USB_DEVICE(IODATA_VENDOR_ID, IODATA_PRODUCT_ID) }, | 54 | { USB_DEVICE(IODATA_VENDOR_ID, IODATA_PRODUCT_ID) }, |
| 54 | { USB_DEVICE(IODATA_VENDOR_ID, IODATA_PRODUCT_ID_RSAQ5) }, | 55 | { USB_DEVICE(IODATA_VENDOR_ID, IODATA_PRODUCT_ID_RSAQ5) }, |
| 55 | { USB_DEVICE(ATEN_VENDOR_ID, ATEN_PRODUCT_ID) }, | 56 | { USB_DEVICE(ATEN_VENDOR_ID, ATEN_PRODUCT_ID) }, |
| @@ -677,9 +678,11 @@ static void pl2303_update_line_status(struct usb_serial_port *port, | |||
| 677 | { | 678 | { |
| 678 | 679 | ||
| 679 | struct pl2303_private *priv = usb_get_serial_port_data(port); | 680 | struct pl2303_private *priv = usb_get_serial_port_data(port); |
| 681 | struct tty_struct *tty; | ||
| 680 | unsigned long flags; | 682 | unsigned long flags; |
| 681 | u8 status_idx = UART_STATE; | 683 | u8 status_idx = UART_STATE; |
| 682 | u8 length = UART_STATE + 1; | 684 | u8 length = UART_STATE + 1; |
| 685 | u8 prev_line_status; | ||
| 683 | u16 idv, idp; | 686 | u16 idv, idp; |
| 684 | 687 | ||
| 685 | idv = le16_to_cpu(port->serial->dev->descriptor.idVendor); | 688 | idv = le16_to_cpu(port->serial->dev->descriptor.idVendor); |
| @@ -701,11 +704,20 @@ static void pl2303_update_line_status(struct usb_serial_port *port, | |||
| 701 | 704 | ||
| 702 | /* Save off the uart status for others to look at */ | 705 | /* Save off the uart status for others to look at */ |
| 703 | spin_lock_irqsave(&priv->lock, flags); | 706 | spin_lock_irqsave(&priv->lock, flags); |
| 707 | prev_line_status = priv->line_status; | ||
| 704 | priv->line_status = data[status_idx]; | 708 | priv->line_status = data[status_idx]; |
| 705 | spin_unlock_irqrestore(&priv->lock, flags); | 709 | spin_unlock_irqrestore(&priv->lock, flags); |
| 706 | if (priv->line_status & UART_BREAK_ERROR) | 710 | if (priv->line_status & UART_BREAK_ERROR) |
| 707 | usb_serial_handle_break(port); | 711 | usb_serial_handle_break(port); |
| 708 | wake_up_interruptible(&priv->delta_msr_wait); | 712 | wake_up_interruptible(&priv->delta_msr_wait); |
| 713 | |||
| 714 | tty = tty_port_tty_get(&port->port); | ||
| 715 | if (!tty) | ||
| 716 | return; | ||
| 717 | if ((priv->line_status ^ prev_line_status) & UART_DCD) | ||
| 718 | usb_serial_handle_dcd_change(port, tty, | ||
| 719 | priv->line_status & UART_DCD); | ||
| 720 | tty_kref_put(tty); | ||
| 709 | } | 721 | } |
| 710 | 722 | ||
| 711 | static void pl2303_read_int_callback(struct urb *urb) | 723 | static void pl2303_read_int_callback(struct urb *urb) |
diff --git a/drivers/usb/serial/pl2303.h b/drivers/usb/serial/pl2303.h index 43eb9bdad42..1b025f75daf 100644 --- a/drivers/usb/serial/pl2303.h +++ b/drivers/usb/serial/pl2303.h | |||
| @@ -21,6 +21,7 @@ | |||
| 21 | #define PL2303_PRODUCT_ID_MMX 0x0612 | 21 | #define PL2303_PRODUCT_ID_MMX 0x0612 |
| 22 | #define PL2303_PRODUCT_ID_GPRS 0x0609 | 22 | #define PL2303_PRODUCT_ID_GPRS 0x0609 |
| 23 | #define PL2303_PRODUCT_ID_HCR331 0x331a | 23 | #define PL2303_PRODUCT_ID_HCR331 0x331a |
| 24 | #define PL2303_PRODUCT_ID_MOTOROLA 0x0307 | ||
| 24 | 25 | ||
| 25 | #define ATEN_VENDOR_ID 0x0557 | 26 | #define ATEN_VENDOR_ID 0x0557 |
| 26 | #define ATEN_VENDOR_ID2 0x0547 | 27 | #define ATEN_VENDOR_ID2 0x0547 |
diff --git a/drivers/usb/serial/qcaux.c b/drivers/usb/serial/qcaux.c index 214a3e50429..30b73e68a90 100644 --- a/drivers/usb/serial/qcaux.c +++ b/drivers/usb/serial/qcaux.c | |||
| @@ -36,6 +36,7 @@ | |||
| 36 | #define UTSTARCOM_PRODUCT_UM175_V1 0x3712 | 36 | #define UTSTARCOM_PRODUCT_UM175_V1 0x3712 |
| 37 | #define UTSTARCOM_PRODUCT_UM175_V2 0x3714 | 37 | #define UTSTARCOM_PRODUCT_UM175_V2 0x3714 |
| 38 | #define UTSTARCOM_PRODUCT_UM175_ALLTEL 0x3715 | 38 | #define UTSTARCOM_PRODUCT_UM175_ALLTEL 0x3715 |
| 39 | #define PANTECH_PRODUCT_UML290_VZW 0x3718 | ||
| 39 | 40 | ||
| 40 | /* CMOTECH devices */ | 41 | /* CMOTECH devices */ |
| 41 | #define CMOTECH_VENDOR_ID 0x16d8 | 42 | #define CMOTECH_VENDOR_ID 0x16d8 |
| @@ -66,6 +67,7 @@ static struct usb_device_id id_table[] = { | |||
| 66 | { USB_DEVICE_AND_INTERFACE_INFO(LG_VENDOR_ID, LG_PRODUCT_VX4400_6000, 0xff, 0xff, 0x00) }, | 67 | { USB_DEVICE_AND_INTERFACE_INFO(LG_VENDOR_ID, LG_PRODUCT_VX4400_6000, 0xff, 0xff, 0x00) }, |
| 67 | { USB_DEVICE_AND_INTERFACE_INFO(SANYO_VENDOR_ID, SANYO_PRODUCT_KATANA_LX, 0xff, 0xff, 0x00) }, | 68 | { USB_DEVICE_AND_INTERFACE_INFO(SANYO_VENDOR_ID, SANYO_PRODUCT_KATANA_LX, 0xff, 0xff, 0x00) }, |
| 68 | { USB_DEVICE_AND_INTERFACE_INFO(SAMSUNG_VENDOR_ID, SAMSUNG_PRODUCT_U520, 0xff, 0x00, 0x00) }, | 69 | { USB_DEVICE_AND_INTERFACE_INFO(SAMSUNG_VENDOR_ID, SAMSUNG_PRODUCT_U520, 0xff, 0x00, 0x00) }, |
| 70 | { USB_DEVICE_AND_INTERFACE_INFO(UTSTARCOM_VENDOR_ID, PANTECH_PRODUCT_UML290_VZW, 0xff, 0xff, 0xff) }, | ||
| 69 | { }, | 71 | { }, |
| 70 | }; | 72 | }; |
| 71 | MODULE_DEVICE_TABLE(usb, id_table); | 73 | MODULE_DEVICE_TABLE(usb, id_table); |
| @@ -84,6 +86,7 @@ static struct usb_serial_driver qcaux_device = { | |||
| 84 | .name = "qcaux", | 86 | .name = "qcaux", |
| 85 | }, | 87 | }, |
| 86 | .id_table = id_table, | 88 | .id_table = id_table, |
| 89 | .usb_driver = &qcaux_driver, | ||
| 87 | .num_ports = 1, | 90 | .num_ports = 1, |
| 88 | }; | 91 | }; |
| 89 | 92 | ||
diff --git a/drivers/usb/serial/siemens_mpi.c b/drivers/usb/serial/siemens_mpi.c index cb8195cabfd..74cd4ccdb3f 100644 --- a/drivers/usb/serial/siemens_mpi.c +++ b/drivers/usb/serial/siemens_mpi.c | |||
| @@ -42,6 +42,7 @@ static struct usb_serial_driver siemens_usb_mpi_device = { | |||
| 42 | .name = "siemens_mpi", | 42 | .name = "siemens_mpi", |
| 43 | }, | 43 | }, |
| 44 | .id_table = id_table, | 44 | .id_table = id_table, |
| 45 | .usb_driver = &siemens_usb_mpi_driver, | ||
| 45 | .num_ports = 1, | 46 | .num_ports = 1, |
| 46 | }; | 47 | }; |
| 47 | 48 | ||
diff --git a/drivers/usb/serial/spcp8x5.c b/drivers/usb/serial/spcp8x5.c index 765aa983bf5..cbfb70bffdd 100644 --- a/drivers/usb/serial/spcp8x5.c +++ b/drivers/usb/serial/spcp8x5.c | |||
| @@ -133,7 +133,7 @@ struct spcp8x5_usb_ctrl_arg { | |||
| 133 | 133 | ||
| 134 | /* how come ??? */ | 134 | /* how come ??? */ |
| 135 | #define UART_STATE 0x08 | 135 | #define UART_STATE 0x08 |
| 136 | #define UART_STATE_TRANSIENT_MASK 0x74 | 136 | #define UART_STATE_TRANSIENT_MASK 0x75 |
| 137 | #define UART_DCD 0x01 | 137 | #define UART_DCD 0x01 |
| 138 | #define UART_DSR 0x02 | 138 | #define UART_DSR 0x02 |
| 139 | #define UART_BREAK_ERROR 0x04 | 139 | #define UART_BREAK_ERROR 0x04 |
| @@ -525,6 +525,10 @@ static void spcp8x5_process_read_urb(struct urb *urb) | |||
| 525 | /* overrun is special, not associated with a char */ | 525 | /* overrun is special, not associated with a char */ |
| 526 | if (status & UART_OVERRUN_ERROR) | 526 | if (status & UART_OVERRUN_ERROR) |
| 527 | tty_insert_flip_char(tty, 0, TTY_OVERRUN); | 527 | tty_insert_flip_char(tty, 0, TTY_OVERRUN); |
| 528 | |||
| 529 | if (status & UART_DCD) | ||
| 530 | usb_serial_handle_dcd_change(port, tty, | ||
| 531 | priv->line_status & MSR_STATUS_LINE_DCD); | ||
| 528 | } | 532 | } |
| 529 | 533 | ||
| 530 | tty_insert_flip_string_fixed_flag(tty, data, tty_flag, | 534 | tty_insert_flip_string_fixed_flag(tty, data, tty_flag, |
| @@ -645,6 +649,7 @@ static struct usb_serial_driver spcp8x5_device = { | |||
| 645 | .name = "SPCP8x5", | 649 | .name = "SPCP8x5", |
| 646 | }, | 650 | }, |
| 647 | .id_table = id_table, | 651 | .id_table = id_table, |
| 652 | .usb_driver = &spcp8x5_driver, | ||
| 648 | .num_ports = 1, | 653 | .num_ports = 1, |
| 649 | .open = spcp8x5_open, | 654 | .open = spcp8x5_open, |
| 650 | .dtr_rts = spcp8x5_dtr_rts, | 655 | .dtr_rts = spcp8x5_dtr_rts, |
diff --git a/drivers/usb/serial/usb-serial.c b/drivers/usb/serial/usb-serial.c index 6954de50c0f..546a52179be 100644 --- a/drivers/usb/serial/usb-serial.c +++ b/drivers/usb/serial/usb-serial.c | |||
| @@ -1344,11 +1344,15 @@ int usb_serial_register(struct usb_serial_driver *driver) | |||
| 1344 | return -ENODEV; | 1344 | return -ENODEV; |
| 1345 | 1345 | ||
| 1346 | fixup_generic(driver); | 1346 | fixup_generic(driver); |
| 1347 | if (driver->usb_driver) | ||
| 1348 | driver->usb_driver->supports_autosuspend = 1; | ||
| 1349 | 1347 | ||
| 1350 | if (!driver->description) | 1348 | if (!driver->description) |
| 1351 | driver->description = driver->driver.name; | 1349 | driver->description = driver->driver.name; |
| 1350 | if (!driver->usb_driver) { | ||
| 1351 | WARN(1, "Serial driver %s has no usb_driver\n", | ||
| 1352 | driver->description); | ||
| 1353 | return -EINVAL; | ||
| 1354 | } | ||
| 1355 | driver->usb_driver->supports_autosuspend = 1; | ||
| 1352 | 1356 | ||
| 1353 | /* Add this device to our list of devices */ | 1357 | /* Add this device to our list of devices */ |
| 1354 | mutex_lock(&table_lock); | 1358 | mutex_lock(&table_lock); |
diff --git a/drivers/usb/serial/usb_debug.c b/drivers/usb/serial/usb_debug.c index f2ed6a31be7..95a82148ee8 100644 --- a/drivers/usb/serial/usb_debug.c +++ b/drivers/usb/serial/usb_debug.c | |||
| @@ -75,6 +75,7 @@ static struct usb_serial_driver debug_device = { | |||
| 75 | .name = "debug", | 75 | .name = "debug", |
| 76 | }, | 76 | }, |
| 77 | .id_table = id_table, | 77 | .id_table = id_table, |
| 78 | .usb_driver = &debug_driver, | ||
| 78 | .num_ports = 1, | 79 | .num_ports = 1, |
| 79 | .bulk_out_size = USB_DEBUG_MAX_PACKET_SIZE, | 80 | .bulk_out_size = USB_DEBUG_MAX_PACKET_SIZE, |
| 80 | .break_ctl = usb_debug_break_ctl, | 81 | .break_ctl = usb_debug_break_ctl, |
diff --git a/drivers/usb/storage/unusual_cypress.h b/drivers/usb/storage/unusual_cypress.h index c854fdebe0a..2c855302622 100644 --- a/drivers/usb/storage/unusual_cypress.h +++ b/drivers/usb/storage/unusual_cypress.h | |||
| @@ -31,4 +31,9 @@ UNUSUAL_DEV( 0x04b4, 0x6831, 0x0000, 0x9999, | |||
| 31 | "Cypress ISD-300LP", | 31 | "Cypress ISD-300LP", |
| 32 | USB_SC_CYP_ATACB, USB_PR_DEVICE, NULL, 0), | 32 | USB_SC_CYP_ATACB, USB_PR_DEVICE, NULL, 0), |
| 33 | 33 | ||
| 34 | UNUSUAL_DEV( 0x14cd, 0x6116, 0x0000, 0x9999, | ||
| 35 | "Super Top", | ||
| 36 | "USB 2.0 SATA BRIDGE", | ||
| 37 | USB_SC_CYP_ATACB, USB_PR_DEVICE, NULL, 0), | ||
| 38 | |||
| 34 | #endif /* defined(CONFIG_USB_STORAGE_CYPRESS_ATACB) || ... */ | 39 | #endif /* defined(CONFIG_USB_STORAGE_CYPRESS_ATACB) || ... */ |
diff --git a/drivers/usb/storage/unusual_devs.h b/drivers/usb/storage/unusual_devs.h index fcc1e32ce25..24bd5d7c3de 100644 --- a/drivers/usb/storage/unusual_devs.h +++ b/drivers/usb/storage/unusual_devs.h | |||
| @@ -1044,6 +1044,15 @@ UNUSUAL_DEV( 0x084d, 0x0011, 0x0110, 0x0110, | |||
| 1044 | USB_SC_DEVICE, USB_PR_DEVICE, NULL, | 1044 | USB_SC_DEVICE, USB_PR_DEVICE, NULL, |
| 1045 | US_FL_BULK32), | 1045 | US_FL_BULK32), |
| 1046 | 1046 | ||
| 1047 | /* Reported by <ttkspam@free.fr> | ||
| 1048 | * The device reports a vendor-specific device class, requiring an | ||
| 1049 | * explicit vendor/product match. | ||
| 1050 | */ | ||
| 1051 | UNUSUAL_DEV( 0x0851, 0x1542, 0x0002, 0x0002, | ||
| 1052 | "MagicPixel", | ||
| 1053 | "FW_Omega2", | ||
| 1054 | USB_SC_DEVICE, USB_PR_DEVICE, NULL, 0), | ||
| 1055 | |||
| 1047 | /* Andrew Lunn <andrew@lunn.ch> | 1056 | /* Andrew Lunn <andrew@lunn.ch> |
| 1048 | * PanDigital Digital Picture Frame. Does not like ALLOW_MEDIUM_REMOVAL | 1057 | * PanDigital Digital Picture Frame. Does not like ALLOW_MEDIUM_REMOVAL |
| 1049 | * on LUN 4. | 1058 | * on LUN 4. |
| @@ -1872,6 +1881,15 @@ UNUSUAL_DEV( 0x1908, 0x3335, 0x0200, 0x0200, | |||
| 1872 | USB_SC_DEVICE, USB_PR_DEVICE, NULL, | 1881 | USB_SC_DEVICE, USB_PR_DEVICE, NULL, |
| 1873 | US_FL_NO_READ_DISC_INFO ), | 1882 | US_FL_NO_READ_DISC_INFO ), |
| 1874 | 1883 | ||
| 1884 | /* Patch by Richard Schütz <r.schtz@t-online.de> | ||
| 1885 | * This external hard drive enclosure uses a JMicron chip which | ||
| 1886 | * needs the US_FL_IGNORE_RESIDUE flag to work properly. */ | ||
| 1887 | UNUSUAL_DEV( 0x1e68, 0x001b, 0x0000, 0x0000, | ||
| 1888 | "TrekStor GmbH & Co. KG", | ||
| 1889 | "DataStation maxi g.u", | ||
| 1890 | USB_SC_DEVICE, USB_PR_DEVICE, NULL, | ||
| 1891 | US_FL_IGNORE_RESIDUE | US_FL_SANE_SENSE ), | ||
| 1892 | |||
| 1875 | UNUSUAL_DEV( 0x2116, 0x0320, 0x0001, 0x0001, | 1893 | UNUSUAL_DEV( 0x2116, 0x0320, 0x0001, 0x0001, |
| 1876 | "ST", | 1894 | "ST", |
| 1877 | "2A", | 1895 | "2A", |
diff --git a/drivers/vhost/net.c b/drivers/vhost/net.c index 9b3ca103135..f616cefc95b 100644 --- a/drivers/vhost/net.c +++ b/drivers/vhost/net.c | |||
| @@ -128,8 +128,7 @@ static void handle_tx(struct vhost_net *net) | |||
| 128 | size_t hdr_size; | 128 | size_t hdr_size; |
| 129 | struct socket *sock; | 129 | struct socket *sock; |
| 130 | 130 | ||
| 131 | /* TODO: check that we are running from vhost_worker? | 131 | /* TODO: check that we are running from vhost_worker? */ |
| 132 | * Not sure it's worth it, it's straight-forward enough. */ | ||
| 133 | sock = rcu_dereference_check(vq->private_data, 1); | 132 | sock = rcu_dereference_check(vq->private_data, 1); |
| 134 | if (!sock) | 133 | if (!sock) |
| 135 | return; | 134 | return; |
| @@ -306,7 +305,8 @@ static void handle_rx_big(struct vhost_net *net) | |||
| 306 | size_t len, total_len = 0; | 305 | size_t len, total_len = 0; |
| 307 | int err; | 306 | int err; |
| 308 | size_t hdr_size; | 307 | size_t hdr_size; |
| 309 | struct socket *sock = rcu_dereference(vq->private_data); | 308 | /* TODO: check that we are running from vhost_worker? */ |
| 309 | struct socket *sock = rcu_dereference_check(vq->private_data, 1); | ||
| 310 | if (!sock || skb_queue_empty(&sock->sk->sk_receive_queue)) | 310 | if (!sock || skb_queue_empty(&sock->sk->sk_receive_queue)) |
| 311 | return; | 311 | return; |
| 312 | 312 | ||
| @@ -415,7 +415,8 @@ static void handle_rx_mergeable(struct vhost_net *net) | |||
| 415 | int err, headcount; | 415 | int err, headcount; |
| 416 | size_t vhost_hlen, sock_hlen; | 416 | size_t vhost_hlen, sock_hlen; |
| 417 | size_t vhost_len, sock_len; | 417 | size_t vhost_len, sock_len; |
| 418 | struct socket *sock = rcu_dereference(vq->private_data); | 418 | /* TODO: check that we are running from vhost_worker? */ |
| 419 | struct socket *sock = rcu_dereference_check(vq->private_data, 1); | ||
| 419 | if (!sock || skb_queue_empty(&sock->sk->sk_receive_queue)) | 420 | if (!sock || skb_queue_empty(&sock->sk->sk_receive_queue)) |
| 420 | return; | 421 | return; |
| 421 | 422 | ||
diff --git a/drivers/vhost/vhost.h b/drivers/vhost/vhost.h index 2af44b7b1f3..b3363ae3851 100644 --- a/drivers/vhost/vhost.h +++ b/drivers/vhost/vhost.h | |||
| @@ -173,9 +173,9 @@ static inline int vhost_has_feature(struct vhost_dev *dev, int bit) | |||
| 173 | { | 173 | { |
| 174 | unsigned acked_features; | 174 | unsigned acked_features; |
| 175 | 175 | ||
| 176 | acked_features = | 176 | /* TODO: check that we are running from vhost_worker or dev mutex is |
| 177 | rcu_dereference_index_check(dev->acked_features, | 177 | * held? */ |
| 178 | lockdep_is_held(&dev->mutex)); | 178 | acked_features = rcu_dereference_index_check(dev->acked_features, 1); |
| 179 | return acked_features & (1 << bit); | 179 | return acked_features & (1 << bit); |
| 180 | } | 180 | } |
| 181 | 181 | ||
diff --git a/drivers/video/arkfb.c b/drivers/video/arkfb.c index d583bea608f..391ac939f01 100644 --- a/drivers/video/arkfb.c +++ b/drivers/video/arkfb.c | |||
| @@ -23,7 +23,7 @@ | |||
| 23 | #include <linux/svga.h> | 23 | #include <linux/svga.h> |
| 24 | #include <linux/init.h> | 24 | #include <linux/init.h> |
| 25 | #include <linux/pci.h> | 25 | #include <linux/pci.h> |
| 26 | #include <linux/console.h> /* Why should fb driver call console functions? because acquire_console_sem() */ | 26 | #include <linux/console.h> /* Why should fb driver call console functions? because console_lock() */ |
| 27 | #include <video/vga.h> | 27 | #include <video/vga.h> |
| 28 | 28 | ||
| 29 | #ifdef CONFIG_MTRR | 29 | #ifdef CONFIG_MTRR |
| @@ -1091,12 +1091,12 @@ static int ark_pci_suspend (struct pci_dev* dev, pm_message_t state) | |||
| 1091 | 1091 | ||
| 1092 | dev_info(info->device, "suspend\n"); | 1092 | dev_info(info->device, "suspend\n"); |
| 1093 | 1093 | ||
| 1094 | acquire_console_sem(); | 1094 | console_lock(); |
| 1095 | mutex_lock(&(par->open_lock)); | 1095 | mutex_lock(&(par->open_lock)); |
| 1096 | 1096 | ||
| 1097 | if ((state.event == PM_EVENT_FREEZE) || (par->ref_count == 0)) { | 1097 | if ((state.event == PM_EVENT_FREEZE) || (par->ref_count == 0)) { |
| 1098 | mutex_unlock(&(par->open_lock)); | 1098 | mutex_unlock(&(par->open_lock)); |
| 1099 | release_console_sem(); | 1099 | console_unlock(); |
| 1100 | return 0; | 1100 | return 0; |
| 1101 | } | 1101 | } |
| 1102 | 1102 | ||
| @@ -1107,7 +1107,7 @@ static int ark_pci_suspend (struct pci_dev* dev, pm_message_t state) | |||
| 1107 | pci_set_power_state(dev, pci_choose_state(dev, state)); | 1107 | pci_set_power_state(dev, pci_choose_state(dev, state)); |
| 1108 | 1108 | ||
| 1109 | mutex_unlock(&(par->open_lock)); | 1109 | mutex_unlock(&(par->open_lock)); |
| 1110 | release_console_sem(); | 1110 | console_unlock(); |
| 1111 | 1111 | ||
| 1112 | return 0; | 1112 | return 0; |
| 1113 | } | 1113 | } |
| @@ -1122,7 +1122,7 @@ static int ark_pci_resume (struct pci_dev* dev) | |||
| 1122 | 1122 | ||
| 1123 | dev_info(info->device, "resume\n"); | 1123 | dev_info(info->device, "resume\n"); |
| 1124 | 1124 | ||
| 1125 | acquire_console_sem(); | 1125 | console_lock(); |
| 1126 | mutex_lock(&(par->open_lock)); | 1126 | mutex_lock(&(par->open_lock)); |
| 1127 | 1127 | ||
| 1128 | if (par->ref_count == 0) | 1128 | if (par->ref_count == 0) |
| @@ -1141,7 +1141,7 @@ static int ark_pci_resume (struct pci_dev* dev) | |||
| 1141 | 1141 | ||
| 1142 | fail: | 1142 | fail: |
| 1143 | mutex_unlock(&(par->open_lock)); | 1143 | mutex_unlock(&(par->open_lock)); |
| 1144 | release_console_sem(); | 1144 | console_unlock(); |
| 1145 | return 0; | 1145 | return 0; |
| 1146 | } | 1146 | } |
| 1147 | #else | 1147 | #else |
diff --git a/drivers/video/aty/aty128fb.c b/drivers/video/aty/aty128fb.c index dd9de2e8058..4cb6a576c56 100644 --- a/drivers/video/aty/aty128fb.c +++ b/drivers/video/aty/aty128fb.c | |||
| @@ -1860,11 +1860,11 @@ static void aty128_early_resume(void *data) | |||
| 1860 | { | 1860 | { |
| 1861 | struct aty128fb_par *par = data; | 1861 | struct aty128fb_par *par = data; |
| 1862 | 1862 | ||
| 1863 | if (try_acquire_console_sem()) | 1863 | if (!console_trylock()) |
| 1864 | return; | 1864 | return; |
| 1865 | pci_restore_state(par->pdev); | 1865 | pci_restore_state(par->pdev); |
| 1866 | aty128_do_resume(par->pdev); | 1866 | aty128_do_resume(par->pdev); |
| 1867 | release_console_sem(); | 1867 | console_unlock(); |
| 1868 | } | 1868 | } |
| 1869 | #endif /* CONFIG_PPC_PMAC */ | 1869 | #endif /* CONFIG_PPC_PMAC */ |
| 1870 | 1870 | ||
| @@ -2438,7 +2438,7 @@ static int aty128_pci_suspend(struct pci_dev *pdev, pm_message_t state) | |||
| 2438 | 2438 | ||
| 2439 | printk(KERN_DEBUG "aty128fb: suspending...\n"); | 2439 | printk(KERN_DEBUG "aty128fb: suspending...\n"); |
| 2440 | 2440 | ||
| 2441 | acquire_console_sem(); | 2441 | console_lock(); |
| 2442 | 2442 | ||
| 2443 | fb_set_suspend(info, 1); | 2443 | fb_set_suspend(info, 1); |
| 2444 | 2444 | ||
| @@ -2470,7 +2470,7 @@ static int aty128_pci_suspend(struct pci_dev *pdev, pm_message_t state) | |||
| 2470 | if (state.event != PM_EVENT_ON) | 2470 | if (state.event != PM_EVENT_ON) |
| 2471 | aty128_set_suspend(par, 1); | 2471 | aty128_set_suspend(par, 1); |
| 2472 | 2472 | ||
| 2473 | release_console_sem(); | 2473 | console_unlock(); |
| 2474 | 2474 | ||
| 2475 | pdev->dev.power.power_state = state; | 2475 | pdev->dev.power.power_state = state; |
| 2476 | 2476 | ||
| @@ -2527,9 +2527,9 @@ static int aty128_pci_resume(struct pci_dev *pdev) | |||
| 2527 | { | 2527 | { |
| 2528 | int rc; | 2528 | int rc; |
| 2529 | 2529 | ||
| 2530 | acquire_console_sem(); | 2530 | console_lock(); |
| 2531 | rc = aty128_do_resume(pdev); | 2531 | rc = aty128_do_resume(pdev); |
| 2532 | release_console_sem(); | 2532 | console_unlock(); |
| 2533 | 2533 | ||
| 2534 | return rc; | 2534 | return rc; |
| 2535 | } | 2535 | } |
diff --git a/drivers/video/aty/atyfb_base.c b/drivers/video/aty/atyfb_base.c index 767ab4fb1a0..94e293fce1d 100644 --- a/drivers/video/aty/atyfb_base.c +++ b/drivers/video/aty/atyfb_base.c | |||
| @@ -2069,7 +2069,7 @@ static int atyfb_pci_suspend(struct pci_dev *pdev, pm_message_t state) | |||
| 2069 | if (state.event == pdev->dev.power.power_state.event) | 2069 | if (state.event == pdev->dev.power.power_state.event) |
| 2070 | return 0; | 2070 | return 0; |
| 2071 | 2071 | ||
| 2072 | acquire_console_sem(); | 2072 | console_lock(); |
| 2073 | 2073 | ||
| 2074 | fb_set_suspend(info, 1); | 2074 | fb_set_suspend(info, 1); |
| 2075 | 2075 | ||
| @@ -2097,14 +2097,14 @@ static int atyfb_pci_suspend(struct pci_dev *pdev, pm_message_t state) | |||
| 2097 | par->lock_blank = 0; | 2097 | par->lock_blank = 0; |
| 2098 | atyfb_blank(FB_BLANK_UNBLANK, info); | 2098 | atyfb_blank(FB_BLANK_UNBLANK, info); |
| 2099 | fb_set_suspend(info, 0); | 2099 | fb_set_suspend(info, 0); |
| 2100 | release_console_sem(); | 2100 | console_unlock(); |
| 2101 | return -EIO; | 2101 | return -EIO; |
| 2102 | } | 2102 | } |
| 2103 | #else | 2103 | #else |
| 2104 | pci_set_power_state(pdev, pci_choose_state(pdev, state)); | 2104 | pci_set_power_state(pdev, pci_choose_state(pdev, state)); |
| 2105 | #endif | 2105 | #endif |
| 2106 | 2106 | ||
| 2107 | release_console_sem(); | 2107 | console_unlock(); |
| 2108 | 2108 | ||
| 2109 | pdev->dev.power.power_state = state; | 2109 | pdev->dev.power.power_state = state; |
| 2110 | 2110 | ||
| @@ -2133,7 +2133,7 @@ static int atyfb_pci_resume(struct pci_dev *pdev) | |||
| 2133 | if (pdev->dev.power.power_state.event == PM_EVENT_ON) | 2133 | if (pdev->dev.power.power_state.event == PM_EVENT_ON) |
| 2134 | return 0; | 2134 | return 0; |
| 2135 | 2135 | ||
| 2136 | acquire_console_sem(); | 2136 | console_lock(); |
| 2137 | 2137 | ||
| 2138 | /* | 2138 | /* |
| 2139 | * PCI state will have been restored by the core, so | 2139 | * PCI state will have been restored by the core, so |
| @@ -2161,7 +2161,7 @@ static int atyfb_pci_resume(struct pci_dev *pdev) | |||
| 2161 | par->lock_blank = 0; | 2161 | par->lock_blank = 0; |
| 2162 | atyfb_blank(FB_BLANK_UNBLANK, info); | 2162 | atyfb_blank(FB_BLANK_UNBLANK, info); |
| 2163 | 2163 | ||
| 2164 | release_console_sem(); | 2164 | console_unlock(); |
| 2165 | 2165 | ||
| 2166 | pdev->dev.power.power_state = PMSG_ON; | 2166 | pdev->dev.power.power_state = PMSG_ON; |
| 2167 | 2167 | ||
diff --git a/drivers/video/aty/radeon_pm.c b/drivers/video/aty/radeon_pm.c index c4e17642d9c..92bda584851 100644 --- a/drivers/video/aty/radeon_pm.c +++ b/drivers/video/aty/radeon_pm.c | |||
| @@ -2626,7 +2626,7 @@ int radeonfb_pci_suspend(struct pci_dev *pdev, pm_message_t mesg) | |||
| 2626 | goto done; | 2626 | goto done; |
| 2627 | } | 2627 | } |
| 2628 | 2628 | ||
| 2629 | acquire_console_sem(); | 2629 | console_lock(); |
| 2630 | 2630 | ||
| 2631 | fb_set_suspend(info, 1); | 2631 | fb_set_suspend(info, 1); |
| 2632 | 2632 | ||
| @@ -2690,7 +2690,7 @@ int radeonfb_pci_suspend(struct pci_dev *pdev, pm_message_t mesg) | |||
| 2690 | if (rinfo->pm_mode & radeon_pm_d2) | 2690 | if (rinfo->pm_mode & radeon_pm_d2) |
| 2691 | radeon_set_suspend(rinfo, 1); | 2691 | radeon_set_suspend(rinfo, 1); |
| 2692 | 2692 | ||
| 2693 | release_console_sem(); | 2693 | console_unlock(); |
| 2694 | 2694 | ||
| 2695 | done: | 2695 | done: |
| 2696 | pdev->dev.power.power_state = mesg; | 2696 | pdev->dev.power.power_state = mesg; |
| @@ -2715,10 +2715,10 @@ int radeonfb_pci_resume(struct pci_dev *pdev) | |||
| 2715 | return 0; | 2715 | return 0; |
| 2716 | 2716 | ||
| 2717 | if (rinfo->no_schedule) { | 2717 | if (rinfo->no_schedule) { |
| 2718 | if (try_acquire_console_sem()) | 2718 | if (!console_trylock()) |
| 2719 | return 0; | 2719 | return 0; |
| 2720 | } else | 2720 | } else |
| 2721 | acquire_console_sem(); | 2721 | console_lock(); |
| 2722 | 2722 | ||
| 2723 | printk(KERN_DEBUG "radeonfb (%s): resuming from state: %d...\n", | 2723 | printk(KERN_DEBUG "radeonfb (%s): resuming from state: %d...\n", |
| 2724 | pci_name(pdev), pdev->dev.power.power_state.event); | 2724 | pci_name(pdev), pdev->dev.power.power_state.event); |
| @@ -2783,7 +2783,7 @@ int radeonfb_pci_resume(struct pci_dev *pdev) | |||
| 2783 | pdev->dev.power.power_state = PMSG_ON; | 2783 | pdev->dev.power.power_state = PMSG_ON; |
| 2784 | 2784 | ||
| 2785 | bail: | 2785 | bail: |
| 2786 | release_console_sem(); | 2786 | console_unlock(); |
| 2787 | 2787 | ||
| 2788 | return rc; | 2788 | return rc; |
| 2789 | } | 2789 | } |
diff --git a/drivers/video/bf537-lq035.c b/drivers/video/bf537-lq035.c index 18c507874ff..47c21fb2c82 100644 --- a/drivers/video/bf537-lq035.c +++ b/drivers/video/bf537-lq035.c | |||
| @@ -696,6 +696,7 @@ static int __devinit bfin_lq035_probe(struct platform_device *pdev) | |||
| 696 | { | 696 | { |
| 697 | struct backlight_properties props; | 697 | struct backlight_properties props; |
| 698 | dma_addr_t dma_handle; | 698 | dma_addr_t dma_handle; |
| 699 | int ret; | ||
| 699 | 700 | ||
| 700 | if (request_dma(CH_PPI, KBUILD_MODNAME)) { | 701 | if (request_dma(CH_PPI, KBUILD_MODNAME)) { |
| 701 | pr_err("couldn't request PPI DMA\n"); | 702 | pr_err("couldn't request PPI DMA\n"); |
| @@ -704,17 +705,16 @@ static int __devinit bfin_lq035_probe(struct platform_device *pdev) | |||
| 704 | 705 | ||
| 705 | if (request_ports()) { | 706 | if (request_ports()) { |
| 706 | pr_err("couldn't request gpio port\n"); | 707 | pr_err("couldn't request gpio port\n"); |
| 707 | free_dma(CH_PPI); | 708 | ret = -EFAULT; |
| 708 | return -EFAULT; | 709 | goto out_ports; |
| 709 | } | 710 | } |
| 710 | 711 | ||
| 711 | fb_buffer = dma_alloc_coherent(NULL, TOTAL_VIDEO_MEM_SIZE, | 712 | fb_buffer = dma_alloc_coherent(NULL, TOTAL_VIDEO_MEM_SIZE, |
| 712 | &dma_handle, GFP_KERNEL); | 713 | &dma_handle, GFP_KERNEL); |
| 713 | if (fb_buffer == NULL) { | 714 | if (fb_buffer == NULL) { |
| 714 | pr_err("couldn't allocate dma buffer\n"); | 715 | pr_err("couldn't allocate dma buffer\n"); |
| 715 | free_dma(CH_PPI); | 716 | ret = -ENOMEM; |
| 716 | free_ports(); | 717 | goto out_dma_coherent; |
| 717 | return -ENOMEM; | ||
| 718 | } | 718 | } |
| 719 | 719 | ||
| 720 | if (L1_DATA_A_LENGTH) | 720 | if (L1_DATA_A_LENGTH) |
| @@ -725,10 +725,8 @@ static int __devinit bfin_lq035_probe(struct platform_device *pdev) | |||
| 725 | 725 | ||
| 726 | if (dma_desc_table == NULL) { | 726 | if (dma_desc_table == NULL) { |
| 727 | pr_err("couldn't allocate dma descriptor\n"); | 727 | pr_err("couldn't allocate dma descriptor\n"); |
| 728 | free_dma(CH_PPI); | 728 | ret = -ENOMEM; |
| 729 | free_ports(); | 729 | goto out_table; |
| 730 | dma_free_coherent(NULL, TOTAL_VIDEO_MEM_SIZE, fb_buffer, 0); | ||
| 731 | return -ENOMEM; | ||
| 732 | } | 730 | } |
| 733 | 731 | ||
| 734 | bfin_lq035_fb.screen_base = (void *)fb_buffer; | 732 | bfin_lq035_fb.screen_base = (void *)fb_buffer; |
| @@ -771,31 +769,21 @@ static int __devinit bfin_lq035_probe(struct platform_device *pdev) | |||
| 771 | bfin_lq035_fb.pseudo_palette = kzalloc(sizeof(u32) * 16, GFP_KERNEL); | 769 | bfin_lq035_fb.pseudo_palette = kzalloc(sizeof(u32) * 16, GFP_KERNEL); |
| 772 | if (bfin_lq035_fb.pseudo_palette == NULL) { | 770 | if (bfin_lq035_fb.pseudo_palette == NULL) { |
| 773 | pr_err("failed to allocate pseudo_palette\n"); | 771 | pr_err("failed to allocate pseudo_palette\n"); |
| 774 | free_dma(CH_PPI); | 772 | ret = -ENOMEM; |
| 775 | free_ports(); | 773 | goto out_palette; |
| 776 | dma_free_coherent(NULL, TOTAL_VIDEO_MEM_SIZE, fb_buffer, 0); | ||
| 777 | return -ENOMEM; | ||
| 778 | } | 774 | } |
| 779 | 775 | ||
| 780 | if (fb_alloc_cmap(&bfin_lq035_fb.cmap, NBR_PALETTE, 0) < 0) { | 776 | if (fb_alloc_cmap(&bfin_lq035_fb.cmap, NBR_PALETTE, 0) < 0) { |
| 781 | pr_err("failed to allocate colormap (%d entries)\n", | 777 | pr_err("failed to allocate colormap (%d entries)\n", |
| 782 | NBR_PALETTE); | 778 | NBR_PALETTE); |
| 783 | free_dma(CH_PPI); | 779 | ret = -EFAULT; |
| 784 | free_ports(); | 780 | goto out_cmap; |
| 785 | dma_free_coherent(NULL, TOTAL_VIDEO_MEM_SIZE, fb_buffer, 0); | ||
| 786 | kfree(bfin_lq035_fb.pseudo_palette); | ||
| 787 | return -EFAULT; | ||
| 788 | } | 781 | } |
| 789 | 782 | ||
| 790 | if (register_framebuffer(&bfin_lq035_fb) < 0) { | 783 | if (register_framebuffer(&bfin_lq035_fb) < 0) { |
| 791 | pr_err("unable to register framebuffer\n"); | 784 | pr_err("unable to register framebuffer\n"); |
| 792 | free_dma(CH_PPI); | 785 | ret = -EINVAL; |
| 793 | free_ports(); | 786 | goto out_reg; |
| 794 | dma_free_coherent(NULL, TOTAL_VIDEO_MEM_SIZE, fb_buffer, 0); | ||
| 795 | fb_buffer = NULL; | ||
| 796 | kfree(bfin_lq035_fb.pseudo_palette); | ||
| 797 | fb_dealloc_cmap(&bfin_lq035_fb.cmap); | ||
| 798 | return -EINVAL; | ||
| 799 | } | 787 | } |
| 800 | 788 | ||
| 801 | i2c_add_driver(&ad5280_driver); | 789 | i2c_add_driver(&ad5280_driver); |
| @@ -807,11 +795,31 @@ static int __devinit bfin_lq035_probe(struct platform_device *pdev) | |||
| 807 | 795 | ||
| 808 | lcd_dev = lcd_device_register(KBUILD_MODNAME, &pdev->dev, NULL, | 796 | lcd_dev = lcd_device_register(KBUILD_MODNAME, &pdev->dev, NULL, |
| 809 | &bfin_lcd_ops); | 797 | &bfin_lcd_ops); |
| 798 | if (IS_ERR(lcd_dev)) { | ||
| 799 | pr_err("unable to register lcd\n"); | ||
| 800 | ret = PTR_ERR(lcd_dev); | ||
| 801 | goto out_lcd; | ||
| 802 | } | ||
| 810 | lcd_dev->props.max_contrast = 255, | 803 | lcd_dev->props.max_contrast = 255, |
| 811 | 804 | ||
| 812 | pr_info("initialized"); | 805 | pr_info("initialized"); |
| 813 | 806 | ||
| 814 | return 0; | 807 | return 0; |
| 808 | out_lcd: | ||
| 809 | unregister_framebuffer(&bfin_lq035_fb); | ||
| 810 | out_reg: | ||
| 811 | fb_dealloc_cmap(&bfin_lq035_fb.cmap); | ||
| 812 | out_cmap: | ||
| 813 | kfree(bfin_lq035_fb.pseudo_palette); | ||
| 814 | out_palette: | ||
| 815 | out_table: | ||
| 816 | dma_free_coherent(NULL, TOTAL_VIDEO_MEM_SIZE, fb_buffer, 0); | ||
| 817 | fb_buffer = NULL; | ||
| 818 | out_dma_coherent: | ||
| 819 | free_ports(); | ||
| 820 | out_ports: | ||
| 821 | free_dma(CH_PPI); | ||
| 822 | return ret; | ||
| 815 | } | 823 | } |
| 816 | 824 | ||
| 817 | static int __devexit bfin_lq035_remove(struct platform_device *pdev) | 825 | static int __devexit bfin_lq035_remove(struct platform_device *pdev) |
diff --git a/drivers/video/chipsfb.c b/drivers/video/chipsfb.c index d637e1f5317..cff742abdc5 100644 --- a/drivers/video/chipsfb.c +++ b/drivers/video/chipsfb.c | |||
| @@ -460,10 +460,10 @@ static int chipsfb_pci_suspend(struct pci_dev *pdev, pm_message_t state) | |||
| 460 | if (!(state.event & PM_EVENT_SLEEP)) | 460 | if (!(state.event & PM_EVENT_SLEEP)) |
| 461 | goto done; | 461 | goto done; |
| 462 | 462 | ||
| 463 | acquire_console_sem(); | 463 | console_lock(); |
| 464 | chipsfb_blank(1, p); | 464 | chipsfb_blank(1, p); |
| 465 | fb_set_suspend(p, 1); | 465 | fb_set_suspend(p, 1); |
| 466 | release_console_sem(); | 466 | console_unlock(); |
| 467 | done: | 467 | done: |
| 468 | pdev->dev.power.power_state = state; | 468 | pdev->dev.power.power_state = state; |
| 469 | return 0; | 469 | return 0; |
| @@ -473,10 +473,10 @@ static int chipsfb_pci_resume(struct pci_dev *pdev) | |||
| 473 | { | 473 | { |
| 474 | struct fb_info *p = pci_get_drvdata(pdev); | 474 | struct fb_info *p = pci_get_drvdata(pdev); |
| 475 | 475 | ||
| 476 | acquire_console_sem(); | 476 | console_lock(); |
| 477 | fb_set_suspend(p, 0); | 477 | fb_set_suspend(p, 0); |
| 478 | chipsfb_blank(0, p); | 478 | chipsfb_blank(0, p); |
| 479 | release_console_sem(); | 479 | console_unlock(); |
| 480 | 480 | ||
| 481 | pdev->dev.power.power_state = PMSG_ON; | 481 | pdev->dev.power.power_state = PMSG_ON; |
| 482 | return 0; | 482 | return 0; |
diff --git a/drivers/video/console/fbcon.c b/drivers/video/console/fbcon.c index 7ccc967831f..9c092b8d64e 100644 --- a/drivers/video/console/fbcon.c +++ b/drivers/video/console/fbcon.c | |||
| @@ -375,14 +375,14 @@ static void fb_flashcursor(struct work_struct *work) | |||
| 375 | int c; | 375 | int c; |
| 376 | int mode; | 376 | int mode; |
| 377 | 377 | ||
| 378 | acquire_console_sem(); | 378 | console_lock(); |
| 379 | if (ops && ops->currcon != -1) | 379 | if (ops && ops->currcon != -1) |
| 380 | vc = vc_cons[ops->currcon].d; | 380 | vc = vc_cons[ops->currcon].d; |
| 381 | 381 | ||
| 382 | if (!vc || !CON_IS_VISIBLE(vc) || | 382 | if (!vc || !CON_IS_VISIBLE(vc) || |
| 383 | registered_fb[con2fb_map[vc->vc_num]] != info || | 383 | registered_fb[con2fb_map[vc->vc_num]] != info || |
| 384 | vc->vc_deccm != 1) { | 384 | vc->vc_deccm != 1) { |
| 385 | release_console_sem(); | 385 | console_unlock(); |
| 386 | return; | 386 | return; |
| 387 | } | 387 | } |
| 388 | 388 | ||
| @@ -392,7 +392,7 @@ static void fb_flashcursor(struct work_struct *work) | |||
| 392 | CM_ERASE : CM_DRAW; | 392 | CM_ERASE : CM_DRAW; |
| 393 | ops->cursor(vc, info, mode, softback_lines, get_color(vc, info, c, 1), | 393 | ops->cursor(vc, info, mode, softback_lines, get_color(vc, info, c, 1), |
| 394 | get_color(vc, info, c, 0)); | 394 | get_color(vc, info, c, 0)); |
| 395 | release_console_sem(); | 395 | console_unlock(); |
| 396 | } | 396 | } |
| 397 | 397 | ||
| 398 | static void cursor_timer_handler(unsigned long dev_addr) | 398 | static void cursor_timer_handler(unsigned long dev_addr) |
| @@ -836,7 +836,7 @@ static int set_con2fb_map(int unit, int newidx, int user) | |||
| 836 | 836 | ||
| 837 | found = search_fb_in_map(newidx); | 837 | found = search_fb_in_map(newidx); |
| 838 | 838 | ||
| 839 | acquire_console_sem(); | 839 | console_lock(); |
| 840 | con2fb_map[unit] = newidx; | 840 | con2fb_map[unit] = newidx; |
| 841 | if (!err && !found) | 841 | if (!err && !found) |
| 842 | err = con2fb_acquire_newinfo(vc, info, unit, oldidx); | 842 | err = con2fb_acquire_newinfo(vc, info, unit, oldidx); |
| @@ -863,7 +863,7 @@ static int set_con2fb_map(int unit, int newidx, int user) | |||
| 863 | if (!search_fb_in_map(info_idx)) | 863 | if (!search_fb_in_map(info_idx)) |
| 864 | info_idx = newidx; | 864 | info_idx = newidx; |
| 865 | 865 | ||
| 866 | release_console_sem(); | 866 | console_unlock(); |
| 867 | return err; | 867 | return err; |
| 868 | } | 868 | } |
| 869 | 869 | ||
| @@ -3321,7 +3321,7 @@ static ssize_t store_rotate(struct device *device, | |||
| 3321 | if (fbcon_has_exited) | 3321 | if (fbcon_has_exited) |
| 3322 | return count; | 3322 | return count; |
| 3323 | 3323 | ||
| 3324 | acquire_console_sem(); | 3324 | console_lock(); |
| 3325 | idx = con2fb_map[fg_console]; | 3325 | idx = con2fb_map[fg_console]; |
| 3326 | 3326 | ||
| 3327 | if (idx == -1 || registered_fb[idx] == NULL) | 3327 | if (idx == -1 || registered_fb[idx] == NULL) |
| @@ -3331,7 +3331,7 @@ static ssize_t store_rotate(struct device *device, | |||
| 3331 | rotate = simple_strtoul(buf, last, 0); | 3331 | rotate = simple_strtoul(buf, last, 0); |
| 3332 | fbcon_rotate(info, rotate); | 3332 | fbcon_rotate(info, rotate); |
| 3333 | err: | 3333 | err: |
| 3334 | release_console_sem(); | 3334 | console_unlock(); |
| 3335 | return count; | 3335 | return count; |
| 3336 | } | 3336 | } |
| 3337 | 3337 | ||
| @@ -3346,7 +3346,7 @@ static ssize_t store_rotate_all(struct device *device, | |||
| 3346 | if (fbcon_has_exited) | 3346 | if (fbcon_has_exited) |
| 3347 | return count; | 3347 | return count; |
| 3348 | 3348 | ||
| 3349 | acquire_console_sem(); | 3349 | console_lock(); |
| 3350 | idx = con2fb_map[fg_console]; | 3350 | idx = con2fb_map[fg_console]; |
| 3351 | 3351 | ||
| 3352 | if (idx == -1 || registered_fb[idx] == NULL) | 3352 | if (idx == -1 || registered_fb[idx] == NULL) |
| @@ -3356,7 +3356,7 @@ static ssize_t store_rotate_all(struct device *device, | |||
| 3356 | rotate = simple_strtoul(buf, last, 0); | 3356 | rotate = simple_strtoul(buf, last, 0); |
| 3357 | fbcon_rotate_all(info, rotate); | 3357 | fbcon_rotate_all(info, rotate); |
| 3358 | err: | 3358 | err: |
| 3359 | release_console_sem(); | 3359 | console_unlock(); |
| 3360 | return count; | 3360 | return count; |
| 3361 | } | 3361 | } |
| 3362 | 3362 | ||
| @@ -3369,7 +3369,7 @@ static ssize_t show_rotate(struct device *device, | |||
| 3369 | if (fbcon_has_exited) | 3369 | if (fbcon_has_exited) |
| 3370 | return 0; | 3370 | return 0; |
| 3371 | 3371 | ||
| 3372 | acquire_console_sem(); | 3372 | console_lock(); |
| 3373 | idx = con2fb_map[fg_console]; | 3373 | idx = con2fb_map[fg_console]; |
| 3374 | 3374 | ||
| 3375 | if (idx == -1 || registered_fb[idx] == NULL) | 3375 | if (idx == -1 || registered_fb[idx] == NULL) |
| @@ -3378,7 +3378,7 @@ static ssize_t show_rotate(struct device *device, | |||
| 3378 | info = registered_fb[idx]; | 3378 | info = registered_fb[idx]; |
| 3379 | rotate = fbcon_get_rotate(info); | 3379 | rotate = fbcon_get_rotate(info); |
| 3380 | err: | 3380 | err: |
| 3381 | release_console_sem(); | 3381 | console_unlock(); |
| 3382 | return snprintf(buf, PAGE_SIZE, "%d\n", rotate); | 3382 | return snprintf(buf, PAGE_SIZE, "%d\n", rotate); |
| 3383 | } | 3383 | } |
| 3384 | 3384 | ||
| @@ -3392,7 +3392,7 @@ static ssize_t show_cursor_blink(struct device *device, | |||
| 3392 | if (fbcon_has_exited) | 3392 | if (fbcon_has_exited) |
| 3393 | return 0; | 3393 | return 0; |
| 3394 | 3394 | ||
| 3395 | acquire_console_sem(); | 3395 | console_lock(); |
| 3396 | idx = con2fb_map[fg_console]; | 3396 | idx = con2fb_map[fg_console]; |
| 3397 | 3397 | ||
| 3398 | if (idx == -1 || registered_fb[idx] == NULL) | 3398 | if (idx == -1 || registered_fb[idx] == NULL) |
| @@ -3406,7 +3406,7 @@ static ssize_t show_cursor_blink(struct device *device, | |||
| 3406 | 3406 | ||
| 3407 | blink = (ops->flags & FBCON_FLAGS_CURSOR_TIMER) ? 1 : 0; | 3407 | blink = (ops->flags & FBCON_FLAGS_CURSOR_TIMER) ? 1 : 0; |
| 3408 | err: | 3408 | err: |
| 3409 | release_console_sem(); | 3409 | console_unlock(); |
| 3410 | return snprintf(buf, PAGE_SIZE, "%d\n", blink); | 3410 | return snprintf(buf, PAGE_SIZE, "%d\n", blink); |
| 3411 | } | 3411 | } |
| 3412 | 3412 | ||
| @@ -3421,7 +3421,7 @@ static ssize_t store_cursor_blink(struct device *device, | |||
| 3421 | if (fbcon_has_exited) | 3421 | if (fbcon_has_exited) |
| 3422 | return count; | 3422 | return count; |
| 3423 | 3423 | ||
| 3424 | acquire_console_sem(); | 3424 | console_lock(); |
| 3425 | idx = con2fb_map[fg_console]; | 3425 | idx = con2fb_map[fg_console]; |
| 3426 | 3426 | ||
| 3427 | if (idx == -1 || registered_fb[idx] == NULL) | 3427 | if (idx == -1 || registered_fb[idx] == NULL) |
| @@ -3443,7 +3443,7 @@ static ssize_t store_cursor_blink(struct device *device, | |||
| 3443 | } | 3443 | } |
| 3444 | 3444 | ||
| 3445 | err: | 3445 | err: |
| 3446 | release_console_sem(); | 3446 | console_unlock(); |
| 3447 | return count; | 3447 | return count; |
| 3448 | } | 3448 | } |
| 3449 | 3449 | ||
| @@ -3482,7 +3482,7 @@ static void fbcon_start(void) | |||
| 3482 | if (num_registered_fb) { | 3482 | if (num_registered_fb) { |
| 3483 | int i; | 3483 | int i; |
| 3484 | 3484 | ||
| 3485 | acquire_console_sem(); | 3485 | console_lock(); |
| 3486 | 3486 | ||
| 3487 | for (i = 0; i < FB_MAX; i++) { | 3487 | for (i = 0; i < FB_MAX; i++) { |
| 3488 | if (registered_fb[i] != NULL) { | 3488 | if (registered_fb[i] != NULL) { |
| @@ -3491,7 +3491,7 @@ static void fbcon_start(void) | |||
| 3491 | } | 3491 | } |
| 3492 | } | 3492 | } |
| 3493 | 3493 | ||
| 3494 | release_console_sem(); | 3494 | console_unlock(); |
| 3495 | fbcon_takeover(0); | 3495 | fbcon_takeover(0); |
| 3496 | } | 3496 | } |
| 3497 | } | 3497 | } |
| @@ -3552,7 +3552,7 @@ static int __init fb_console_init(void) | |||
| 3552 | { | 3552 | { |
| 3553 | int i; | 3553 | int i; |
| 3554 | 3554 | ||
| 3555 | acquire_console_sem(); | 3555 | console_lock(); |
| 3556 | fb_register_client(&fbcon_event_notifier); | 3556 | fb_register_client(&fbcon_event_notifier); |
| 3557 | fbcon_device = device_create(fb_class, NULL, MKDEV(0, 0), NULL, | 3557 | fbcon_device = device_create(fb_class, NULL, MKDEV(0, 0), NULL, |
| 3558 | "fbcon"); | 3558 | "fbcon"); |
| @@ -3568,7 +3568,7 @@ static int __init fb_console_init(void) | |||
| 3568 | for (i = 0; i < MAX_NR_CONSOLES; i++) | 3568 | for (i = 0; i < MAX_NR_CONSOLES; i++) |
| 3569 | con2fb_map[i] = -1; | 3569 | con2fb_map[i] = -1; |
| 3570 | 3570 | ||
| 3571 | release_console_sem(); | 3571 | console_unlock(); |
| 3572 | fbcon_start(); | 3572 | fbcon_start(); |
| 3573 | return 0; | 3573 | return 0; |
| 3574 | } | 3574 | } |
| @@ -3591,12 +3591,12 @@ static void __exit fbcon_deinit_device(void) | |||
| 3591 | 3591 | ||
| 3592 | static void __exit fb_console_exit(void) | 3592 | static void __exit fb_console_exit(void) |
| 3593 | { | 3593 | { |
| 3594 | acquire_console_sem(); | 3594 | console_lock(); |
| 3595 | fb_unregister_client(&fbcon_event_notifier); | 3595 | fb_unregister_client(&fbcon_event_notifier); |
| 3596 | fbcon_deinit_device(); | 3596 | fbcon_deinit_device(); |
| 3597 | device_destroy(fb_class, MKDEV(0, 0)); | 3597 | device_destroy(fb_class, MKDEV(0, 0)); |
| 3598 | fbcon_exit(); | 3598 | fbcon_exit(); |
| 3599 | release_console_sem(); | 3599 | console_unlock(); |
| 3600 | unregister_con_driver(&fb_con); | 3600 | unregister_con_driver(&fb_con); |
| 3601 | } | 3601 | } |
| 3602 | 3602 | ||
diff --git a/drivers/video/console/vgacon.c b/drivers/video/console/vgacon.c index c97491b8b39..915fd74da7a 100644 --- a/drivers/video/console/vgacon.c +++ b/drivers/video/console/vgacon.c | |||
| @@ -202,11 +202,7 @@ static void vgacon_scrollback_init(int pitch) | |||
| 202 | } | 202 | } |
| 203 | } | 203 | } |
| 204 | 204 | ||
| 205 | /* | 205 | static void vgacon_scrollback_startup(void) |
| 206 | * Called only duing init so call of alloc_bootmen is ok. | ||
| 207 | * Marked __init_refok to silence modpost. | ||
| 208 | */ | ||
| 209 | static void __init_refok vgacon_scrollback_startup(void) | ||
| 210 | { | 206 | { |
| 211 | vgacon_scrollback = kcalloc(CONFIG_VGACON_SOFT_SCROLLBACK_SIZE, 1024, GFP_NOWAIT); | 207 | vgacon_scrollback = kcalloc(CONFIG_VGACON_SOFT_SCROLLBACK_SIZE, 1024, GFP_NOWAIT); |
| 212 | vgacon_scrollback_init(vga_video_num_columns * 2); | 208 | vgacon_scrollback_init(vga_video_num_columns * 2); |
diff --git a/drivers/video/da8xx-fb.c b/drivers/video/da8xx-fb.c index c265aed09e0..8d61ef96eed 100644 --- a/drivers/video/da8xx-fb.c +++ b/drivers/video/da8xx-fb.c | |||
| @@ -1092,9 +1092,10 @@ static int __init fb_probe(struct platform_device *device) | |||
| 1092 | 1092 | ||
| 1093 | irq_freq: | 1093 | irq_freq: |
| 1094 | #ifdef CONFIG_CPU_FREQ | 1094 | #ifdef CONFIG_CPU_FREQ |
| 1095 | lcd_da8xx_cpufreq_deregister(par); | ||
| 1096 | #endif | ||
| 1095 | err_cpu_freq: | 1097 | err_cpu_freq: |
| 1096 | unregister_framebuffer(da8xx_fb_info); | 1098 | unregister_framebuffer(da8xx_fb_info); |
| 1097 | #endif | ||
| 1098 | 1099 | ||
| 1099 | err_dealloc_cmap: | 1100 | err_dealloc_cmap: |
| 1100 | fb_dealloc_cmap(&da8xx_fb_info->cmap); | 1101 | fb_dealloc_cmap(&da8xx_fb_info->cmap); |
| @@ -1130,14 +1131,14 @@ static int fb_suspend(struct platform_device *dev, pm_message_t state) | |||
| 1130 | struct fb_info *info = platform_get_drvdata(dev); | 1131 | struct fb_info *info = platform_get_drvdata(dev); |
| 1131 | struct da8xx_fb_par *par = info->par; | 1132 | struct da8xx_fb_par *par = info->par; |
| 1132 | 1133 | ||
| 1133 | acquire_console_sem(); | 1134 | console_lock(); |
| 1134 | if (par->panel_power_ctrl) | 1135 | if (par->panel_power_ctrl) |
| 1135 | par->panel_power_ctrl(0); | 1136 | par->panel_power_ctrl(0); |
| 1136 | 1137 | ||
| 1137 | fb_set_suspend(info, 1); | 1138 | fb_set_suspend(info, 1); |
| 1138 | lcd_disable_raster(); | 1139 | lcd_disable_raster(); |
| 1139 | clk_disable(par->lcdc_clk); | 1140 | clk_disable(par->lcdc_clk); |
| 1140 | release_console_sem(); | 1141 | console_unlock(); |
| 1141 | 1142 | ||
| 1142 | return 0; | 1143 | return 0; |
| 1143 | } | 1144 | } |
| @@ -1146,14 +1147,14 @@ static int fb_resume(struct platform_device *dev) | |||
| 1146 | struct fb_info *info = platform_get_drvdata(dev); | 1147 | struct fb_info *info = platform_get_drvdata(dev); |
| 1147 | struct da8xx_fb_par *par = info->par; | 1148 | struct da8xx_fb_par *par = info->par; |
| 1148 | 1149 | ||
| 1149 | acquire_console_sem(); | 1150 | console_lock(); |
| 1150 | if (par->panel_power_ctrl) | 1151 | if (par->panel_power_ctrl) |
| 1151 | par->panel_power_ctrl(1); | 1152 | par->panel_power_ctrl(1); |
| 1152 | 1153 | ||
| 1153 | clk_enable(par->lcdc_clk); | 1154 | clk_enable(par->lcdc_clk); |
| 1154 | lcd_enable_raster(); | 1155 | lcd_enable_raster(); |
| 1155 | fb_set_suspend(info, 0); | 1156 | fb_set_suspend(info, 0); |
| 1156 | release_console_sem(); | 1157 | console_unlock(); |
| 1157 | 1158 | ||
| 1158 | return 0; | 1159 | return 0; |
| 1159 | } | 1160 | } |
diff --git a/drivers/video/fbmem.c b/drivers/video/fbmem.c index 4ac1201ad6c..e2bf95370e4 100644 --- a/drivers/video/fbmem.c +++ b/drivers/video/fbmem.c | |||
| @@ -1036,11 +1036,11 @@ static long do_fb_ioctl(struct fb_info *info, unsigned int cmd, | |||
| 1036 | return -EFAULT; | 1036 | return -EFAULT; |
| 1037 | if (!lock_fb_info(info)) | 1037 | if (!lock_fb_info(info)) |
| 1038 | return -ENODEV; | 1038 | return -ENODEV; |
| 1039 | acquire_console_sem(); | 1039 | console_lock(); |
| 1040 | info->flags |= FBINFO_MISC_USEREVENT; | 1040 | info->flags |= FBINFO_MISC_USEREVENT; |
| 1041 | ret = fb_set_var(info, &var); | 1041 | ret = fb_set_var(info, &var); |
| 1042 | info->flags &= ~FBINFO_MISC_USEREVENT; | 1042 | info->flags &= ~FBINFO_MISC_USEREVENT; |
| 1043 | release_console_sem(); | 1043 | console_unlock(); |
| 1044 | unlock_fb_info(info); | 1044 | unlock_fb_info(info); |
| 1045 | if (!ret && copy_to_user(argp, &var, sizeof(var))) | 1045 | if (!ret && copy_to_user(argp, &var, sizeof(var))) |
| 1046 | ret = -EFAULT; | 1046 | ret = -EFAULT; |
| @@ -1072,9 +1072,9 @@ static long do_fb_ioctl(struct fb_info *info, unsigned int cmd, | |||
| 1072 | return -EFAULT; | 1072 | return -EFAULT; |
| 1073 | if (!lock_fb_info(info)) | 1073 | if (!lock_fb_info(info)) |
| 1074 | return -ENODEV; | 1074 | return -ENODEV; |
| 1075 | acquire_console_sem(); | 1075 | console_lock(); |
| 1076 | ret = fb_pan_display(info, &var); | 1076 | ret = fb_pan_display(info, &var); |
| 1077 | release_console_sem(); | 1077 | console_unlock(); |
| 1078 | unlock_fb_info(info); | 1078 | unlock_fb_info(info); |
| 1079 | if (ret == 0 && copy_to_user(argp, &var, sizeof(var))) | 1079 | if (ret == 0 && copy_to_user(argp, &var, sizeof(var))) |
| 1080 | return -EFAULT; | 1080 | return -EFAULT; |
| @@ -1119,11 +1119,11 @@ static long do_fb_ioctl(struct fb_info *info, unsigned int cmd, | |||
| 1119 | case FBIOBLANK: | 1119 | case FBIOBLANK: |
| 1120 | if (!lock_fb_info(info)) | 1120 | if (!lock_fb_info(info)) |
| 1121 | return -ENODEV; | 1121 | return -ENODEV; |
| 1122 | acquire_console_sem(); | 1122 | console_lock(); |
| 1123 | info->flags |= FBINFO_MISC_USEREVENT; | 1123 | info->flags |= FBINFO_MISC_USEREVENT; |
| 1124 | ret = fb_blank(info, arg); | 1124 | ret = fb_blank(info, arg); |
| 1125 | info->flags &= ~FBINFO_MISC_USEREVENT; | 1125 | info->flags &= ~FBINFO_MISC_USEREVENT; |
| 1126 | release_console_sem(); | 1126 | console_unlock(); |
| 1127 | unlock_fb_info(info); | 1127 | unlock_fb_info(info); |
| 1128 | break; | 1128 | break; |
| 1129 | default: | 1129 | default: |
diff --git a/drivers/video/fbsysfs.c b/drivers/video/fbsysfs.c index 0a08f134122..f4a32779168 100644 --- a/drivers/video/fbsysfs.c +++ b/drivers/video/fbsysfs.c | |||
| @@ -90,11 +90,11 @@ static int activate(struct fb_info *fb_info, struct fb_var_screeninfo *var) | |||
| 90 | int err; | 90 | int err; |
| 91 | 91 | ||
| 92 | var->activate |= FB_ACTIVATE_FORCE; | 92 | var->activate |= FB_ACTIVATE_FORCE; |
| 93 | acquire_console_sem(); | 93 | console_lock(); |
| 94 | fb_info->flags |= FBINFO_MISC_USEREVENT; | 94 | fb_info->flags |= FBINFO_MISC_USEREVENT; |
| 95 | err = fb_set_var(fb_info, var); | 95 | err = fb_set_var(fb_info, var); |
| 96 | fb_info->flags &= ~FBINFO_MISC_USEREVENT; | 96 | fb_info->flags &= ~FBINFO_MISC_USEREVENT; |
| 97 | release_console_sem(); | 97 | console_unlock(); |
| 98 | if (err) | 98 | if (err) |
| 99 | return err; | 99 | return err; |
| 100 | return 0; | 100 | return 0; |
| @@ -175,7 +175,7 @@ static ssize_t store_modes(struct device *device, | |||
| 175 | if (i * sizeof(struct fb_videomode) != count) | 175 | if (i * sizeof(struct fb_videomode) != count) |
| 176 | return -EINVAL; | 176 | return -EINVAL; |
| 177 | 177 | ||
| 178 | acquire_console_sem(); | 178 | console_lock(); |
| 179 | list_splice(&fb_info->modelist, &old_list); | 179 | list_splice(&fb_info->modelist, &old_list); |
| 180 | fb_videomode_to_modelist((const struct fb_videomode *)buf, i, | 180 | fb_videomode_to_modelist((const struct fb_videomode *)buf, i, |
| 181 | &fb_info->modelist); | 181 | &fb_info->modelist); |
| @@ -185,7 +185,7 @@ static ssize_t store_modes(struct device *device, | |||
| 185 | } else | 185 | } else |
| 186 | fb_destroy_modelist(&old_list); | 186 | fb_destroy_modelist(&old_list); |
| 187 | 187 | ||
| 188 | release_console_sem(); | 188 | console_unlock(); |
| 189 | 189 | ||
| 190 | return 0; | 190 | return 0; |
| 191 | } | 191 | } |
| @@ -301,11 +301,11 @@ static ssize_t store_blank(struct device *device, | |||
| 301 | char *last = NULL; | 301 | char *last = NULL; |
| 302 | int err; | 302 | int err; |
| 303 | 303 | ||
| 304 | acquire_console_sem(); | 304 | console_lock(); |
| 305 | fb_info->flags |= FBINFO_MISC_USEREVENT; | 305 | fb_info->flags |= FBINFO_MISC_USEREVENT; |
| 306 | err = fb_blank(fb_info, simple_strtoul(buf, &last, 0)); | 306 | err = fb_blank(fb_info, simple_strtoul(buf, &last, 0)); |
| 307 | fb_info->flags &= ~FBINFO_MISC_USEREVENT; | 307 | fb_info->flags &= ~FBINFO_MISC_USEREVENT; |
| 308 | release_console_sem(); | 308 | console_unlock(); |
| 309 | if (err < 0) | 309 | if (err < 0) |
| 310 | return err; | 310 | return err; |
| 311 | return count; | 311 | return count; |
| @@ -364,9 +364,9 @@ static ssize_t store_pan(struct device *device, | |||
| 364 | return -EINVAL; | 364 | return -EINVAL; |
| 365 | var.yoffset = simple_strtoul(last, &last, 0); | 365 | var.yoffset = simple_strtoul(last, &last, 0); |
| 366 | 366 | ||
| 367 | acquire_console_sem(); | 367 | console_lock(); |
| 368 | err = fb_pan_display(fb_info, &var); | 368 | err = fb_pan_display(fb_info, &var); |
| 369 | release_console_sem(); | 369 | console_unlock(); |
| 370 | 370 | ||
| 371 | if (err < 0) | 371 | if (err < 0) |
| 372 | return err; | 372 | return err; |
| @@ -399,9 +399,9 @@ static ssize_t store_fbstate(struct device *device, | |||
| 399 | 399 | ||
| 400 | state = simple_strtoul(buf, &last, 0); | 400 | state = simple_strtoul(buf, &last, 0); |
| 401 | 401 | ||
| 402 | acquire_console_sem(); | 402 | console_lock(); |
| 403 | fb_set_suspend(fb_info, (int)state); | 403 | fb_set_suspend(fb_info, (int)state); |
| 404 | release_console_sem(); | 404 | console_unlock(); |
| 405 | 405 | ||
| 406 | return count; | 406 | return count; |
| 407 | } | 407 | } |
diff --git a/drivers/video/geode/gxfb_core.c b/drivers/video/geode/gxfb_core.c index 70b1d9d51c9..b4f19db9bb5 100644 --- a/drivers/video/geode/gxfb_core.c +++ b/drivers/video/geode/gxfb_core.c | |||
| @@ -344,10 +344,10 @@ static int gxfb_suspend(struct pci_dev *pdev, pm_message_t state) | |||
| 344 | struct fb_info *info = pci_get_drvdata(pdev); | 344 | struct fb_info *info = pci_get_drvdata(pdev); |
| 345 | 345 | ||
| 346 | if (state.event == PM_EVENT_SUSPEND) { | 346 | if (state.event == PM_EVENT_SUSPEND) { |
| 347 | acquire_console_sem(); | 347 | console_lock(); |
| 348 | gx_powerdown(info); | 348 | gx_powerdown(info); |
| 349 | fb_set_suspend(info, 1); | 349 | fb_set_suspend(info, 1); |
| 350 | release_console_sem(); | 350 | console_unlock(); |
| 351 | } | 351 | } |
| 352 | 352 | ||
| 353 | /* there's no point in setting PCI states; we emulate PCI, so | 353 | /* there's no point in setting PCI states; we emulate PCI, so |
| @@ -361,7 +361,7 @@ static int gxfb_resume(struct pci_dev *pdev) | |||
| 361 | struct fb_info *info = pci_get_drvdata(pdev); | 361 | struct fb_info *info = pci_get_drvdata(pdev); |
| 362 | int ret; | 362 | int ret; |
| 363 | 363 | ||
| 364 | acquire_console_sem(); | 364 | console_lock(); |
| 365 | ret = gx_powerup(info); | 365 | ret = gx_powerup(info); |
| 366 | if (ret) { | 366 | if (ret) { |
| 367 | printk(KERN_ERR "gxfb: power up failed!\n"); | 367 | printk(KERN_ERR "gxfb: power up failed!\n"); |
| @@ -369,7 +369,7 @@ static int gxfb_resume(struct pci_dev *pdev) | |||
| 369 | } | 369 | } |
| 370 | 370 | ||
| 371 | fb_set_suspend(info, 0); | 371 | fb_set_suspend(info, 0); |
| 372 | release_console_sem(); | 372 | console_unlock(); |
| 373 | return 0; | 373 | return 0; |
| 374 | } | 374 | } |
| 375 | #endif | 375 | #endif |
diff --git a/drivers/video/geode/lxfb_core.c b/drivers/video/geode/lxfb_core.c index 39bdbedf43b..416851ca875 100644 --- a/drivers/video/geode/lxfb_core.c +++ b/drivers/video/geode/lxfb_core.c | |||
| @@ -465,10 +465,10 @@ static int lxfb_suspend(struct pci_dev *pdev, pm_message_t state) | |||
| 465 | struct fb_info *info = pci_get_drvdata(pdev); | 465 | struct fb_info *info = pci_get_drvdata(pdev); |
| 466 | 466 | ||
| 467 | if (state.event == PM_EVENT_SUSPEND) { | 467 | if (state.event == PM_EVENT_SUSPEND) { |
| 468 | acquire_console_sem(); | 468 | console_lock(); |
| 469 | lx_powerdown(info); | 469 | lx_powerdown(info); |
| 470 | fb_set_suspend(info, 1); | 470 | fb_set_suspend(info, 1); |
| 471 | release_console_sem(); | 471 | console_unlock(); |
| 472 | } | 472 | } |
| 473 | 473 | ||
| 474 | /* there's no point in setting PCI states; we emulate PCI, so | 474 | /* there's no point in setting PCI states; we emulate PCI, so |
| @@ -482,7 +482,7 @@ static int lxfb_resume(struct pci_dev *pdev) | |||
| 482 | struct fb_info *info = pci_get_drvdata(pdev); | 482 | struct fb_info *info = pci_get_drvdata(pdev); |
| 483 | int ret; | 483 | int ret; |
| 484 | 484 | ||
| 485 | acquire_console_sem(); | 485 | console_lock(); |
| 486 | ret = lx_powerup(info); | 486 | ret = lx_powerup(info); |
| 487 | if (ret) { | 487 | if (ret) { |
| 488 | printk(KERN_ERR "lxfb: power up failed!\n"); | 488 | printk(KERN_ERR "lxfb: power up failed!\n"); |
| @@ -490,7 +490,7 @@ static int lxfb_resume(struct pci_dev *pdev) | |||
| 490 | } | 490 | } |
| 491 | 491 | ||
| 492 | fb_set_suspend(info, 0); | 492 | fb_set_suspend(info, 0); |
| 493 | release_console_sem(); | 493 | console_unlock(); |
| 494 | return 0; | 494 | return 0; |
| 495 | } | 495 | } |
| 496 | #else | 496 | #else |
diff --git a/drivers/video/i810/i810_main.c b/drivers/video/i810/i810_main.c index 5743ea25e81..318f6fb895b 100644 --- a/drivers/video/i810/i810_main.c +++ b/drivers/video/i810/i810_main.c | |||
| @@ -1574,7 +1574,7 @@ static int i810fb_suspend(struct pci_dev *dev, pm_message_t mesg) | |||
| 1574 | return 0; | 1574 | return 0; |
| 1575 | } | 1575 | } |
| 1576 | 1576 | ||
| 1577 | acquire_console_sem(); | 1577 | console_lock(); |
| 1578 | fb_set_suspend(info, 1); | 1578 | fb_set_suspend(info, 1); |
| 1579 | 1579 | ||
| 1580 | if (info->fbops->fb_sync) | 1580 | if (info->fbops->fb_sync) |
| @@ -1587,7 +1587,7 @@ static int i810fb_suspend(struct pci_dev *dev, pm_message_t mesg) | |||
| 1587 | pci_save_state(dev); | 1587 | pci_save_state(dev); |
| 1588 | pci_disable_device(dev); | 1588 | pci_disable_device(dev); |
| 1589 | pci_set_power_state(dev, pci_choose_state(dev, mesg)); | 1589 | pci_set_power_state(dev, pci_choose_state(dev, mesg)); |
| 1590 | release_console_sem(); | 1590 | console_unlock(); |
| 1591 | 1591 | ||
| 1592 | return 0; | 1592 | return 0; |
| 1593 | } | 1593 | } |
| @@ -1605,7 +1605,7 @@ static int i810fb_resume(struct pci_dev *dev) | |||
| 1605 | return 0; | 1605 | return 0; |
| 1606 | } | 1606 | } |
| 1607 | 1607 | ||
| 1608 | acquire_console_sem(); | 1608 | console_lock(); |
| 1609 | pci_set_power_state(dev, PCI_D0); | 1609 | pci_set_power_state(dev, PCI_D0); |
| 1610 | pci_restore_state(dev); | 1610 | pci_restore_state(dev); |
| 1611 | 1611 | ||
| @@ -1621,7 +1621,7 @@ static int i810fb_resume(struct pci_dev *dev) | |||
| 1621 | fb_set_suspend (info, 0); | 1621 | fb_set_suspend (info, 0); |
| 1622 | info->fbops->fb_blank(VESA_NO_BLANKING, info); | 1622 | info->fbops->fb_blank(VESA_NO_BLANKING, info); |
| 1623 | fail: | 1623 | fail: |
| 1624 | release_console_sem(); | 1624 | console_unlock(); |
| 1625 | return 0; | 1625 | return 0; |
| 1626 | } | 1626 | } |
| 1627 | /*********************************************************************** | 1627 | /*********************************************************************** |
diff --git a/drivers/video/jz4740_fb.c b/drivers/video/jz4740_fb.c index 670ecaa0385..de366937c93 100644 --- a/drivers/video/jz4740_fb.c +++ b/drivers/video/jz4740_fb.c | |||
| @@ -778,9 +778,9 @@ static int jzfb_suspend(struct device *dev) | |||
| 778 | { | 778 | { |
| 779 | struct jzfb *jzfb = dev_get_drvdata(dev); | 779 | struct jzfb *jzfb = dev_get_drvdata(dev); |
| 780 | 780 | ||
| 781 | acquire_console_sem(); | 781 | console_lock(); |
| 782 | fb_set_suspend(jzfb->fb, 1); | 782 | fb_set_suspend(jzfb->fb, 1); |
| 783 | release_console_sem(); | 783 | console_unlock(); |
| 784 | 784 | ||
| 785 | mutex_lock(&jzfb->lock); | 785 | mutex_lock(&jzfb->lock); |
| 786 | if (jzfb->is_enabled) | 786 | if (jzfb->is_enabled) |
| @@ -800,9 +800,9 @@ static int jzfb_resume(struct device *dev) | |||
| 800 | jzfb_enable(jzfb); | 800 | jzfb_enable(jzfb); |
| 801 | mutex_unlock(&jzfb->lock); | 801 | mutex_unlock(&jzfb->lock); |
| 802 | 802 | ||
| 803 | acquire_console_sem(); | 803 | console_lock(); |
| 804 | fb_set_suspend(jzfb->fb, 0); | 804 | fb_set_suspend(jzfb->fb, 0); |
| 805 | release_console_sem(); | 805 | console_unlock(); |
| 806 | 806 | ||
| 807 | return 0; | 807 | return 0; |
| 808 | } | 808 | } |
diff --git a/drivers/video/mx3fb.c b/drivers/video/mx3fb.c index cb013919e9c..7e3a490e8d7 100644 --- a/drivers/video/mx3fb.c +++ b/drivers/video/mx3fb.c | |||
| @@ -1177,9 +1177,9 @@ static int mx3fb_suspend(struct platform_device *pdev, pm_message_t state) | |||
| 1177 | struct mx3fb_data *mx3fb = platform_get_drvdata(pdev); | 1177 | struct mx3fb_data *mx3fb = platform_get_drvdata(pdev); |
| 1178 | struct mx3fb_info *mx3_fbi = mx3fb->fbi->par; | 1178 | struct mx3fb_info *mx3_fbi = mx3fb->fbi->par; |
| 1179 | 1179 | ||
| 1180 | acquire_console_sem(); | 1180 | console_lock(); |
| 1181 | fb_set_suspend(mx3fb->fbi, 1); | 1181 | fb_set_suspend(mx3fb->fbi, 1); |
| 1182 | release_console_sem(); | 1182 | console_unlock(); |
| 1183 | 1183 | ||
| 1184 | if (mx3_fbi->blank == FB_BLANK_UNBLANK) { | 1184 | if (mx3_fbi->blank == FB_BLANK_UNBLANK) { |
| 1185 | sdc_disable_channel(mx3_fbi); | 1185 | sdc_disable_channel(mx3_fbi); |
| @@ -1202,9 +1202,9 @@ static int mx3fb_resume(struct platform_device *pdev) | |||
| 1202 | sdc_set_brightness(mx3fb, mx3fb->backlight_level); | 1202 | sdc_set_brightness(mx3fb, mx3fb->backlight_level); |
| 1203 | } | 1203 | } |
| 1204 | 1204 | ||
| 1205 | acquire_console_sem(); | 1205 | console_lock(); |
| 1206 | fb_set_suspend(mx3fb->fbi, 0); | 1206 | fb_set_suspend(mx3fb->fbi, 0); |
| 1207 | release_console_sem(); | 1207 | console_unlock(); |
| 1208 | 1208 | ||
| 1209 | return 0; | 1209 | return 0; |
| 1210 | } | 1210 | } |
diff --git a/drivers/video/nuc900fb.c b/drivers/video/nuc900fb.c index 62498bd662f..f838d9e277f 100644 --- a/drivers/video/nuc900fb.c +++ b/drivers/video/nuc900fb.c | |||
| @@ -696,6 +696,8 @@ static int nuc900fb_remove(struct platform_device *pdev) | |||
| 696 | nuc900fb_stop_lcd(fbinfo); | 696 | nuc900fb_stop_lcd(fbinfo); |
| 697 | msleep(1); | 697 | msleep(1); |
| 698 | 698 | ||
| 699 | unregister_framebuffer(fbinfo); | ||
| 700 | nuc900fb_cpufreq_deregister(fbi); | ||
| 699 | nuc900fb_unmap_video_memory(fbinfo); | 701 | nuc900fb_unmap_video_memory(fbinfo); |
| 700 | 702 | ||
| 701 | iounmap(fbi->io); | 703 | iounmap(fbi->io); |
| @@ -723,7 +725,7 @@ static int nuc900fb_suspend(struct platform_device *dev, pm_message_t state) | |||
| 723 | struct fb_info *fbinfo = platform_get_drvdata(dev); | 725 | struct fb_info *fbinfo = platform_get_drvdata(dev); |
| 724 | struct nuc900fb_info *info = fbinfo->par; | 726 | struct nuc900fb_info *info = fbinfo->par; |
| 725 | 727 | ||
| 726 | nuc900fb_stop_lcd(); | 728 | nuc900fb_stop_lcd(fbinfo); |
| 727 | msleep(1); | 729 | msleep(1); |
| 728 | clk_disable(info->clk); | 730 | clk_disable(info->clk); |
| 729 | return 0; | 731 | return 0; |
| @@ -740,7 +742,7 @@ static int nuc900fb_resume(struct platform_device *dev) | |||
| 740 | msleep(1); | 742 | msleep(1); |
| 741 | 743 | ||
| 742 | nuc900fb_init_registers(fbinfo); | 744 | nuc900fb_init_registers(fbinfo); |
| 743 | nuc900fb_activate_var(bfinfo); | 745 | nuc900fb_activate_var(fbinfo); |
| 744 | 746 | ||
| 745 | return 0; | 747 | return 0; |
| 746 | } | 748 | } |
diff --git a/drivers/video/nvidia/nvidia.c b/drivers/video/nvidia/nvidia.c index efe10ff86d6..081dc474527 100644 --- a/drivers/video/nvidia/nvidia.c +++ b/drivers/video/nvidia/nvidia.c | |||
| @@ -1057,7 +1057,7 @@ static int nvidiafb_suspend(struct pci_dev *dev, pm_message_t mesg) | |||
| 1057 | 1057 | ||
| 1058 | if (mesg.event == PM_EVENT_PRETHAW) | 1058 | if (mesg.event == PM_EVENT_PRETHAW) |
| 1059 | mesg.event = PM_EVENT_FREEZE; | 1059 | mesg.event = PM_EVENT_FREEZE; |
| 1060 | acquire_console_sem(); | 1060 | console_lock(); |
| 1061 | par->pm_state = mesg.event; | 1061 | par->pm_state = mesg.event; |
| 1062 | 1062 | ||
| 1063 | if (mesg.event & PM_EVENT_SLEEP) { | 1063 | if (mesg.event & PM_EVENT_SLEEP) { |
| @@ -1070,7 +1070,7 @@ static int nvidiafb_suspend(struct pci_dev *dev, pm_message_t mesg) | |||
| 1070 | } | 1070 | } |
| 1071 | dev->dev.power.power_state = mesg; | 1071 | dev->dev.power.power_state = mesg; |
| 1072 | 1072 | ||
| 1073 | release_console_sem(); | 1073 | console_unlock(); |
| 1074 | return 0; | 1074 | return 0; |
| 1075 | } | 1075 | } |
| 1076 | 1076 | ||
| @@ -1079,7 +1079,7 @@ static int nvidiafb_resume(struct pci_dev *dev) | |||
| 1079 | struct fb_info *info = pci_get_drvdata(dev); | 1079 | struct fb_info *info = pci_get_drvdata(dev); |
| 1080 | struct nvidia_par *par = info->par; | 1080 | struct nvidia_par *par = info->par; |
| 1081 | 1081 | ||
| 1082 | acquire_console_sem(); | 1082 | console_lock(); |
| 1083 | pci_set_power_state(dev, PCI_D0); | 1083 | pci_set_power_state(dev, PCI_D0); |
| 1084 | 1084 | ||
| 1085 | if (par->pm_state != PM_EVENT_FREEZE) { | 1085 | if (par->pm_state != PM_EVENT_FREEZE) { |
| @@ -1097,7 +1097,7 @@ static int nvidiafb_resume(struct pci_dev *dev) | |||
| 1097 | nvidiafb_blank(FB_BLANK_UNBLANK, info); | 1097 | nvidiafb_blank(FB_BLANK_UNBLANK, info); |
| 1098 | 1098 | ||
| 1099 | fail: | 1099 | fail: |
| 1100 | release_console_sem(); | 1100 | console_unlock(); |
| 1101 | return 0; | 1101 | return 0; |
| 1102 | } | 1102 | } |
| 1103 | #else | 1103 | #else |
diff --git a/drivers/video/ps3fb.c b/drivers/video/ps3fb.c index 9c0144ee7ae..65560a1a043 100644 --- a/drivers/video/ps3fb.c +++ b/drivers/video/ps3fb.c | |||
| @@ -513,9 +513,9 @@ static int ps3fb_release(struct fb_info *info, int user) | |||
| 513 | if (atomic_dec_and_test(&ps3fb.f_count)) { | 513 | if (atomic_dec_and_test(&ps3fb.f_count)) { |
| 514 | if (atomic_read(&ps3fb.ext_flip)) { | 514 | if (atomic_read(&ps3fb.ext_flip)) { |
| 515 | atomic_set(&ps3fb.ext_flip, 0); | 515 | atomic_set(&ps3fb.ext_flip, 0); |
| 516 | if (!try_acquire_console_sem()) { | 516 | if (console_trylock()) { |
| 517 | ps3fb_sync(info, 0); /* single buffer */ | 517 | ps3fb_sync(info, 0); /* single buffer */ |
| 518 | release_console_sem(); | 518 | console_unlock(); |
| 519 | } | 519 | } |
| 520 | } | 520 | } |
| 521 | } | 521 | } |
| @@ -830,14 +830,14 @@ static int ps3fb_ioctl(struct fb_info *info, unsigned int cmd, | |||
| 830 | if (vmode) { | 830 | if (vmode) { |
| 831 | var = info->var; | 831 | var = info->var; |
| 832 | fb_videomode_to_var(&var, vmode); | 832 | fb_videomode_to_var(&var, vmode); |
| 833 | acquire_console_sem(); | 833 | console_lock(); |
| 834 | info->flags |= FBINFO_MISC_USEREVENT; | 834 | info->flags |= FBINFO_MISC_USEREVENT; |
| 835 | /* Force, in case only special bits changed */ | 835 | /* Force, in case only special bits changed */ |
| 836 | var.activate |= FB_ACTIVATE_FORCE; | 836 | var.activate |= FB_ACTIVATE_FORCE; |
| 837 | par->new_mode_id = val; | 837 | par->new_mode_id = val; |
| 838 | retval = fb_set_var(info, &var); | 838 | retval = fb_set_var(info, &var); |
| 839 | info->flags &= ~FBINFO_MISC_USEREVENT; | 839 | info->flags &= ~FBINFO_MISC_USEREVENT; |
| 840 | release_console_sem(); | 840 | console_unlock(); |
| 841 | } | 841 | } |
| 842 | break; | 842 | break; |
| 843 | } | 843 | } |
| @@ -881,9 +881,9 @@ static int ps3fb_ioctl(struct fb_info *info, unsigned int cmd, | |||
| 881 | break; | 881 | break; |
| 882 | 882 | ||
| 883 | dev_dbg(info->device, "PS3FB_IOCTL_FSEL:%d\n", val); | 883 | dev_dbg(info->device, "PS3FB_IOCTL_FSEL:%d\n", val); |
| 884 | acquire_console_sem(); | 884 | console_lock(); |
| 885 | retval = ps3fb_sync(info, val); | 885 | retval = ps3fb_sync(info, val); |
| 886 | release_console_sem(); | 886 | console_unlock(); |
| 887 | break; | 887 | break; |
| 888 | 888 | ||
| 889 | default: | 889 | default: |
| @@ -903,9 +903,9 @@ static int ps3fbd(void *arg) | |||
| 903 | set_current_state(TASK_INTERRUPTIBLE); | 903 | set_current_state(TASK_INTERRUPTIBLE); |
| 904 | if (ps3fb.is_kicked) { | 904 | if (ps3fb.is_kicked) { |
| 905 | ps3fb.is_kicked = 0; | 905 | ps3fb.is_kicked = 0; |
| 906 | acquire_console_sem(); | 906 | console_lock(); |
| 907 | ps3fb_sync(info, 0); /* single buffer */ | 907 | ps3fb_sync(info, 0); /* single buffer */ |
| 908 | release_console_sem(); | 908 | console_unlock(); |
| 909 | } | 909 | } |
| 910 | schedule(); | 910 | schedule(); |
| 911 | } | 911 | } |
diff --git a/drivers/video/pxa168fb.c b/drivers/video/pxa168fb.c index cea6403ae71..35f61dd0cb3 100644 --- a/drivers/video/pxa168fb.c +++ b/drivers/video/pxa168fb.c | |||
| @@ -701,16 +701,12 @@ static int __devinit pxa168fb_probe(struct platform_device *pdev) | |||
| 701 | */ | 701 | */ |
| 702 | pxa168fb_init_mode(info, mi); | 702 | pxa168fb_init_mode(info, mi); |
| 703 | 703 | ||
| 704 | ret = pxa168fb_check_var(&info->var, info); | ||
| 705 | if (ret) | ||
| 706 | goto failed_free_fbmem; | ||
| 707 | |||
| 708 | /* | 704 | /* |
| 709 | * Fill in sane defaults. | 705 | * Fill in sane defaults. |
| 710 | */ | 706 | */ |
| 711 | ret = pxa168fb_check_var(&info->var, info); | 707 | ret = pxa168fb_check_var(&info->var, info); |
| 712 | if (ret) | 708 | if (ret) |
| 713 | goto failed; | 709 | goto failed_free_fbmem; |
| 714 | 710 | ||
| 715 | /* | 711 | /* |
| 716 | * enable controller clock | 712 | * enable controller clock |
diff --git a/drivers/video/pxa3xx-gcu.c b/drivers/video/pxa3xx-gcu.c index b81168df253..cf4beb9dc9b 100644 --- a/drivers/video/pxa3xx-gcu.c +++ b/drivers/video/pxa3xx-gcu.c | |||
| @@ -1,5 +1,5 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * pxa3xx-gc.c - Linux kernel module for PXA3xx graphics controllers | 2 | * pxa3xx-gcu.c - Linux kernel module for PXA3xx graphics controllers |
| 3 | * | 3 | * |
| 4 | * This driver needs a DirectFB counterpart in user space, communication | 4 | * This driver needs a DirectFB counterpart in user space, communication |
| 5 | * is handled via mmap()ed memory areas and an ioctl. | 5 | * is handled via mmap()ed memory areas and an ioctl. |
| @@ -421,7 +421,7 @@ pxa3xx_gcu_misc_write(struct file *filp, const char *buff, | |||
| 421 | buffer->next = priv->free; | 421 | buffer->next = priv->free; |
| 422 | priv->free = buffer; | 422 | priv->free = buffer; |
| 423 | spin_unlock_irqrestore(&priv->spinlock, flags); | 423 | spin_unlock_irqrestore(&priv->spinlock, flags); |
| 424 | return ret; | 424 | return -EFAULT; |
| 425 | } | 425 | } |
| 426 | 426 | ||
| 427 | buffer->length = words; | 427 | buffer->length = words; |
diff --git a/drivers/video/s3fb.c b/drivers/video/s3fb.c index dce8c97b433..75738a92861 100644 --- a/drivers/video/s3fb.c +++ b/drivers/video/s3fb.c | |||
| @@ -22,7 +22,7 @@ | |||
| 22 | #include <linux/svga.h> | 22 | #include <linux/svga.h> |
| 23 | #include <linux/init.h> | 23 | #include <linux/init.h> |
| 24 | #include <linux/pci.h> | 24 | #include <linux/pci.h> |
| 25 | #include <linux/console.h> /* Why should fb driver call console functions? because acquire_console_sem() */ | 25 | #include <linux/console.h> /* Why should fb driver call console functions? because console_lock() */ |
| 26 | #include <video/vga.h> | 26 | #include <video/vga.h> |
| 27 | 27 | ||
| 28 | #ifdef CONFIG_MTRR | 28 | #ifdef CONFIG_MTRR |
| @@ -1113,12 +1113,12 @@ static int s3_pci_suspend(struct pci_dev* dev, pm_message_t state) | |||
| 1113 | 1113 | ||
| 1114 | dev_info(info->device, "suspend\n"); | 1114 | dev_info(info->device, "suspend\n"); |
| 1115 | 1115 | ||
| 1116 | acquire_console_sem(); | 1116 | console_lock(); |
| 1117 | mutex_lock(&(par->open_lock)); | 1117 | mutex_lock(&(par->open_lock)); |
| 1118 | 1118 | ||
| 1119 | if ((state.event == PM_EVENT_FREEZE) || (par->ref_count == 0)) { | 1119 | if ((state.event == PM_EVENT_FREEZE) || (par->ref_count == 0)) { |
| 1120 | mutex_unlock(&(par->open_lock)); | 1120 | mutex_unlock(&(par->open_lock)); |
| 1121 | release_console_sem(); | 1121 | console_unlock(); |
| 1122 | return 0; | 1122 | return 0; |
| 1123 | } | 1123 | } |
| 1124 | 1124 | ||
| @@ -1129,7 +1129,7 @@ static int s3_pci_suspend(struct pci_dev* dev, pm_message_t state) | |||
| 1129 | pci_set_power_state(dev, pci_choose_state(dev, state)); | 1129 | pci_set_power_state(dev, pci_choose_state(dev, state)); |
| 1130 | 1130 | ||
| 1131 | mutex_unlock(&(par->open_lock)); | 1131 | mutex_unlock(&(par->open_lock)); |
| 1132 | release_console_sem(); | 1132 | console_unlock(); |
| 1133 | 1133 | ||
| 1134 | return 0; | 1134 | return 0; |
| 1135 | } | 1135 | } |
| @@ -1145,12 +1145,12 @@ static int s3_pci_resume(struct pci_dev* dev) | |||
| 1145 | 1145 | ||
| 1146 | dev_info(info->device, "resume\n"); | 1146 | dev_info(info->device, "resume\n"); |
| 1147 | 1147 | ||
| 1148 | acquire_console_sem(); | 1148 | console_lock(); |
| 1149 | mutex_lock(&(par->open_lock)); | 1149 | mutex_lock(&(par->open_lock)); |
| 1150 | 1150 | ||
| 1151 | if (par->ref_count == 0) { | 1151 | if (par->ref_count == 0) { |
| 1152 | mutex_unlock(&(par->open_lock)); | 1152 | mutex_unlock(&(par->open_lock)); |
| 1153 | release_console_sem(); | 1153 | console_unlock(); |
| 1154 | return 0; | 1154 | return 0; |
| 1155 | } | 1155 | } |
| 1156 | 1156 | ||
| @@ -1159,7 +1159,7 @@ static int s3_pci_resume(struct pci_dev* dev) | |||
| 1159 | err = pci_enable_device(dev); | 1159 | err = pci_enable_device(dev); |
| 1160 | if (err) { | 1160 | if (err) { |
| 1161 | mutex_unlock(&(par->open_lock)); | 1161 | mutex_unlock(&(par->open_lock)); |
| 1162 | release_console_sem(); | 1162 | console_unlock(); |
| 1163 | dev_err(info->device, "error %d enabling device for resume\n", err); | 1163 | dev_err(info->device, "error %d enabling device for resume\n", err); |
| 1164 | return err; | 1164 | return err; |
| 1165 | } | 1165 | } |
| @@ -1169,7 +1169,7 @@ static int s3_pci_resume(struct pci_dev* dev) | |||
| 1169 | fb_set_suspend(info, 0); | 1169 | fb_set_suspend(info, 0); |
| 1170 | 1170 | ||
| 1171 | mutex_unlock(&(par->open_lock)); | 1171 | mutex_unlock(&(par->open_lock)); |
| 1172 | release_console_sem(); | 1172 | console_unlock(); |
| 1173 | 1173 | ||
| 1174 | return 0; | 1174 | return 0; |
| 1175 | } | 1175 | } |
diff --git a/drivers/video/savage/savagefb_driver.c b/drivers/video/savage/savagefb_driver.c index 842d157e102..487911e2926 100644 --- a/drivers/video/savage/savagefb_driver.c +++ b/drivers/video/savage/savagefb_driver.c | |||
| @@ -2373,7 +2373,7 @@ static int savagefb_suspend(struct pci_dev *dev, pm_message_t mesg) | |||
| 2373 | if (mesg.event == PM_EVENT_FREEZE) | 2373 | if (mesg.event == PM_EVENT_FREEZE) |
| 2374 | return 0; | 2374 | return 0; |
| 2375 | 2375 | ||
| 2376 | acquire_console_sem(); | 2376 | console_lock(); |
| 2377 | fb_set_suspend(info, 1); | 2377 | fb_set_suspend(info, 1); |
| 2378 | 2378 | ||
| 2379 | if (info->fbops->fb_sync) | 2379 | if (info->fbops->fb_sync) |
| @@ -2385,7 +2385,7 @@ static int savagefb_suspend(struct pci_dev *dev, pm_message_t mesg) | |||
| 2385 | pci_save_state(dev); | 2385 | pci_save_state(dev); |
| 2386 | pci_disable_device(dev); | 2386 | pci_disable_device(dev); |
| 2387 | pci_set_power_state(dev, pci_choose_state(dev, mesg)); | 2387 | pci_set_power_state(dev, pci_choose_state(dev, mesg)); |
| 2388 | release_console_sem(); | 2388 | console_unlock(); |
| 2389 | 2389 | ||
| 2390 | return 0; | 2390 | return 0; |
| 2391 | } | 2391 | } |
| @@ -2409,7 +2409,7 @@ static int savagefb_resume(struct pci_dev* dev) | |||
| 2409 | return 0; | 2409 | return 0; |
| 2410 | } | 2410 | } |
| 2411 | 2411 | ||
| 2412 | acquire_console_sem(); | 2412 | console_lock(); |
| 2413 | 2413 | ||
| 2414 | pci_set_power_state(dev, PCI_D0); | 2414 | pci_set_power_state(dev, PCI_D0); |
| 2415 | pci_restore_state(dev); | 2415 | pci_restore_state(dev); |
| @@ -2423,7 +2423,7 @@ static int savagefb_resume(struct pci_dev* dev) | |||
| 2423 | savagefb_set_par(info); | 2423 | savagefb_set_par(info); |
| 2424 | fb_set_suspend(info, 0); | 2424 | fb_set_suspend(info, 0); |
| 2425 | savagefb_blank(FB_BLANK_UNBLANK, info); | 2425 | savagefb_blank(FB_BLANK_UNBLANK, info); |
| 2426 | release_console_sem(); | 2426 | console_unlock(); |
| 2427 | 2427 | ||
| 2428 | return 0; | 2428 | return 0; |
| 2429 | } | 2429 | } |
diff --git a/drivers/video/sh_mobile_hdmi.c b/drivers/video/sh_mobile_hdmi.c index 74d9f546a2e..2b9e56a6bde 100644 --- a/drivers/video/sh_mobile_hdmi.c +++ b/drivers/video/sh_mobile_hdmi.c | |||
| @@ -1151,7 +1151,7 @@ static void sh_hdmi_edid_work_fn(struct work_struct *work) | |||
| 1151 | 1151 | ||
| 1152 | ch = info->par; | 1152 | ch = info->par; |
| 1153 | 1153 | ||
| 1154 | acquire_console_sem(); | 1154 | console_lock(); |
| 1155 | 1155 | ||
| 1156 | /* HDMI plug in */ | 1156 | /* HDMI plug in */ |
| 1157 | if (!sh_hdmi_must_reconfigure(hdmi) && | 1157 | if (!sh_hdmi_must_reconfigure(hdmi) && |
| @@ -1171,7 +1171,7 @@ static void sh_hdmi_edid_work_fn(struct work_struct *work) | |||
| 1171 | fb_set_suspend(info, 0); | 1171 | fb_set_suspend(info, 0); |
| 1172 | } | 1172 | } |
| 1173 | 1173 | ||
| 1174 | release_console_sem(); | 1174 | console_unlock(); |
| 1175 | } else { | 1175 | } else { |
| 1176 | ret = 0; | 1176 | ret = 0; |
| 1177 | if (!hdmi->info) | 1177 | if (!hdmi->info) |
| @@ -1181,12 +1181,12 @@ static void sh_hdmi_edid_work_fn(struct work_struct *work) | |||
| 1181 | fb_destroy_modedb(hdmi->monspec.modedb); | 1181 | fb_destroy_modedb(hdmi->monspec.modedb); |
| 1182 | hdmi->monspec.modedb = NULL; | 1182 | hdmi->monspec.modedb = NULL; |
| 1183 | 1183 | ||
| 1184 | acquire_console_sem(); | 1184 | console_lock(); |
| 1185 | 1185 | ||
| 1186 | /* HDMI disconnect */ | 1186 | /* HDMI disconnect */ |
| 1187 | fb_set_suspend(hdmi->info, 1); | 1187 | fb_set_suspend(hdmi->info, 1); |
| 1188 | 1188 | ||
| 1189 | release_console_sem(); | 1189 | console_unlock(); |
| 1190 | pm_runtime_put(hdmi->dev); | 1190 | pm_runtime_put(hdmi->dev); |
| 1191 | } | 1191 | } |
| 1192 | 1192 | ||
diff --git a/drivers/video/sh_mobile_lcdcfb.c b/drivers/video/sh_mobile_lcdcfb.c index bd4840a8a6b..bf12e53aed5 100644 --- a/drivers/video/sh_mobile_lcdcfb.c +++ b/drivers/video/sh_mobile_lcdcfb.c | |||
| @@ -912,9 +912,9 @@ static int sh_mobile_release(struct fb_info *info, int user) | |||
| 912 | 912 | ||
| 913 | /* Nothing to reconfigure, when called from fbcon */ | 913 | /* Nothing to reconfigure, when called from fbcon */ |
| 914 | if (user) { | 914 | if (user) { |
| 915 | acquire_console_sem(); | 915 | console_lock(); |
| 916 | sh_mobile_fb_reconfig(info); | 916 | sh_mobile_fb_reconfig(info); |
| 917 | release_console_sem(); | 917 | console_unlock(); |
| 918 | } | 918 | } |
| 919 | 919 | ||
| 920 | mutex_unlock(&ch->open_lock); | 920 | mutex_unlock(&ch->open_lock); |
diff --git a/drivers/video/sm501fb.c b/drivers/video/sm501fb.c index b7dc1800efa..bcb44a594eb 100644 --- a/drivers/video/sm501fb.c +++ b/drivers/video/sm501fb.c | |||
| @@ -2010,9 +2010,9 @@ static int sm501fb_suspend_fb(struct sm501fb_info *info, | |||
| 2010 | 2010 | ||
| 2011 | /* tell console/fb driver we are suspending */ | 2011 | /* tell console/fb driver we are suspending */ |
| 2012 | 2012 | ||
| 2013 | acquire_console_sem(); | 2013 | console_lock(); |
| 2014 | fb_set_suspend(fbi, 1); | 2014 | fb_set_suspend(fbi, 1); |
| 2015 | release_console_sem(); | 2015 | console_unlock(); |
| 2016 | 2016 | ||
| 2017 | /* backup copies in case chip is powered down over suspend */ | 2017 | /* backup copies in case chip is powered down over suspend */ |
| 2018 | 2018 | ||
| @@ -2069,9 +2069,9 @@ static void sm501fb_resume_fb(struct sm501fb_info *info, | |||
| 2069 | memcpy_toio(par->cursor.k_addr, par->store_cursor, | 2069 | memcpy_toio(par->cursor.k_addr, par->store_cursor, |
| 2070 | par->cursor.size); | 2070 | par->cursor.size); |
| 2071 | 2071 | ||
| 2072 | acquire_console_sem(); | 2072 | console_lock(); |
| 2073 | fb_set_suspend(fbi, 0); | 2073 | fb_set_suspend(fbi, 0); |
| 2074 | release_console_sem(); | 2074 | console_unlock(); |
| 2075 | 2075 | ||
| 2076 | vfree(par->store_fb); | 2076 | vfree(par->store_fb); |
| 2077 | vfree(par->store_cursor); | 2077 | vfree(par->store_cursor); |
diff --git a/drivers/video/tmiofb.c b/drivers/video/tmiofb.c index 6913fe168c2..dfef88c803d 100644 --- a/drivers/video/tmiofb.c +++ b/drivers/video/tmiofb.c | |||
| @@ -25,7 +25,7 @@ | |||
| 25 | #include <linux/fb.h> | 25 | #include <linux/fb.h> |
| 26 | #include <linux/interrupt.h> | 26 | #include <linux/interrupt.h> |
| 27 | #include <linux/delay.h> | 27 | #include <linux/delay.h> |
| 28 | /* Why should fb driver call console functions? because acquire_console_sem() */ | 28 | /* Why should fb driver call console functions? because console_lock() */ |
| 29 | #include <linux/console.h> | 29 | #include <linux/console.h> |
| 30 | #include <linux/mfd/core.h> | 30 | #include <linux/mfd/core.h> |
| 31 | #include <linux/mfd/tmio.h> | 31 | #include <linux/mfd/tmio.h> |
| @@ -944,7 +944,7 @@ static int tmiofb_suspend(struct platform_device *dev, pm_message_t state) | |||
| 944 | struct mfd_cell *cell = dev->dev.platform_data; | 944 | struct mfd_cell *cell = dev->dev.platform_data; |
| 945 | int retval = 0; | 945 | int retval = 0; |
| 946 | 946 | ||
| 947 | acquire_console_sem(); | 947 | console_lock(); |
| 948 | 948 | ||
| 949 | fb_set_suspend(info, 1); | 949 | fb_set_suspend(info, 1); |
| 950 | 950 | ||
| @@ -965,7 +965,7 @@ static int tmiofb_suspend(struct platform_device *dev, pm_message_t state) | |||
| 965 | if (cell->suspend) | 965 | if (cell->suspend) |
| 966 | retval = cell->suspend(dev); | 966 | retval = cell->suspend(dev); |
| 967 | 967 | ||
| 968 | release_console_sem(); | 968 | console_unlock(); |
| 969 | 969 | ||
| 970 | return retval; | 970 | return retval; |
| 971 | } | 971 | } |
| @@ -976,7 +976,7 @@ static int tmiofb_resume(struct platform_device *dev) | |||
| 976 | struct mfd_cell *cell = dev->dev.platform_data; | 976 | struct mfd_cell *cell = dev->dev.platform_data; |
| 977 | int retval = 0; | 977 | int retval = 0; |
| 978 | 978 | ||
| 979 | acquire_console_sem(); | 979 | console_lock(); |
| 980 | 980 | ||
| 981 | if (cell->resume) { | 981 | if (cell->resume) { |
| 982 | retval = cell->resume(dev); | 982 | retval = cell->resume(dev); |
| @@ -992,7 +992,7 @@ static int tmiofb_resume(struct platform_device *dev) | |||
| 992 | 992 | ||
| 993 | fb_set_suspend(info, 0); | 993 | fb_set_suspend(info, 0); |
| 994 | out: | 994 | out: |
| 995 | release_console_sem(); | 995 | console_unlock(); |
| 996 | return retval; | 996 | return retval; |
| 997 | } | 997 | } |
| 998 | #else | 998 | #else |
diff --git a/drivers/video/via/viafbdev.c b/drivers/video/via/viafbdev.c index 289edd51952..4e66349e436 100644 --- a/drivers/video/via/viafbdev.c +++ b/drivers/video/via/viafbdev.c | |||
| @@ -1674,17 +1674,17 @@ static int parse_mode(const char *str, u32 *xres, u32 *yres) | |||
| 1674 | #ifdef CONFIG_PM | 1674 | #ifdef CONFIG_PM |
| 1675 | static int viafb_suspend(void *unused) | 1675 | static int viafb_suspend(void *unused) |
| 1676 | { | 1676 | { |
| 1677 | acquire_console_sem(); | 1677 | console_lock(); |
| 1678 | fb_set_suspend(viafbinfo, 1); | 1678 | fb_set_suspend(viafbinfo, 1); |
| 1679 | viafb_sync(viafbinfo); | 1679 | viafb_sync(viafbinfo); |
| 1680 | release_console_sem(); | 1680 | console_unlock(); |
| 1681 | 1681 | ||
| 1682 | return 0; | 1682 | return 0; |
| 1683 | } | 1683 | } |
| 1684 | 1684 | ||
| 1685 | static int viafb_resume(void *unused) | 1685 | static int viafb_resume(void *unused) |
| 1686 | { | 1686 | { |
| 1687 | acquire_console_sem(); | 1687 | console_lock(); |
| 1688 | if (viaparinfo->shared->vdev->engine_mmio) | 1688 | if (viaparinfo->shared->vdev->engine_mmio) |
| 1689 | viafb_reset_engine(viaparinfo); | 1689 | viafb_reset_engine(viaparinfo); |
| 1690 | viafb_set_par(viafbinfo); | 1690 | viafb_set_par(viafbinfo); |
| @@ -1692,7 +1692,7 @@ static int viafb_resume(void *unused) | |||
| 1692 | viafb_set_par(viafbinfo1); | 1692 | viafb_set_par(viafbinfo1); |
| 1693 | fb_set_suspend(viafbinfo, 0); | 1693 | fb_set_suspend(viafbinfo, 0); |
| 1694 | 1694 | ||
| 1695 | release_console_sem(); | 1695 | console_unlock(); |
| 1696 | return 0; | 1696 | return 0; |
| 1697 | } | 1697 | } |
| 1698 | 1698 | ||
diff --git a/drivers/video/vt8623fb.c b/drivers/video/vt8623fb.c index 85d76ec4c63..a2965ab92cf 100644 --- a/drivers/video/vt8623fb.c +++ b/drivers/video/vt8623fb.c | |||
| @@ -23,7 +23,7 @@ | |||
| 23 | #include <linux/svga.h> | 23 | #include <linux/svga.h> |
| 24 | #include <linux/init.h> | 24 | #include <linux/init.h> |
| 25 | #include <linux/pci.h> | 25 | #include <linux/pci.h> |
| 26 | #include <linux/console.h> /* Why should fb driver call console functions? because acquire_console_sem() */ | 26 | #include <linux/console.h> /* Why should fb driver call console functions? because console_lock() */ |
| 27 | #include <video/vga.h> | 27 | #include <video/vga.h> |
| 28 | 28 | ||
| 29 | #ifdef CONFIG_MTRR | 29 | #ifdef CONFIG_MTRR |
| @@ -819,12 +819,12 @@ static int vt8623_pci_suspend(struct pci_dev* dev, pm_message_t state) | |||
| 819 | 819 | ||
| 820 | dev_info(info->device, "suspend\n"); | 820 | dev_info(info->device, "suspend\n"); |
| 821 | 821 | ||
| 822 | acquire_console_sem(); | 822 | console_lock(); |
| 823 | mutex_lock(&(par->open_lock)); | 823 | mutex_lock(&(par->open_lock)); |
| 824 | 824 | ||
| 825 | if ((state.event == PM_EVENT_FREEZE) || (par->ref_count == 0)) { | 825 | if ((state.event == PM_EVENT_FREEZE) || (par->ref_count == 0)) { |
| 826 | mutex_unlock(&(par->open_lock)); | 826 | mutex_unlock(&(par->open_lock)); |
| 827 | release_console_sem(); | 827 | console_unlock(); |
| 828 | return 0; | 828 | return 0; |
| 829 | } | 829 | } |
| 830 | 830 | ||
| @@ -835,7 +835,7 @@ static int vt8623_pci_suspend(struct pci_dev* dev, pm_message_t state) | |||
| 835 | pci_set_power_state(dev, pci_choose_state(dev, state)); | 835 | pci_set_power_state(dev, pci_choose_state(dev, state)); |
| 836 | 836 | ||
| 837 | mutex_unlock(&(par->open_lock)); | 837 | mutex_unlock(&(par->open_lock)); |
| 838 | release_console_sem(); | 838 | console_unlock(); |
| 839 | 839 | ||
| 840 | return 0; | 840 | return 0; |
| 841 | } | 841 | } |
| @@ -850,7 +850,7 @@ static int vt8623_pci_resume(struct pci_dev* dev) | |||
| 850 | 850 | ||
| 851 | dev_info(info->device, "resume\n"); | 851 | dev_info(info->device, "resume\n"); |
| 852 | 852 | ||
| 853 | acquire_console_sem(); | 853 | console_lock(); |
| 854 | mutex_lock(&(par->open_lock)); | 854 | mutex_lock(&(par->open_lock)); |
| 855 | 855 | ||
| 856 | if (par->ref_count == 0) | 856 | if (par->ref_count == 0) |
| @@ -869,7 +869,7 @@ static int vt8623_pci_resume(struct pci_dev* dev) | |||
| 869 | 869 | ||
| 870 | fail: | 870 | fail: |
| 871 | mutex_unlock(&(par->open_lock)); | 871 | mutex_unlock(&(par->open_lock)); |
| 872 | release_console_sem(); | 872 | console_unlock(); |
| 873 | 873 | ||
| 874 | return 0; | 874 | return 0; |
| 875 | } | 875 | } |
diff --git a/drivers/video/xen-fbfront.c b/drivers/video/xen-fbfront.c index 3e6934d4bea..a20218c2fda 100644 --- a/drivers/video/xen-fbfront.c +++ b/drivers/video/xen-fbfront.c | |||
| @@ -491,12 +491,12 @@ xenfb_make_preferred_console(void) | |||
| 491 | if (console_set_on_cmdline) | 491 | if (console_set_on_cmdline) |
| 492 | return; | 492 | return; |
| 493 | 493 | ||
| 494 | acquire_console_sem(); | 494 | console_lock(); |
| 495 | for_each_console(c) { | 495 | for_each_console(c) { |
| 496 | if (!strcmp(c->name, "tty") && c->index == 0) | 496 | if (!strcmp(c->name, "tty") && c->index == 0) |
| 497 | break; | 497 | break; |
| 498 | } | 498 | } |
| 499 | release_console_sem(); | 499 | console_unlock(); |
| 500 | if (c) { | 500 | if (c) { |
| 501 | unregister_console(c); | 501 | unregister_console(c); |
| 502 | c->flags |= CON_CONSDEV; | 502 | c->flags |= CON_CONSDEV; |
