Commit ebb067d2 authored by Linus Torvalds's avatar Linus Torvalds

Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/s390/linux

Pull s390 updates from Martin Schwidefsky:
 "Mostly cleanups and bug-fixes, with two exceptions.

  The first is lazy flushing of I/O-TLBs for PCI to improve performance,
  the second is software dirty bits in the pmd for the madvise-free
  implementation"

* 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/s390/linux: (24 commits)
  s390/locking: Reenable optimistic spinning
  s390/mm: implement dirty bits for large segment table entries
  KVM: s390/mm: Fix page table locking vs. split pmd lock
  s390/dasd: fix camel case
  s390/3215: fix hanging console issue
  s390/irq: improve displayed interrupt order in /proc/interrupts
  s390/seccomp: fix error return for filtered system calls
  s390/pci: introduce lazy IOTLB flushing for DMA unmap
  dasd: fix error recovery for alias devices during format
  dasd: fix list_del corruption during format
  dasd: fix unresponsive device during format
  dasd: use aliases for formatted devices during format
  s390/pci: fix kmsg component
  s390/kdump: Return NOTIFY_OK for all actions other than MEM_GOING_OFFLINE
  s390/watchdog: Fix module name in Kconfig help text
  s390/dasd: replace seq_printf by seq_puts
  s390/dasd: replace pr_warning by pr_warn
  s390/dasd: Move EXPORT_SYMBOL after function/variable
  s390/dasd: remove unnecessary null test before debugfs_remove
  s390/zfcp: use qdio buffer helpers
  ...
parents 33caee39 36e7fdaa
...@@ -3058,6 +3058,13 @@ bytes respectively. Such letter suffixes can also be entirely omitted. ...@@ -3058,6 +3058,13 @@ bytes respectively. Such letter suffixes can also be entirely omitted.
S [KNL] Run init in single mode S [KNL] Run init in single mode
s390_iommu= [HW,S390]
Set s390 IOTLB flushing mode
strict
With strict flushing every unmap operation will result in
an IOTLB flush. Default is lazy flushing before reuse,
which is faster.
sa1100ir [NET] sa1100ir [NET]
See drivers/net/irda/sa1100_ir.c. See drivers/net/irda/sa1100_ir.c.
......
...@@ -92,6 +92,7 @@ config S390 ...@@ -92,6 +92,7 @@ config S390
select ARCH_INLINE_WRITE_UNLOCK_IRQ select ARCH_INLINE_WRITE_UNLOCK_IRQ
select ARCH_INLINE_WRITE_UNLOCK_IRQRESTORE select ARCH_INLINE_WRITE_UNLOCK_IRQRESTORE
select ARCH_SAVE_PAGE_KEYS if HIBERNATION select ARCH_SAVE_PAGE_KEYS if HIBERNATION
select ARCH_SUPPORTS_ATOMIC_RMW
select ARCH_USE_CMPXCHG_LOCKREF select ARCH_USE_CMPXCHG_LOCKREF
select ARCH_WANT_IPC_PARSE_VERSION select ARCH_WANT_IPC_PARSE_VERSION
select BUILDTIME_EXTABLE_SORT select BUILDTIME_EXTABLE_SORT
......
This diff is collapsed.
...@@ -415,6 +415,10 @@ struct qdio_brinfo_entry_l2 { ...@@ -415,6 +415,10 @@ struct qdio_brinfo_entry_l2 {
#define QDIO_FLAG_SYNC_OUTPUT 0x02 #define QDIO_FLAG_SYNC_OUTPUT 0x02
#define QDIO_FLAG_PCI_OUT 0x10 #define QDIO_FLAG_PCI_OUT 0x10
int qdio_alloc_buffers(struct qdio_buffer **buf, unsigned int count);
void qdio_free_buffers(struct qdio_buffer **buf, unsigned int count);
void qdio_reset_buffers(struct qdio_buffer **buf, unsigned int count);
extern int qdio_allocate(struct qdio_initialize *); extern int qdio_allocate(struct qdio_initialize *);
extern int qdio_establish(struct qdio_initialize *); extern int qdio_establish(struct qdio_initialize *);
extern int qdio_activate(struct ccw_device *); extern int qdio_activate(struct ccw_device *);
......
...@@ -54,7 +54,7 @@ static inline void syscall_set_return_value(struct task_struct *task, ...@@ -54,7 +54,7 @@ static inline void syscall_set_return_value(struct task_struct *task,
struct pt_regs *regs, struct pt_regs *regs,
int error, long val) int error, long val)
{ {
regs->gprs[2] = error ? -error : val; regs->gprs[2] = error ? error : val;
} }
static inline void syscall_get_arguments(struct task_struct *task, static inline void syscall_get_arguments(struct task_struct *task,
......
...@@ -30,6 +30,7 @@ DEFINE_PER_CPU_SHARED_ALIGNED(struct irq_stat, irq_stat); ...@@ -30,6 +30,7 @@ DEFINE_PER_CPU_SHARED_ALIGNED(struct irq_stat, irq_stat);
EXPORT_PER_CPU_SYMBOL_GPL(irq_stat); EXPORT_PER_CPU_SYMBOL_GPL(irq_stat);
struct irq_class { struct irq_class {
int irq;
char *name; char *name;
char *desc; char *desc;
}; };
...@@ -45,9 +46,9 @@ struct irq_class { ...@@ -45,9 +46,9 @@ struct irq_class {
* up with having a sum which accounts each interrupt twice. * up with having a sum which accounts each interrupt twice.
*/ */
static const struct irq_class irqclass_main_desc[NR_IRQS_BASE] = { static const struct irq_class irqclass_main_desc[NR_IRQS_BASE] = {
[EXT_INTERRUPT] = {.name = "EXT"}, {.irq = EXT_INTERRUPT, .name = "EXT"},
[IO_INTERRUPT] = {.name = "I/O"}, {.irq = IO_INTERRUPT, .name = "I/O"},
[THIN_INTERRUPT] = {.name = "AIO"}, {.irq = THIN_INTERRUPT, .name = "AIO"},
}; };
/* /*
...@@ -56,38 +57,38 @@ static const struct irq_class irqclass_main_desc[NR_IRQS_BASE] = { ...@@ -56,38 +57,38 @@ static const struct irq_class irqclass_main_desc[NR_IRQS_BASE] = {
* In addition this list contains non external / I/O events like NMIs. * In addition this list contains non external / I/O events like NMIs.
*/ */
static const struct irq_class irqclass_sub_desc[NR_ARCH_IRQS] = { static const struct irq_class irqclass_sub_desc[NR_ARCH_IRQS] = {
[IRQEXT_CLK] = {.name = "CLK", .desc = "[EXT] Clock Comparator"}, {.irq = IRQEXT_CLK, .name = "CLK", .desc = "[EXT] Clock Comparator"},
[IRQEXT_EXC] = {.name = "EXC", .desc = "[EXT] External Call"}, {.irq = IRQEXT_EXC, .name = "EXC", .desc = "[EXT] External Call"},
[IRQEXT_EMS] = {.name = "EMS", .desc = "[EXT] Emergency Signal"}, {.irq = IRQEXT_EMS, .name = "EMS", .desc = "[EXT] Emergency Signal"},
[IRQEXT_TMR] = {.name = "TMR", .desc = "[EXT] CPU Timer"}, {.irq = IRQEXT_TMR, .name = "TMR", .desc = "[EXT] CPU Timer"},
[IRQEXT_TLA] = {.name = "TAL", .desc = "[EXT] Timing Alert"}, {.irq = IRQEXT_TLA, .name = "TAL", .desc = "[EXT] Timing Alert"},
[IRQEXT_PFL] = {.name = "PFL", .desc = "[EXT] Pseudo Page Fault"}, {.irq = IRQEXT_PFL, .name = "PFL", .desc = "[EXT] Pseudo Page Fault"},
[IRQEXT_DSD] = {.name = "DSD", .desc = "[EXT] DASD Diag"}, {.irq = IRQEXT_DSD, .name = "DSD", .desc = "[EXT] DASD Diag"},
[IRQEXT_VRT] = {.name = "VRT", .desc = "[EXT] Virtio"}, {.irq = IRQEXT_VRT, .name = "VRT", .desc = "[EXT] Virtio"},
[IRQEXT_SCP] = {.name = "SCP", .desc = "[EXT] Service Call"}, {.irq = IRQEXT_SCP, .name = "SCP", .desc = "[EXT] Service Call"},
[IRQEXT_IUC] = {.name = "IUC", .desc = "[EXT] IUCV"}, {.irq = IRQEXT_IUC, .name = "IUC", .desc = "[EXT] IUCV"},
[IRQEXT_CMS] = {.name = "CMS", .desc = "[EXT] CPU-Measurement: Sampling"}, {.irq = IRQEXT_CMS, .name = "CMS", .desc = "[EXT] CPU-Measurement: Sampling"},
[IRQEXT_CMC] = {.name = "CMC", .desc = "[EXT] CPU-Measurement: Counter"}, {.irq = IRQEXT_CMC, .name = "CMC", .desc = "[EXT] CPU-Measurement: Counter"},
[IRQEXT_CMR] = {.name = "CMR", .desc = "[EXT] CPU-Measurement: RI"}, {.irq = IRQEXT_CMR, .name = "CMR", .desc = "[EXT] CPU-Measurement: RI"},
[IRQIO_CIO] = {.name = "CIO", .desc = "[I/O] Common I/O Layer Interrupt"}, {.irq = IRQIO_CIO, .name = "CIO", .desc = "[I/O] Common I/O Layer Interrupt"},
[IRQIO_QAI] = {.name = "QAI", .desc = "[I/O] QDIO Adapter Interrupt"}, {.irq = IRQIO_QAI, .name = "QAI", .desc = "[I/O] QDIO Adapter Interrupt"},
[IRQIO_DAS] = {.name = "DAS", .desc = "[I/O] DASD"}, {.irq = IRQIO_DAS, .name = "DAS", .desc = "[I/O] DASD"},
[IRQIO_C15] = {.name = "C15", .desc = "[I/O] 3215"}, {.irq = IRQIO_C15, .name = "C15", .desc = "[I/O] 3215"},
[IRQIO_C70] = {.name = "C70", .desc = "[I/O] 3270"}, {.irq = IRQIO_C70, .name = "C70", .desc = "[I/O] 3270"},
[IRQIO_TAP] = {.name = "TAP", .desc = "[I/O] Tape"}, {.irq = IRQIO_TAP, .name = "TAP", .desc = "[I/O] Tape"},
[IRQIO_VMR] = {.name = "VMR", .desc = "[I/O] Unit Record Devices"}, {.irq = IRQIO_VMR, .name = "VMR", .desc = "[I/O] Unit Record Devices"},
[IRQIO_LCS] = {.name = "LCS", .desc = "[I/O] LCS"}, {.irq = IRQIO_LCS, .name = "LCS", .desc = "[I/O] LCS"},
[IRQIO_CLW] = {.name = "CLW", .desc = "[I/O] CLAW"}, {.irq = IRQIO_CLW, .name = "CLW", .desc = "[I/O] CLAW"},
[IRQIO_CTC] = {.name = "CTC", .desc = "[I/O] CTC"}, {.irq = IRQIO_CTC, .name = "CTC", .desc = "[I/O] CTC"},
[IRQIO_APB] = {.name = "APB", .desc = "[I/O] AP Bus"}, {.irq = IRQIO_APB, .name = "APB", .desc = "[I/O] AP Bus"},
[IRQIO_ADM] = {.name = "ADM", .desc = "[I/O] EADM Subchannel"}, {.irq = IRQIO_ADM, .name = "ADM", .desc = "[I/O] EADM Subchannel"},
[IRQIO_CSC] = {.name = "CSC", .desc = "[I/O] CHSC Subchannel"}, {.irq = IRQIO_CSC, .name = "CSC", .desc = "[I/O] CHSC Subchannel"},
[IRQIO_PCI] = {.name = "PCI", .desc = "[I/O] PCI Interrupt" }, {.irq = IRQIO_PCI, .name = "PCI", .desc = "[I/O] PCI Interrupt" },
[IRQIO_MSI] = {.name = "MSI", .desc = "[I/O] MSI Interrupt" }, {.irq = IRQIO_MSI, .name = "MSI", .desc = "[I/O] MSI Interrupt" },
[IRQIO_VIR] = {.name = "VIR", .desc = "[I/O] Virtual I/O Devices"}, {.irq = IRQIO_VIR, .name = "VIR", .desc = "[I/O] Virtual I/O Devices"},
[IRQIO_VAI] = {.name = "VAI", .desc = "[I/O] Virtual I/O Devices AI"}, {.irq = IRQIO_VAI, .name = "VAI", .desc = "[I/O] Virtual I/O Devices AI"},
[NMI_NMI] = {.name = "NMI", .desc = "[NMI] Machine Check"}, {.irq = NMI_NMI, .name = "NMI", .desc = "[NMI] Machine Check"},
[CPU_RST] = {.name = "RST", .desc = "[CPU] CPU Restart"}, {.irq = CPU_RST, .name = "RST", .desc = "[CPU] CPU Restart"},
}; };
void __init init_IRQ(void) void __init init_IRQ(void)
...@@ -116,33 +117,37 @@ void do_IRQ(struct pt_regs *regs, int irq) ...@@ -116,33 +117,37 @@ void do_IRQ(struct pt_regs *regs, int irq)
*/ */
int show_interrupts(struct seq_file *p, void *v) int show_interrupts(struct seq_file *p, void *v)
{ {
int irq = *(loff_t *) v; int index = *(loff_t *) v;
int cpu; int cpu, irq;
get_online_cpus(); get_online_cpus();
if (irq == 0) { if (index == 0) {
seq_puts(p, " "); seq_puts(p, " ");
for_each_online_cpu(cpu) for_each_online_cpu(cpu)
seq_printf(p, "CPU%d ", cpu); seq_printf(p, "CPU%d ", cpu);
seq_putc(p, '\n'); seq_putc(p, '\n');
goto out; goto out;
} }
if (irq < NR_IRQS) { if (index < NR_IRQS) {
if (irq >= NR_IRQS_BASE) if (index >= NR_IRQS_BASE)
goto out; goto out;
seq_printf(p, "%s: ", irqclass_main_desc[irq].name); /* Adjust index to process irqclass_main_desc array entries */
index--;
seq_printf(p, "%s: ", irqclass_main_desc[index].name);
irq = irqclass_main_desc[index].irq;
for_each_online_cpu(cpu) for_each_online_cpu(cpu)
seq_printf(p, "%10u ", kstat_irqs_cpu(irq, cpu)); seq_printf(p, "%10u ", kstat_irqs_cpu(irq, cpu));
seq_putc(p, '\n'); seq_putc(p, '\n');
goto out; goto out;
} }
for (irq = 0; irq < NR_ARCH_IRQS; irq++) { for (index = 0; index < NR_ARCH_IRQS; index++) {
seq_printf(p, "%s: ", irqclass_sub_desc[irq].name); seq_printf(p, "%s: ", irqclass_sub_desc[index].name);
irq = irqclass_sub_desc[index].irq;
for_each_online_cpu(cpu) for_each_online_cpu(cpu)
seq_printf(p, "%10u ", seq_printf(p, "%10u ",
per_cpu(irq_stat, cpu).irqs[irq]); per_cpu(irq_stat, cpu).irqs[irq]);
if (irqclass_sub_desc[irq].desc) if (irqclass_sub_desc[index].desc)
seq_printf(p, " %s", irqclass_sub_desc[irq].desc); seq_printf(p, " %s", irqclass_sub_desc[index].desc);
seq_putc(p, '\n'); seq_putc(p, '\n');
} }
out: out:
......
...@@ -501,6 +501,8 @@ static int kdump_mem_notifier(struct notifier_block *nb, ...@@ -501,6 +501,8 @@ static int kdump_mem_notifier(struct notifier_block *nb,
{ {
struct memory_notify *arg = data; struct memory_notify *arg = data;
if (action != MEM_GOING_OFFLINE)
return NOTIFY_OK;
if (arg->start_pfn < PFN_DOWN(resource_size(&crashk_res))) if (arg->start_pfn < PFN_DOWN(resource_size(&crashk_res)))
return NOTIFY_BAD; return NOTIFY_BAD;
if (arg->start_pfn > PFN_DOWN(crashk_res.end)) if (arg->start_pfn > PFN_DOWN(crashk_res.end))
......
...@@ -10,42 +10,33 @@ ...@@ -10,42 +10,33 @@
static inline pmd_t __pte_to_pmd(pte_t pte) static inline pmd_t __pte_to_pmd(pte_t pte)
{ {
int none, young, prot;
pmd_t pmd; pmd_t pmd;
/* /*
* Convert encoding pte bits pmd bits * Convert encoding pte bits pmd bits
* .IR...wrdytp ..R...I...y. * .IR...wrdytp dy..R...I...wr
* empty .10...000000 -> ..0...1...0. * empty .10...000000 -> 00..0...1...00
* prot-none, clean, old .11...000001 -> ..0...1...1. * prot-none, clean, old .11...000001 -> 00..1...1...00
* prot-none, clean, young .11...000101 -> ..1...1...1. * prot-none, clean, young .11...000101 -> 01..1...1...00
* prot-none, dirty, old .10...001001 -> ..0...1...1. * prot-none, dirty, old .10...001001 -> 10..1...1...00
* prot-none, dirty, young .10...001101 -> ..1...1...1. * prot-none, dirty, young .10...001101 -> 11..1...1...00
* read-only, clean, old .11...010001 -> ..1...1...0. * read-only, clean, old .11...010001 -> 00..1...1...01
* read-only, clean, young .01...010101 -> ..1...0...1. * read-only, clean, young .01...010101 -> 01..1...0...01
* read-only, dirty, old .11...011001 -> ..1...1...0. * read-only, dirty, old .11...011001 -> 10..1...1...01
* read-only, dirty, young .01...011101 -> ..1...0...1. * read-only, dirty, young .01...011101 -> 11..1...0...01
* read-write, clean, old .11...110001 -> ..0...1...0. * read-write, clean, old .11...110001 -> 00..0...1...11
* read-write, clean, young .01...110101 -> ..0...0...1. * read-write, clean, young .01...110101 -> 01..0...0...11
* read-write, dirty, old .10...111001 -> ..0...1...0. * read-write, dirty, old .10...111001 -> 10..0...1...11
* read-write, dirty, young .00...111101 -> ..0...0...1. * read-write, dirty, young .00...111101 -> 11..0...0...11
* Huge ptes are dirty by definition, a clean pte is made dirty
* by the conversion.
*/ */
if (pte_present(pte)) { if (pte_present(pte)) {
pmd_val(pmd) = pte_val(pte) & PAGE_MASK; pmd_val(pmd) = pte_val(pte) & PAGE_MASK;
if (pte_val(pte) & _PAGE_INVALID) pmd_val(pmd) |= (pte_val(pte) & _PAGE_READ) >> 4;
pmd_val(pmd) |= _SEGMENT_ENTRY_INVALID; pmd_val(pmd) |= (pte_val(pte) & _PAGE_WRITE) >> 4;
none = (pte_val(pte) & _PAGE_PRESENT) && pmd_val(pmd) |= (pte_val(pte) & _PAGE_INVALID) >> 5;
!(pte_val(pte) & _PAGE_READ) && pmd_val(pmd) |= (pte_val(pte) & _PAGE_PROTECT);
!(pte_val(pte) & _PAGE_WRITE); pmd_val(pmd) |= (pte_val(pte) & _PAGE_DIRTY) << 10;
prot = (pte_val(pte) & _PAGE_PROTECT) && pmd_val(pmd) |= (pte_val(pte) & _PAGE_YOUNG) << 10;
!(pte_val(pte) & _PAGE_WRITE);
young = pte_val(pte) & _PAGE_YOUNG;
if (none || young)
pmd_val(pmd) |= _SEGMENT_ENTRY_YOUNG;
if (prot || (none && young))
pmd_val(pmd) |= _SEGMENT_ENTRY_PROTECT;
} else } else
pmd_val(pmd) = _SEGMENT_ENTRY_INVALID; pmd_val(pmd) = _SEGMENT_ENTRY_INVALID;
return pmd; return pmd;
...@@ -57,33 +48,30 @@ static inline pte_t __pmd_to_pte(pmd_t pmd) ...@@ -57,33 +48,30 @@ static inline pte_t __pmd_to_pte(pmd_t pmd)
/* /*
* Convert encoding pmd bits pte bits * Convert encoding pmd bits pte bits
* ..R...I...y. .IR...wrdytp * dy..R...I...wr .IR...wrdytp
* empty ..0...1...0. -> .10...000000 * empty 00..0...1...00 -> .10...001100
* prot-none, old ..0...1...1. -> .10...001001 * prot-none, clean, old 00..0...1...00 -> .10...000001
* prot-none, young ..1...1...1. -> .10...001101 * prot-none, clean, young 01..0...1...00 -> .10...000101
* read-only, old ..1...1...0. -> .11...011001 * prot-none, dirty, old 10..0...1...00 -> .10...001001
* read-only, young ..1...0...1. -> .01...011101 * prot-none, dirty, young 11..0...1...00 -> .10...001101
* read-write, old ..0...1...0. -> .10...111001 * read-only, clean, old 00..1...1...01 -> .11...010001
* read-write, young ..0...0...1. -> .00...111101 * read-only, clean, young 01..1...1...01 -> .11...010101
* Huge ptes are dirty by definition * read-only, dirty, old 10..1...1...01 -> .11...011001
* read-only, dirty, young 11..1...1...01 -> .11...011101
* read-write, clean, old 00..0...1...11 -> .10...110001
* read-write, clean, young 01..0...1...11 -> .10...110101
* read-write, dirty, old 10..0...1...11 -> .10...111001
* read-write, dirty, young 11..0...1...11 -> .10...111101
*/ */
if (pmd_present(pmd)) { if (pmd_present(pmd)) {
pte_val(pte) = _PAGE_PRESENT | _PAGE_LARGE | _PAGE_DIRTY | pte_val(pte) = pmd_val(pmd) & _SEGMENT_ENTRY_ORIGIN_LARGE;
(pmd_val(pmd) & PAGE_MASK); pte_val(pte) |= _PAGE_LARGE | _PAGE_PRESENT;
if (pmd_val(pmd) & _SEGMENT_ENTRY_INVALID) pte_val(pte) |= (pmd_val(pmd) & _SEGMENT_ENTRY_READ) << 4;
pte_val(pte) |= _PAGE_INVALID; pte_val(pte) |= (pmd_val(pmd) & _SEGMENT_ENTRY_WRITE) << 4;
if (pmd_prot_none(pmd)) { pte_val(pte) |= (pmd_val(pmd) & _SEGMENT_ENTRY_INVALID) << 5;
if (pmd_val(pmd) & _SEGMENT_ENTRY_PROTECT) pte_val(pte) |= (pmd_val(pmd) & _SEGMENT_ENTRY_PROTECT);
pte_val(pte) |= _PAGE_YOUNG; pmd_val(pmd) |= (pte_val(pte) & _PAGE_DIRTY) << 10;
} else { pmd_val(pmd) |= (pte_val(pte) & _PAGE_YOUNG) << 10;
pte_val(pte) |= _PAGE_READ;
if (pmd_val(pmd) & _SEGMENT_ENTRY_PROTECT)
pte_val(pte) |= _PAGE_PROTECT;
else
pte_val(pte) |= _PAGE_WRITE;
if (pmd_val(pmd) & _SEGMENT_ENTRY_YOUNG)
pte_val(pte) |= _PAGE_YOUNG;
}
} else } else
pte_val(pte) = _PAGE_INVALID; pte_val(pte) = _PAGE_INVALID;
return pte; return pte;
...@@ -96,6 +84,7 @@ void set_huge_pte_at(struct mm_struct *mm, unsigned long addr, ...@@ -96,6 +84,7 @@ void set_huge_pte_at(struct mm_struct *mm, unsigned long addr,
pmd = __pte_to_pmd(pte); pmd = __pte_to_pmd(pte);
if (!MACHINE_HAS_HPAGE) { if (!MACHINE_HAS_HPAGE) {
/* Emulated huge ptes loose the dirty and young bit */
pmd_val(pmd) &= ~_SEGMENT_ENTRY_ORIGIN; pmd_val(pmd) &= ~_SEGMENT_ENTRY_ORIGIN;
pmd_val(pmd) |= pte_page(pte)[1].index; pmd_val(pmd) |= pte_page(pte)[1].index;
} else } else
...@@ -113,6 +102,8 @@ pte_t huge_ptep_get(pte_t *ptep) ...@@ -113,6 +102,8 @@ pte_t huge_ptep_get(pte_t *ptep)
origin = pmd_val(pmd) & _SEGMENT_ENTRY_ORIGIN; origin = pmd_val(pmd) & _SEGMENT_ENTRY_ORIGIN;
pmd_val(pmd) &= ~_SEGMENT_ENTRY_ORIGIN; pmd_val(pmd) &= ~_SEGMENT_ENTRY_ORIGIN;
pmd_val(pmd) |= *(unsigned long *) origin; pmd_val(pmd) |= *(unsigned long *) origin;
/* Emulated huge ptes are young and dirty by definition */
pmd_val(pmd) |= _SEGMENT_ENTRY_YOUNG | _SEGMENT_ENTRY_DIRTY;
} }
return __pmd_to_pte(pmd); return __pmd_to_pte(pmd);
} }
......
...@@ -1279,6 +1279,7 @@ static unsigned long page_table_realloc_pmd(struct mmu_gather *tlb, ...@@ -1279,6 +1279,7 @@ static unsigned long page_table_realloc_pmd(struct mmu_gather *tlb,
{ {
unsigned long next, *table, *new; unsigned long next, *table, *new;
struct page *page; struct page *page;
spinlock_t *ptl;
pmd_t *pmd; pmd_t *pmd;
pmd = pmd_offset(pud, addr); pmd = pmd_offset(pud, addr);
...@@ -1296,7 +1297,7 @@ static unsigned long page_table_realloc_pmd(struct mmu_gather *tlb, ...@@ -1296,7 +1297,7 @@ static unsigned long page_table_realloc_pmd(struct mmu_gather *tlb,
if (!new) if (!new)
return -ENOMEM; return -ENOMEM;
spin_lock(&mm->page_table_lock); ptl = pmd_lock(mm, pmd);
if (likely((unsigned long *) pmd_deref(*pmd) == table)) { if (likely((unsigned long *) pmd_deref(*pmd) == table)) {
/* Nuke pmd entry pointing to the "short" page table */ /* Nuke pmd entry pointing to the "short" page table */
pmdp_flush_lazy(mm, addr, pmd); pmdp_flush_lazy(mm, addr, pmd);
...@@ -1310,7 +1311,7 @@ static unsigned long page_table_realloc_pmd(struct mmu_gather *tlb, ...@@ -1310,7 +1311,7 @@ static unsigned long page_table_realloc_pmd(struct mmu_gather *tlb,
page_table_free_rcu(tlb, table); page_table_free_rcu(tlb, table);
new = NULL; new = NULL;
} }
spin_unlock(&mm->page_table_lock); spin_unlock(ptl);
if (new) { if (new) {
page_table_free_pgste(new); page_table_free_pgste(new);
goto again; goto again;
...@@ -1432,6 +1433,9 @@ int pmdp_set_access_flags(struct vm_area_struct *vma, ...@@ -1432,6 +1433,9 @@ int pmdp_set_access_flags(struct vm_area_struct *vma,
{ {
VM_BUG_ON(address & ~HPAGE_PMD_MASK); VM_BUG_ON(address & ~HPAGE_PMD_MASK);
entry = pmd_mkyoung(entry);
if (dirty)
entry = pmd_mkdirty(entry);
if (pmd_same(*pmdp, entry)) if (pmd_same(*pmdp, entry))
return 0; return 0;
pmdp_invalidate(vma, address, pmdp); pmdp_invalidate(vma, address, pmdp);
......
...@@ -15,8 +15,8 @@ ...@@ -15,8 +15,8 @@
* Thomas Klein * Thomas Klein
*/ */
#define COMPONENT "zPCI" #define KMSG_COMPONENT "zpci"
#define pr_fmt(fmt) COMPONENT ": " fmt #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/slab.h> #include <linux/slab.h>
......
...@@ -5,8 +5,8 @@ ...@@ -5,8 +5,8 @@
* Jan Glauber <jang@linux.vnet.ibm.com> * Jan Glauber <jang@linux.vnet.ibm.com>
*/ */
#define COMPONENT "zPCI" #define KMSG_COMPONENT "zpci"
#define pr_fmt(fmt) COMPONENT ": " fmt #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/slab.h> #include <linux/slab.h>
......
...@@ -5,8 +5,8 @@ ...@@ -5,8 +5,8 @@
* Jan Glauber <jang@linux.vnet.ibm.com> * Jan Glauber <jang@linux.vnet.ibm.com>
*/ */
#define COMPONENT "zPCI" #define KMSG_COMPONENT "zpci"
#define pr_fmt(fmt) COMPONENT ": " fmt #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/seq_file.h> #include <linux/seq_file.h>
......
...@@ -16,6 +16,13 @@ ...@@ -16,6 +16,13 @@
static struct kmem_cache *dma_region_table_cache; static struct kmem_cache *dma_region_table_cache;
static struct kmem_cache *dma_page_table_cache; static struct kmem_cache *dma_page_table_cache;
static int s390_iommu_strict;
static int zpci_refresh_global(struct zpci_dev *zdev)
{
return zpci_refresh_trans((u64) zdev->fh << 32, zdev->start_dma,
zdev->iommu_pages * PAGE_SIZE);
}
static unsigned long *dma_alloc_cpu_table(void) static unsigned long *dma_alloc_cpu_table(void)
{ {
...@@ -155,18 +162,15 @@ static int dma_update_trans(struct zpci_dev *zdev, unsigned long pa, ...@@ -155,18 +162,15 @@ static int dma_update_trans(struct zpci_dev *zdev, unsigned long pa,
} }
/* /*
* rpcit is not required to establish new translations when previously * With zdev->tlb_refresh == 0, rpcit is not required to establish new
* invalid translation-table entries are validated, however it is * translations when previously invalid translation-table entries are
* required when altering previously valid entries. * validated. With lazy unmap, it also is skipped for previously valid
* entries, but a global rpcit is then required before any address can
* be re-used, i.e. after each iommu bitmap wrap-around.
*/ */
if (!zdev->tlb_refresh && if (!zdev->tlb_refresh &&
((flags & ZPCI_PTE_VALID_MASK) == ZPCI_PTE_VALID)) (!s390_iommu_strict ||
/* ((flags & ZPCI_PTE_VALID_MASK) == ZPCI_PTE_VALID)))
* TODO: also need to check that the old entry is indeed INVALID
* and not only for one page but for the whole range...
* -> now we WARN_ON in that case but with lazy unmap that
* needs to be redone!
*/
goto no_refresh; goto no_refresh;
rc = zpci_refresh_trans((u64) zdev->fh << 32, start_dma_addr, rc = zpci_refresh_trans((u64) zdev->fh << 32, start_dma_addr,
...@@ -220,16 +224,21 @@ static unsigned long __dma_alloc_iommu(struct zpci_dev *zdev, ...@@ -220,16 +224,21 @@ static unsigned long __dma_alloc_iommu(struct zpci_dev *zdev,
static unsigned long dma_alloc_iommu(struct zpci_dev *zdev, int size) static unsigned long dma_alloc_iommu(struct zpci_dev *zdev, int size)
{ {
unsigned long offset, flags; unsigned long offset, flags;
int wrap = 0;
spin_lock_irqsave(&zdev->iommu_bitmap_lock, flags); spin_lock_irqsave(&zdev->iommu_bitmap_lock, flags);
offset = __dma_alloc_iommu(zdev, zdev->next_bit, size); offset = __dma_alloc_iommu(zdev, zdev->next_bit, size);
if (offset == -1) if (offset == -1) {
/* wrap-around */
offset = __dma_alloc_iommu(zdev, 0, size); offset = __dma_alloc_iommu(zdev, 0, size);
wrap = 1;
}
if (offset != -1) { if (offset != -1) {
zdev->next_bit = offset + size; zdev->next_bit = offset + size;
if (zdev->next_bit >= zdev->iommu_pages) if (!zdev->tlb_refresh && !s390_iommu_strict && wrap)
zdev->next_bit = 0; /* global flush after wrap-around with lazy unmap */
zpci_refresh_global(zdev);
} }
spin_unlock_irqrestore(&zdev->iommu_bitmap_lock, flags); spin_unlock_irqrestore(&zdev->iommu_bitmap_lock, flags);
return offset; return offset;
...@@ -243,7 +252,11 @@ static void dma_free_iommu(struct zpci_dev *zdev, unsigned long offset, int size ...@@ -243,7 +252,11 @@ static void dma_free_iommu(struct zpci_dev *zdev, unsigned long offset, int size
if (!zdev->iommu_bitmap) if (!zdev->iommu_bitmap)
goto out; goto out;
bitmap_clear(zdev->iommu_bitmap, offset, size); bitmap_clear(zdev->iommu_bitmap, offset, size);
if (offset >= zdev->next_bit) /*
* Lazy flush for unmap: need to move next_bit to avoid address re-use
* until wrap-around.
*/
if (!s390_iommu_strict && offset >= zdev->next_bit)
zdev->next_bit = offset + size; zdev->next_bit = offset + size;
out: out:
spin_unlock_irqrestore(&zdev->iommu_bitmap_lock, flags); spin_unlock_irqrestore(&zdev->iommu_bitmap_lock, flags);
...@@ -504,3 +517,12 @@ struct dma_map_ops s390_dma_ops = { ...@@ -504,3 +517,12 @@ struct dma_map_ops s390_dma_ops = {
/* dma_supported is unconditionally true without a callback */ /* dma_supported is unconditionally true without a callback */
}; };
EXPORT_SYMBOL_GPL(s390_dma_ops); EXPORT_SYMBOL_GPL(s390_dma_ops);
static int __init s390_iommu_setup(char *str)
{
if (!strncmp(str, "strict", 6))
s390_iommu_strict = 1;
return 0;
}
__setup("s390_iommu=", s390_iommu_setup);
...@@ -5,8 +5,8 @@ ...@@ -5,8 +5,8 @@
* Jan Glauber <jang@linux.vnet.ibm.com> * Jan Glauber <jang@linux.vnet.ibm.com>
*/ */
#define COMPONENT "zPCI" #define KMSG_COMPONENT "zpci"
#define pr_fmt(fmt) COMPONENT ": " fmt #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/pci.h> #include <linux/pci.h>
......
...@@ -5,8 +5,8 @@ ...@@ -5,8 +5,8 @@
* Jan Glauber <jang@linux.vnet.ibm.com> * Jan Glauber <jang@linux.vnet.ibm.com>
*/ */
#define COMPONENT "zPCI" #define KMSG_COMPONENT "zpci"
#define pr_fmt(fmt) COMPONENT ": " fmt #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/stat.h> #include <linux/stat.h>
......
...@@ -7,8 +7,8 @@ ...@@ -7,8 +7,8 @@
* Jan Glauber <jang@linux.vnet.ibm.com> * Jan Glauber <jang@linux.vnet.ibm.com>
*/ */
#define COMPONENT "zPCI hpc" #define KMSG_COMPONENT "zpci"
#define pr_fmt(fmt) COMPONENT ": " fmt #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
#include <linux/module.h> #include <linux/module.h>
#include <linux/kernel.h> #include <linux/kernel.h>
......
This diff is collapsed.
...@@ -2039,7 +2039,7 @@ static int dasd_eckd_online_to_ready(struct dasd_device *device) ...@@ -2039,7 +2039,7 @@ static int dasd_eckd_online_to_ready(struct dasd_device *device)
return 0; return 0;
}; };
static int dasd_eckd_ready_to_basic(struct dasd_device *device) static int dasd_eckd_basic_to_known(struct dasd_device *device)
{ {
return dasd_alias_remove_device(device); return dasd_alias_remove_device(device);
}; };
...@@ -2061,11 +2061,12 @@ dasd_eckd_fill_geometry(struct dasd_block *block, struct hd_geometry *geo) ...@@ -2061,11 +2061,12 @@ dasd_eckd_fill_geometry(struct dasd_block *block, struct hd_geometry *geo)
static struct dasd_ccw_req * static struct dasd_ccw_req *
dasd_eckd_build_format(struct dasd_device *base, dasd_eckd_build_format(struct dasd_device *base,
struct format_data_t *fdata) struct format_data_t *fdata,
int enable_pav)
{ {
struct dasd_eckd_private *base_priv; struct dasd_eckd_private *base_priv;
struct dasd_eckd_private *start_priv; struct dasd_eckd_private *start_priv;
struct dasd_device *startdev; struct dasd_device *startdev = NULL;
struct dasd_ccw_req *fcp; struct dasd_ccw_req *fcp;
struct eckd_count *ect; struct eckd_count *ect;
struct ch_t address; struct ch_t address;
...@@ -2079,7 +2080,9 @@ dasd_eckd_build_format(struct dasd_device *base, ...@@ -2079,7 +2080,9 @@ dasd_eckd_build_format(struct dasd_device *base,
int nr_tracks; int nr_tracks;
int use_prefix; int use_prefix;
if (enable_pav)
startdev = dasd_alias_get_start_dev(base); startdev = dasd_alias_get_start_dev(base);
if (!startdev) if (!startdev)
startdev = base; startdev = base;
...@@ -2309,6 +2312,7 @@ dasd_eckd_build_format(struct dasd_device *base, ...@@ -2309,6 +2312,7 @@ dasd_eckd_build_format(struct dasd_device *base,
fcp->startdev = startdev; fcp->startdev = startdev;
fcp->memdev = startdev; fcp->memdev = startdev;
fcp->basedev = base;
fcp->retries = 256; fcp->retries = 256;
fcp->expires = startdev->default_expires * HZ; fcp->expires = startdev->default_expires * HZ;
fcp->buildclk = get_tod_clock(); fcp->buildclk = get_tod_clock();
...@@ -2319,7 +2323,8 @@ dasd_eckd_build_format(struct dasd_device *base, ...@@ -2319,7 +2323,8 @@ dasd_eckd_build_format(struct dasd_device *base,
static int static int
dasd_eckd_format_device(struct dasd_device *base, dasd_eckd_format_device(struct dasd_device *base,
struct format_data_t *fdata) struct format_data_t *fdata,
int enable_pav)
{ {
struct dasd_ccw_req *cqr, *n; struct dasd_ccw_req *cqr, *n;
struct dasd_block *block; struct dasd_block *block;
...@@ -2327,7 +2332,7 @@ dasd_eckd_format_device(struct dasd_device *base, ...@@ -2327,7 +2332,7 @@ dasd_eckd_format_device(struct dasd_device *base,
struct list_head format_queue; struct list_head format_queue;
struct dasd_device *device; struct dasd_device *device;
int old_stop, format_step; int old_stop, format_step;
int step, rc = 0; int step, rc = 0, sleep_rc;
block = base->block; block = base->block;
private = (struct dasd_eckd_private *) base->private; private = (struct dasd_eckd_private *) base->private;
...@@ -2361,11 +2366,11 @@ dasd_eckd_format_device(struct dasd_device *base, ...@@ -2361,11 +2366,11 @@ dasd_eckd_format_device(struct dasd_device *base,
} }
INIT_LIST_HEAD(&format_queue); INIT_LIST_HEAD(&format_queue);
old_stop = fdata->stop_unit;
old_stop = fdata->stop_unit;
while (fdata->start_unit <= 1) { while (fdata->start_unit <= 1) {
fdata->stop_unit = fdata->start_unit; fdata->stop_unit = fdata->start_unit;
cqr = dasd_eckd_build_format(base, fdata); cqr = dasd_eckd_build_format(base, fdata, enable_pav);
list_add(&cqr->blocklist, &format_queue); list_add(&cqr->blocklist, &format_queue);
fdata->stop_unit = old_stop; fdata->stop_unit = old_stop;
...@@ -2383,7 +2388,7 @@ dasd_eckd_format_device(struct dasd_device *base, ...@@ -2383,7 +2388,7 @@ dasd_eckd_format_device(struct dasd_device *base,
if (step > format_step) if (step > format_step)
fdata->stop_unit = fdata->start_unit + format_step - 1; fdata->stop_unit = fdata->start_unit + format_step - 1;
cqr = dasd_eckd_build_format(base, fdata); cqr = dasd_eckd_build_format(base, fdata, enable_pav);
if (IS_ERR(cqr)) { if (IS_ERR(cqr)) {
if (PTR_ERR(cqr) == -ENOMEM) { if (PTR_ERR(cqr) == -ENOMEM) {
/* /*
...@@ -2403,7 +2408,7 @@ dasd_eckd_format_device(struct dasd_device *base, ...@@ -2403,7 +2408,7 @@ dasd_eckd_format_device(struct dasd_device *base,
} }
sleep: sleep:
dasd_sleep_on_queue(&format_queue); sleep_rc = dasd_sleep_on_queue(&format_queue);
list_for_each_entry_safe(cqr, n, &format_queue, blocklist) { list_for_each_entry_safe(cqr, n, &format_queue, blocklist) {
device = cqr->startdev; device = cqr->startdev;
...@@ -2415,6 +2420,9 @@ dasd_eckd_format_device(struct dasd_device *base, ...@@ -2415,6 +2420,9 @@ dasd_eckd_format_device(struct dasd_device *base,
private->count--; private->count--;
} }
if (sleep_rc)
return sleep_rc;
/* /*
* in case of ENOMEM we need to retry after * in case of ENOMEM we need to retry after
* first requests are finished * first requests are finished
...@@ -4511,7 +4519,7 @@ static struct dasd_discipline dasd_eckd_discipline = { ...@@ -4511,7 +4519,7 @@ static struct dasd_discipline dasd_eckd_discipline = {
.verify_path = dasd_eckd_verify_path, .verify_path = dasd_eckd_verify_path,
.basic_to_ready = dasd_eckd_basic_to_ready, .basic_to_ready = dasd_eckd_basic_to_ready,
.online_to_ready = dasd_eckd_online_to_ready, .online_to_ready = dasd_eckd_online_to_ready,
.ready_to_basic = dasd_eckd_ready_to_basic, .basic_to_known = dasd_eckd_basic_to_known,
.fill_geometry = dasd_eckd_fill_geometry, .fill_geometry = dasd_eckd_fill_geometry,
.start_IO = dasd_start_IO, .start_IO = dasd_start_IO,
.term_IO = dasd_term_IO, .term_IO = dasd_term_IO,
......
...@@ -175,6 +175,7 @@ struct dasd_ccw_req { ...@@ -175,6 +175,7 @@ struct dasd_ccw_req {
struct dasd_block *block; /* the originating block device */ struct dasd_block *block; /* the originating block device */
struct dasd_device *memdev; /* the device used to allocate this */ struct dasd_device *memdev; /* the device used to allocate this */
struct dasd_device *startdev; /* device the request is started on */ struct dasd_device *startdev; /* device the request is started on */
struct dasd_device *basedev; /* base device if no block->base */
void *cpaddr; /* address of ccw or tcw */ void *cpaddr; /* address of ccw or tcw */
unsigned char cpmode; /* 0 = cmd mode, 1 = itcw */ unsigned char cpmode; /* 0 = cmd mode, 1 = itcw */
char status; /* status of this request */ char status; /* status of this request */
...@@ -304,7 +305,7 @@ struct dasd_discipline { ...@@ -304,7 +305,7 @@ struct dasd_discipline {
*/ */
int (*basic_to_ready) (struct dasd_device *); int (*basic_to_ready) (struct dasd_device *);
int (*online_to_ready) (struct dasd_device *); int (*online_to_ready) (struct dasd_device *);
int (*ready_to_basic) (struct dasd_device *); int (*basic_to_known)(struct dasd_device *);
/* (struct dasd_device *); /* (struct dasd_device *);
* Device operation functions. build_cp creates a ccw chain for * Device operation functions. build_cp creates a ccw chain for
...@@ -321,7 +322,7 @@ struct dasd_discipline { ...@@ -321,7 +322,7 @@ struct dasd_discipline {
int (*term_IO) (struct dasd_ccw_req *); int (*term_IO) (struct dasd_ccw_req *);
void (*handle_terminated_request) (struct dasd_ccw_req *); void (*handle_terminated_request) (struct dasd_ccw_req *);
int (*format_device) (struct dasd_device *, int (*format_device) (struct dasd_device *,
struct format_data_t *); struct format_data_t *, int enable_pav);
int (*free_cp) (struct dasd_ccw_req *, struct request *); int (*free_cp) (struct dasd_ccw_req *, struct request *);
/* /*
......
...@@ -203,7 +203,9 @@ static int ...@@ -203,7 +203,9 @@ static int
dasd_format(struct dasd_block *block, struct format_data_t *fdata) dasd_format(struct dasd_block *block, struct format_data_t *fdata)
{ {
struct dasd_device *base; struct dasd_device *base;
int rc; int enable_pav = 1;
int rc, retries;
int start, stop;
base = block->base; base = block->base;
if (base->discipline->format_device == NULL) if (base->discipline->format_device == NULL)
...@@ -231,10 +233,29 @@ dasd_format(struct dasd_block *block, struct format_data_t *fdata) ...@@ -231,10 +233,29 @@ dasd_format(struct dasd_block *block, struct format_data_t *fdata)
bdput(bdev); bdput(bdev);
} }
rc = base->discipline->format_device(base, fdata); retries = 255;
if (rc) /* backup start- and endtrack for retries */
start = fdata->start_unit;
stop = fdata->stop_unit;
do {
rc = base->discipline->format_device(base, fdata, enable_pav);
if (rc) {
if (rc == -EAGAIN) {
retries--;
/* disable PAV in case of errors */
enable_pav = 0;
fdata->start_unit = start;
fdata->stop_unit = stop;
} else
return rc; return rc;
} else
/* success */
break;
} while (retries);
if (!retries)
return -EIO;
else
return 0; return 0;
} }
......
...@@ -288,12 +288,16 @@ static void raw3215_timeout(unsigned long __data) ...@@ -288,12 +288,16 @@ static void raw3215_timeout(unsigned long __data)
unsigned long flags; unsigned long flags;
spin_lock_irqsave(get_ccwdev_lock(raw->cdev), flags); spin_lock_irqsave(get_ccwdev_lock(raw->cdev), flags);
if (raw->flags & RAW3215_TIMER_RUNS) {
del_timer(&raw->timer);
raw->flags &= ~RAW3215_TIMER_RUNS; raw->flags &= ~RAW3215_TIMER_RUNS;
if (!(raw->port.flags & ASYNC_SUSPENDED)) { if (!(raw->port.flags & ASYNC_SUSPENDED)) {
raw3215_mk_write_req(raw); raw3215_mk_write_req(raw);
raw3215_start_io(raw); raw3215_start_io(raw);
if ((raw->queued_read || raw->queued_write) &&
!(raw->flags & RAW3215_WORKING) &&
!(raw->flags & RAW3215_TIMER_RUNS)) {
raw->timer.expires = RAW3215_TIMEOUT + jiffies;
add_timer(&raw->timer);
raw->flags |= RAW3215_TIMER_RUNS;
} }
} }
spin_unlock_irqrestore(get_ccwdev_lock(raw->cdev), flags); spin_unlock_irqrestore(get_ccwdev_lock(raw->cdev), flags);
...@@ -317,17 +321,15 @@ static inline void raw3215_try_io(struct raw3215_info *raw) ...@@ -317,17 +321,15 @@ static inline void raw3215_try_io(struct raw3215_info *raw)
(raw->flags & RAW3215_FLUSHING)) { (raw->flags & RAW3215_FLUSHING)) {
/* execute write requests bigger than minimum size */ /* execute write requests bigger than minimum size */
raw3215_start_io(raw); raw3215_start_io(raw);
if (raw->flags & RAW3215_TIMER_RUNS) {
del_timer(&raw->timer);
raw->flags &= ~RAW3215_TIMER_RUNS;
} }
} else if (!(raw->flags & RAW3215_TIMER_RUNS)) { }
/* delay small writes */ if ((raw->queued_read || raw->queued_write) &&
!(raw->flags & RAW3215_WORKING) &&
!(raw->flags & RAW3215_TIMER_RUNS)) {
raw->timer.expires = RAW3215_TIMEOUT + jiffies; raw->timer.expires = RAW3215_TIMEOUT + jiffies;
add_timer(&raw->timer); add_timer(&raw->timer);
raw->flags |= RAW3215_TIMER_RUNS; raw->flags |= RAW3215_TIMER_RUNS;
} }
}
} }
/* /*
......
...@@ -17,6 +17,8 @@ ...@@ -17,6 +17,8 @@
#include "qdio.h" #include "qdio.h"
#include "qdio_debug.h" #include "qdio_debug.h"
#define QBUFF_PER_PAGE (PAGE_SIZE / sizeof(struct qdio_buffer))
static struct kmem_cache *qdio_q_cache; static struct kmem_cache *qdio_q_cache;
static struct kmem_cache *qdio_aob_cache; static struct kmem_cache *qdio_aob_cache;
...@@ -32,6 +34,57 @@ void qdio_release_aob(struct qaob *aob) ...@@ -32,6 +34,57 @@ void qdio_release_aob(struct qaob *aob)
} }
EXPORT_SYMBOL_GPL(qdio_release_aob); EXPORT_SYMBOL_GPL(qdio_release_aob);
/**
* qdio_free_buffers() - free qdio buffers
* @buf: array of pointers to qdio buffers
* @count: number of qdio buffers to free
*/
void qdio_free_buffers(struct qdio_buffer **buf, unsigned int count)
{
int pos;
for (pos = 0; pos < count; pos += QBUFF_PER_PAGE)
free_page((unsigned long) buf[pos]);
}
EXPORT_SYMBOL_GPL(qdio_free_buffers);
/**
* qdio_alloc_buffers() - allocate qdio buffers
* @buf: array of pointers to qdio buffers
* @count: number of qdio buffers to allocate
*/
int qdio_alloc_buffers(struct qdio_buffer **buf, unsigned int count)
{
int pos;
for (pos = 0; pos < count; pos += QBUFF_PER_PAGE) {
buf[pos] = (void *) get_zeroed_page(GFP_KERNEL);
if (!buf[pos]) {
qdio_free_buffers(buf, count);
return -ENOMEM;
}
}
for (pos = 0; pos < count; pos++)
if (pos % QBUFF_PER_PAGE)
buf[pos] = buf[pos - 1] + 1;
return 0;
}
EXPORT_SYMBOL_GPL(qdio_alloc_buffers);
/**
* qdio_reset_buffers() - reset qdio buffers
* @buf: array of pointers to qdio buffers
* @count: number of qdio buffers that will be zeroed
*/
void qdio_reset_buffers(struct qdio_buffer **buf, unsigned int count)
{
int pos;
for (pos = 0; pos < count; pos++)
memset(buf[pos], 0, sizeof(struct qdio_buffer));
}
EXPORT_SYMBOL_GPL(qdio_reset_buffers);
/* /*
* qebsm is only available under 64bit but the adapter sets the feature * qebsm is only available under 64bit but the adapter sets the feature
* flag anyway, so we manually override it. * flag anyway, so we manually override it.
......
...@@ -439,10 +439,10 @@ struct qeth_qdio_buffer { ...@@ -439,10 +439,10 @@ struct qeth_qdio_buffer {
}; };
struct qeth_qdio_q { struct qeth_qdio_q {
struct qdio_buffer qdio_bufs[QDIO_MAX_BUFFERS_PER_Q]; struct qdio_buffer *qdio_bufs[QDIO_MAX_BUFFERS_PER_Q];
struct qeth_qdio_buffer bufs[QDIO_MAX_BUFFERS_PER_Q]; struct qeth_qdio_buffer bufs[QDIO_MAX_BUFFERS_PER_Q];
int next_buf_to_init; int next_buf_to_init;
} __attribute__ ((aligned(256))); };
struct qeth_qdio_out_buffer { struct qeth_qdio_out_buffer {
struct qdio_buffer *buffer; struct qdio_buffer *buffer;
...@@ -465,7 +465,7 @@ enum qeth_out_q_states { ...@@ -465,7 +465,7 @@ enum qeth_out_q_states {
}; };
struct qeth_qdio_out_q { struct qeth_qdio_out_q {
struct qdio_buffer qdio_bufs[QDIO_MAX_BUFFERS_PER_Q]; struct qdio_buffer *qdio_bufs[QDIO_MAX_BUFFERS_PER_Q];
struct qeth_qdio_out_buffer *bufs[QDIO_MAX_BUFFERS_PER_Q]; struct qeth_qdio_out_buffer *bufs[QDIO_MAX_BUFFERS_PER_Q];
struct qdio_outbuf_state *bufstates; /* convenience pointer */ struct qdio_outbuf_state *bufstates; /* convenience pointer */
int queue_no; int queue_no;
...@@ -483,7 +483,7 @@ struct qeth_qdio_out_q { ...@@ -483,7 +483,7 @@ struct qeth_qdio_out_q {
atomic_t used_buffers; atomic_t used_buffers;
/* indicates whether PCI flag must be set (or if one is outstanding) */ /* indicates whether PCI flag must be set (or if one is outstanding) */
atomic_t set_pci_flags_count; atomic_t set_pci_flags_count;
} __attribute__ ((aligned(256))); };
struct qeth_qdio_info { struct qeth_qdio_info {
atomic_t state; atomic_t state;
......
...@@ -292,14 +292,43 @@ int qeth_realloc_buffer_pool(struct qeth_card *card, int bufcnt) ...@@ -292,14 +292,43 @@ int qeth_realloc_buffer_pool(struct qeth_card *card, int bufcnt)
} }
EXPORT_SYMBOL_GPL(qeth_realloc_buffer_pool); EXPORT_SYMBOL_GPL(qeth_realloc_buffer_pool);
static void qeth_free_qdio_queue(struct qeth_qdio_q *q)
{
if (!q)
return;
qdio_free_buffers(q->qdio_bufs, QDIO_MAX_BUFFERS_PER_Q);
kfree(q);
}
static struct qeth_qdio_q *qeth_alloc_qdio_queue(void)
{
struct qeth_qdio_q *q = kzalloc(sizeof(*q), GFP_KERNEL);
int i;
if (!q)
return NULL;
if (qdio_alloc_buffers(q->qdio_bufs, QDIO_MAX_BUFFERS_PER_Q)) {
kfree(q);
return NULL;
}
for (i = 0; i < QDIO_MAX_BUFFERS_PER_Q; ++i)
q->bufs[i].buffer = q->qdio_bufs[i];
QETH_DBF_HEX(SETUP, 2, &q, sizeof(void *));
return q;
}
static inline int qeth_cq_init(struct qeth_card *card) static inline int qeth_cq_init(struct qeth_card *card)
{ {
int rc; int rc;
if (card->options.cq == QETH_CQ_ENABLED) { if (card->options.cq == QETH_CQ_ENABLED) {
QETH_DBF_TEXT(SETUP, 2, "cqinit"); QETH_DBF_TEXT(SETUP, 2, "cqinit");
memset(card->qdio.c_q->qdio_bufs, 0, qdio_reset_buffers(card->qdio.c_q->qdio_bufs,
QDIO_MAX_BUFFERS_PER_Q * sizeof(struct qdio_buffer)); QDIO_MAX_BUFFERS_PER_Q);
card->qdio.c_q->next_buf_to_init = 127; card->qdio.c_q->next_buf_to_init = 127;
rc = do_QDIO(CARD_DDEV(card), QDIO_FLAG_SYNC_INPUT, rc = do_QDIO(CARD_DDEV(card), QDIO_FLAG_SYNC_INPUT,
card->qdio.no_in_queues - 1, 0, card->qdio.no_in_queues - 1, 0,
...@@ -323,21 +352,12 @@ static inline int qeth_alloc_cq(struct qeth_card *card) ...@@ -323,21 +352,12 @@ static inline int qeth_alloc_cq(struct qeth_card *card)
struct qdio_outbuf_state *outbuf_states; struct qdio_outbuf_state *outbuf_states;
QETH_DBF_TEXT(SETUP, 2, "cqon"); QETH_DBF_TEXT(SETUP, 2, "cqon");
card->qdio.c_q = kzalloc(sizeof(struct qeth_qdio_q), card->qdio.c_q = qeth_alloc_qdio_queue();
GFP_KERNEL);
if (!card->qdio.c_q) { if (!card->qdio.c_q) {
rc = -1; rc = -1;
goto kmsg_out; goto kmsg_out;
} }
QETH_DBF_HEX(SETUP, 2, &card->qdio.c_q, sizeof(void *));
for (i = 0; i < QDIO_MAX_BUFFERS_PER_Q; ++i) {
card->qdio.c_q->bufs[i].buffer =
&card->qdio.c_q->qdio_bufs[i];
}
card->qdio.no_in_queues = 2; card->qdio.no_in_queues = 2;
card->qdio.out_bufstates = card->qdio.out_bufstates =
kzalloc(card->qdio.no_out_queues * kzalloc(card->qdio.no_out_queues *
QDIO_MAX_BUFFERS_PER_Q * QDIO_MAX_BUFFERS_PER_Q *
...@@ -361,7 +381,7 @@ static inline int qeth_alloc_cq(struct qeth_card *card) ...@@ -361,7 +381,7 @@ static inline int qeth_alloc_cq(struct qeth_card *card)
out: out:
return rc; return rc;
free_cq_out: free_cq_out:
kfree(card->qdio.c_q); qeth_free_qdio_queue(card->qdio.c_q);
card->qdio.c_q = NULL; card->qdio.c_q = NULL;
kmsg_out: kmsg_out:
dev_err(&card->gdev->dev, "Failed to create completion queue\n"); dev_err(&card->gdev->dev, "Failed to create completion queue\n");
...@@ -372,7 +392,7 @@ static inline void qeth_free_cq(struct qeth_card *card) ...@@ -372,7 +392,7 @@ static inline void qeth_free_cq(struct qeth_card *card)
{ {
if (card->qdio.c_q) { if (card->qdio.c_q) {
--card->qdio.no_in_queues; --card->qdio.no_in_queues;
kfree(card->qdio.c_q); qeth_free_qdio_queue(card->qdio.c_q);
card->qdio.c_q = NULL; card->qdio.c_q = NULL;
} }
kfree(card->qdio.out_bufstates); kfree(card->qdio.out_bufstates);
...@@ -1282,35 +1302,6 @@ static void qeth_free_buffer_pool(struct qeth_card *card) ...@@ -1282,35 +1302,6 @@ static void qeth_free_buffer_pool(struct qeth_card *card)
} }
} }
static void qeth_free_qdio_buffers(struct qeth_card *card)
{
int i, j;
if (atomic_xchg(&card->qdio.state, QETH_QDIO_UNINITIALIZED) ==
QETH_QDIO_UNINITIALIZED)
return;
qeth_free_cq(card);
cancel_delayed_work_sync(&card->buffer_reclaim_work);
for (j = 0; j < QDIO_MAX_BUFFERS_PER_Q; ++j) {
if (card->qdio.in_q->bufs[j].rx_skb)
dev_kfree_skb_any(card->qdio.in_q->bufs[j].rx_skb);
}
kfree(card->qdio.in_q);
card->qdio.in_q = NULL;
/* inbound buffer pool */
qeth_free_buffer_pool(card);
/* free outbound qdio_qs */
if (card->qdio.out_qs) {
for (i = 0; i < card->qdio.no_out_queues; ++i) {
qeth_clear_outq_buffers(card->qdio.out_qs[i], 1);
kfree(card->qdio.out_qs[i]);
}
kfree(card->qdio.out_qs);
card->qdio.out_qs = NULL;
}
}
static void qeth_clean_channel(struct qeth_channel *channel) static void qeth_clean_channel(struct qeth_channel *channel)
{ {
int cnt; int cnt;
...@@ -2392,7 +2383,7 @@ static int qeth_init_qdio_out_buf(struct qeth_qdio_out_q *q, int bidx) ...@@ -2392,7 +2383,7 @@ static int qeth_init_qdio_out_buf(struct qeth_qdio_out_q *q, int bidx)
rc = -ENOMEM; rc = -ENOMEM;
goto out; goto out;
} }
newbuf->buffer = &q->qdio_bufs[bidx]; newbuf->buffer = q->qdio_bufs[bidx];
skb_queue_head_init(&newbuf->skb_list); skb_queue_head_init(&newbuf->skb_list);
lockdep_set_class(&newbuf->skb_list.lock, &qdio_out_skb_queue_key); lockdep_set_class(&newbuf->skb_list.lock, &qdio_out_skb_queue_key);
newbuf->q = q; newbuf->q = q;
...@@ -2411,6 +2402,28 @@ static int qeth_init_qdio_out_buf(struct qeth_qdio_out_q *q, int bidx) ...@@ -2411,6 +2402,28 @@ static int qeth_init_qdio_out_buf(struct qeth_qdio_out_q *q, int bidx)
return rc; return rc;
} }
static void qeth_free_qdio_out_buf(struct qeth_qdio_out_q *q)
{
if (!q)
return;
qdio_free_buffers(q->qdio_bufs, QDIO_MAX_BUFFERS_PER_Q);
kfree(q);
}
static struct qeth_qdio_out_q *qeth_alloc_qdio_out_buf(void)
{
struct qeth_qdio_out_q *q = kzalloc(sizeof(*q), GFP_KERNEL);
if (!q)
return NULL;
if (qdio_alloc_buffers(q->qdio_bufs, QDIO_MAX_BUFFERS_PER_Q)) {
kfree(q);
return NULL;
}
return q;
}
static int qeth_alloc_qdio_buffers(struct qeth_card *card) static int qeth_alloc_qdio_buffers(struct qeth_card *card)
{ {
...@@ -2422,19 +2435,11 @@ static int qeth_alloc_qdio_buffers(struct qeth_card *card) ...@@ -2422,19 +2435,11 @@ static int qeth_alloc_qdio_buffers(struct qeth_card *card)
QETH_QDIO_ALLOCATED) != QETH_QDIO_UNINITIALIZED) QETH_QDIO_ALLOCATED) != QETH_QDIO_UNINITIALIZED)
return 0; return 0;
card->qdio.in_q = kzalloc(sizeof(struct qeth_qdio_q), QETH_DBF_TEXT(SETUP, 2, "inq");
GFP_KERNEL); card->qdio.in_q = qeth_alloc_qdio_queue();
if (!card->qdio.in_q) if (!card->qdio.in_q)
goto out_nomem; goto out_nomem;
QETH_DBF_TEXT(SETUP, 2, "inq");
QETH_DBF_HEX(SETUP, 2, &card->qdio.in_q, sizeof(void *));
memset(card->qdio.in_q, 0, sizeof(struct qeth_qdio_q));
/* give inbound qeth_qdio_buffers their qdio_buffers */
for (i = 0; i < QDIO_MAX_BUFFERS_PER_Q; ++i) {
card->qdio.in_q->bufs[i].buffer =
&card->qdio.in_q->qdio_bufs[i];
card->qdio.in_q->bufs[i].rx_skb = NULL;
}
/* inbound buffer pool */ /* inbound buffer pool */
if (qeth_alloc_buffer_pool(card)) if (qeth_alloc_buffer_pool(card))
goto out_freeinq; goto out_freeinq;
...@@ -2446,8 +2451,7 @@ static int qeth_alloc_qdio_buffers(struct qeth_card *card) ...@@ -2446,8 +2451,7 @@ static int qeth_alloc_qdio_buffers(struct qeth_card *card)
if (!card->qdio.out_qs) if (!card->qdio.out_qs)
goto out_freepool; goto out_freepool;
for (i = 0; i < card->qdio.no_out_queues; ++i) { for (i = 0; i < card->qdio.no_out_queues; ++i) {
card->qdio.out_qs[i] = kzalloc(sizeof(struct qeth_qdio_out_q), card->qdio.out_qs[i] = qeth_alloc_qdio_out_buf();
GFP_KERNEL);
if (!card->qdio.out_qs[i]) if (!card->qdio.out_qs[i])
goto out_freeoutq; goto out_freeoutq;
QETH_DBF_TEXT_(SETUP, 2, "outq %i", i); QETH_DBF_TEXT_(SETUP, 2, "outq %i", i);
...@@ -2476,7 +2480,7 @@ static int qeth_alloc_qdio_buffers(struct qeth_card *card) ...@@ -2476,7 +2480,7 @@ static int qeth_alloc_qdio_buffers(struct qeth_card *card)
} }
out_freeoutq: out_freeoutq:
while (i > 0) { while (i > 0) {
kfree(card->qdio.out_qs[--i]); qeth_free_qdio_out_buf(card->qdio.out_qs[--i]);
qeth_clear_outq_buffers(card->qdio.out_qs[i], 1); qeth_clear_outq_buffers(card->qdio.out_qs[i], 1);
} }
kfree(card->qdio.out_qs); kfree(card->qdio.out_qs);
...@@ -2484,13 +2488,42 @@ static int qeth_alloc_qdio_buffers(struct qeth_card *card) ...@@ -2484,13 +2488,42 @@ static int qeth_alloc_qdio_buffers(struct qeth_card *card)
out_freepool: out_freepool:
qeth_free_buffer_pool(card); qeth_free_buffer_pool(card);
out_freeinq: out_freeinq:
kfree(card->qdio.in_q); qeth_free_qdio_queue(card->qdio.in_q);
card->qdio.in_q = NULL; card->qdio.in_q = NULL;
out_nomem: out_nomem:
atomic_set(&card->qdio.state, QETH_QDIO_UNINITIALIZED); atomic_set(&card->qdio.state, QETH_QDIO_UNINITIALIZED);
return -ENOMEM; return -ENOMEM;
} }
static void qeth_free_qdio_buffers(struct qeth_card *card)
{
int i, j;
if (atomic_xchg(&card->qdio.state, QETH_QDIO_UNINITIALIZED) ==
QETH_QDIO_UNINITIALIZED)
return;
qeth_free_cq(card);
cancel_delayed_work_sync(&card->buffer_reclaim_work);
for (j = 0; j < QDIO_MAX_BUFFERS_PER_Q; ++j) {
if (card->qdio.in_q->bufs[j].rx_skb)
dev_kfree_skb_any(card->qdio.in_q->bufs[j].rx_skb);
}
qeth_free_qdio_queue(card->qdio.in_q);
card->qdio.in_q = NULL;
/* inbound buffer pool */
qeth_free_buffer_pool(card);
/* free outbound qdio_qs */
if (card->qdio.out_qs) {
for (i = 0; i < card->qdio.no_out_queues; ++i) {
qeth_clear_outq_buffers(card->qdio.out_qs[i], 1);
qeth_free_qdio_out_buf(card->qdio.out_qs[i]);
}
kfree(card->qdio.out_qs);
card->qdio.out_qs = NULL;
}
}
static void qeth_create_qib_param_field(struct qeth_card *card, static void qeth_create_qib_param_field(struct qeth_card *card,
char *param_field) char *param_field)
{ {
...@@ -2788,8 +2821,8 @@ int qeth_init_qdio_queues(struct qeth_card *card) ...@@ -2788,8 +2821,8 @@ int qeth_init_qdio_queues(struct qeth_card *card)
QETH_DBF_TEXT(SETUP, 2, "initqdqs"); QETH_DBF_TEXT(SETUP, 2, "initqdqs");
/* inbound queue */ /* inbound queue */
memset(card->qdio.in_q->qdio_bufs, 0, qdio_reset_buffers(card->qdio.in_q->qdio_bufs,
QDIO_MAX_BUFFERS_PER_Q * sizeof(struct qdio_buffer)); QDIO_MAX_BUFFERS_PER_Q);
qeth_initialize_working_pool_list(card); qeth_initialize_working_pool_list(card);
/*give only as many buffers to hardware as we have buffer pool entries*/ /*give only as many buffers to hardware as we have buffer pool entries*/
for (i = 0; i < card->qdio.in_buf_pool.buf_count - 1; ++i) for (i = 0; i < card->qdio.in_buf_pool.buf_count - 1; ++i)
...@@ -2811,8 +2844,8 @@ int qeth_init_qdio_queues(struct qeth_card *card) ...@@ -2811,8 +2844,8 @@ int qeth_init_qdio_queues(struct qeth_card *card)
/* outbound queue */ /* outbound queue */
for (i = 0; i < card->qdio.no_out_queues; ++i) { for (i = 0; i < card->qdio.no_out_queues; ++i) {
memset(card->qdio.out_qs[i]->qdio_bufs, 0, qdio_reset_buffers(card->qdio.out_qs[i]->qdio_bufs,
QDIO_MAX_BUFFERS_PER_Q * sizeof(struct qdio_buffer)); QDIO_MAX_BUFFERS_PER_Q);
for (j = 0; j < QDIO_MAX_BUFFERS_PER_Q; ++j) { for (j = 0; j < QDIO_MAX_BUFFERS_PER_Q; ++j) {
qeth_clear_output_buffer(card->qdio.out_qs[i], qeth_clear_output_buffer(card->qdio.out_qs[i],
card->qdio.out_qs[i]->bufs[j], card->qdio.out_qs[i]->bufs[j],
...@@ -3569,7 +3602,7 @@ static void qeth_qdio_cq_handler(struct qeth_card *card, ...@@ -3569,7 +3602,7 @@ static void qeth_qdio_cq_handler(struct qeth_card *card,
for (i = first_element; i < first_element + count; ++i) { for (i = first_element; i < first_element + count; ++i) {
int bidx = i % QDIO_MAX_BUFFERS_PER_Q; int bidx = i % QDIO_MAX_BUFFERS_PER_Q;
struct qdio_buffer *buffer = &cq->qdio_bufs[bidx]; struct qdio_buffer *buffer = cq->qdio_bufs[bidx];
int e; int e;
e = 0; e = 0;
......
...@@ -14,27 +14,10 @@ ...@@ -14,27 +14,10 @@
#include "zfcp_ext.h" #include "zfcp_ext.h"
#include "zfcp_qdio.h" #include "zfcp_qdio.h"
#define QBUFF_PER_PAGE (PAGE_SIZE / sizeof(struct qdio_buffer))
static bool enable_multibuffer = 1; static bool enable_multibuffer = 1;
module_param_named(datarouter, enable_multibuffer, bool, 0400); module_param_named(datarouter, enable_multibuffer, bool, 0400);
MODULE_PARM_DESC(datarouter, "Enable hardware data router support (default on)"); MODULE_PARM_DESC(datarouter, "Enable hardware data router support (default on)");
static int zfcp_qdio_buffers_enqueue(struct qdio_buffer **sbal)
{
int pos;
for (pos = 0; pos < QDIO_MAX_BUFFERS_PER_Q; pos += QBUFF_PER_PAGE) {
sbal[pos] = (struct qdio_buffer *) get_zeroed_page(GFP_KERNEL);
if (!sbal[pos])
return -ENOMEM;
}
for (pos = 0; pos < QDIO_MAX_BUFFERS_PER_Q; pos++)
if (pos % QBUFF_PER_PAGE)
sbal[pos] = sbal[pos - 1] + 1;
return 0;
}
static void zfcp_qdio_handler_error(struct zfcp_qdio *qdio, char *id, static void zfcp_qdio_handler_error(struct zfcp_qdio *qdio, char *id,
unsigned int qdio_err) unsigned int qdio_err)
{ {
...@@ -326,15 +309,30 @@ static void zfcp_qdio_setup_init_data(struct qdio_initialize *id, ...@@ -326,15 +309,30 @@ static void zfcp_qdio_setup_init_data(struct qdio_initialize *id,
static int zfcp_qdio_allocate(struct zfcp_qdio *qdio) static int zfcp_qdio_allocate(struct zfcp_qdio *qdio)
{ {
struct qdio_initialize init_data; struct qdio_initialize init_data;
int ret;
if (zfcp_qdio_buffers_enqueue(qdio->req_q) || ret = qdio_alloc_buffers(qdio->req_q, QDIO_MAX_BUFFERS_PER_Q);
zfcp_qdio_buffers_enqueue(qdio->res_q)) if (ret)
return -ENOMEM; return -ENOMEM;
ret = qdio_alloc_buffers(qdio->res_q, QDIO_MAX_BUFFERS_PER_Q);
if (ret)
goto free_req_q;
zfcp_qdio_setup_init_data(&init_data, qdio); zfcp_qdio_setup_init_data(&init_data, qdio);
init_waitqueue_head(&qdio->req_q_wq); init_waitqueue_head(&qdio->req_q_wq);
return qdio_allocate(&init_data); ret = qdio_allocate(&init_data);
if (ret)
goto free_res_q;
return 0;
free_res_q:
qdio_free_buffers(qdio->res_q, QDIO_MAX_BUFFERS_PER_Q);
free_req_q:
qdio_free_buffers(qdio->req_q, QDIO_MAX_BUFFERS_PER_Q);
return ret;
} }
/** /**
...@@ -448,19 +446,14 @@ int zfcp_qdio_open(struct zfcp_qdio *qdio) ...@@ -448,19 +446,14 @@ int zfcp_qdio_open(struct zfcp_qdio *qdio)
void zfcp_qdio_destroy(struct zfcp_qdio *qdio) void zfcp_qdio_destroy(struct zfcp_qdio *qdio)
{ {
int p;
if (!qdio) if (!qdio)
return; return;
if (qdio->adapter->ccw_device) if (qdio->adapter->ccw_device)
qdio_free(qdio->adapter->ccw_device); qdio_free(qdio->adapter->ccw_device);
for (p = 0; p < QDIO_MAX_BUFFERS_PER_Q; p += QBUFF_PER_PAGE) { qdio_free_buffers(qdio->req_q, QDIO_MAX_BUFFERS_PER_Q);
free_page((unsigned long) qdio->req_q[p]); qdio_free_buffers(qdio->res_q, QDIO_MAX_BUFFERS_PER_Q);
free_page((unsigned long) qdio->res_q[p]);
}
kfree(qdio); kfree(qdio);
} }
...@@ -475,7 +468,7 @@ int zfcp_qdio_setup(struct zfcp_adapter *adapter) ...@@ -475,7 +468,7 @@ int zfcp_qdio_setup(struct zfcp_adapter *adapter)
qdio->adapter = adapter; qdio->adapter = adapter;
if (zfcp_qdio_allocate(qdio)) { if (zfcp_qdio_allocate(qdio)) {
zfcp_qdio_destroy(qdio); kfree(qdio);
return -ENOMEM; return -ENOMEM;
} }
......
...@@ -1293,7 +1293,7 @@ config DIAG288_WATCHDOG ...@@ -1293,7 +1293,7 @@ config DIAG288_WATCHDOG
both. both.
To compile this driver as a module, choose M here. The module To compile this driver as a module, choose M here. The module
will be called vmwatchdog. will be called diag288_wdt.
# SUPERH (sh + sh64) Architecture # SUPERH (sh + sh64) Architecture
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment