aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--arch/sparc64/mm/init.c129
-rw-r--r--include/asm-sparc64/pgtable.h14
2 files changed, 62 insertions, 81 deletions
diff --git a/arch/sparc64/mm/init.c b/arch/sparc64/mm/init.c
index 56be943a6f0e..3cacea5da6ce 100644
--- a/arch/sparc64/mm/init.c
+++ b/arch/sparc64/mm/init.c
@@ -392,6 +392,67 @@ out:
392 put_cpu(); 392 put_cpu();
393} 393}
394 394
395struct linux_prom_translation {
396 unsigned long virt;
397 unsigned long size;
398 unsigned long data;
399};
400
401/* Exported for kernel TLB miss handling in ktlb.S */
402struct linux_prom_translation prom_trans[512] __read_mostly;
403unsigned int prom_trans_ents __read_mostly;
404
405/*
406 * Translate PROM's mapping we capture at boot time into physical address.
407 * The second parameter is only set from prom_callback() invocations.
408 */
409static unsigned long prom_virt_to_phys(unsigned long promva)
410{
411 unsigned long mask;
412 int i;
413
414 mask = _PAGE_PADDR_4U;
415 if (tlb_type == hypervisor)
416 mask = _PAGE_PADDR_4V;
417
418 for (i = 0; i < prom_trans_ents; i++) {
419 struct linux_prom_translation *p = &prom_trans[i];
420
421 if (promva >= p->virt &&
422 promva < (p->virt + p->size)) {
423 unsigned long base = p->data & mask;
424
425 return base + (promva & (8192 - 1));
426 }
427 }
428 return 0UL;
429}
430
431static unsigned long kvaddr_to_phys(unsigned long addr)
432{
433 pgd_t *pgdp;
434 pud_t *pudp;
435 pmd_t *pmdp;
436 pte_t *ptep;
437 unsigned long mask = _PAGE_PADDR_4U;
438
439 if (tlb_type == hypervisor)
440 mask = _PAGE_PADDR_4V;
441
442 if (addr >= PAGE_OFFSET)
443 return addr & mask;
444
445 if ((addr >= LOW_OBP_ADDRESS) && (addr < HI_OBP_ADDRESS))
446 return prom_virt_to_phys(addr);
447
448 pgdp = pgd_offset_k(addr);
449 pudp = pud_offset(pgdp, addr);
450 pmdp = pmd_offset(pudp, addr);
451 ptep = pte_offset_kernel(pmdp, addr);
452
453 return pte_val(*ptep) & mask;
454}
455
395void __kprobes flush_icache_range(unsigned long start, unsigned long end) 456void __kprobes flush_icache_range(unsigned long start, unsigned long end)
396{ 457{
397 /* Cheetah and Hypervisor platform cpus have coherent I-cache. */ 458 /* Cheetah and Hypervisor platform cpus have coherent I-cache. */
@@ -399,7 +460,7 @@ void __kprobes flush_icache_range(unsigned long start, unsigned long end)
399 unsigned long kaddr; 460 unsigned long kaddr;
400 461
401 for (kaddr = start; kaddr < end; kaddr += PAGE_SIZE) 462 for (kaddr = start; kaddr < end; kaddr += PAGE_SIZE)
402 __flush_icache_page(__get_phys(kaddr)); 463 __flush_icache_page(kvaddr_to_phys(kaddr));
403 } 464 }
404} 465}
405 466
@@ -436,16 +497,6 @@ void mmu_info(struct seq_file *m)
436#endif /* CONFIG_DEBUG_DCFLUSH */ 497#endif /* CONFIG_DEBUG_DCFLUSH */
437} 498}
438 499
439struct linux_prom_translation {
440 unsigned long virt;
441 unsigned long size;
442 unsigned long data;
443};
444
445/* Exported for kernel TLB miss handling in ktlb.S */
446struct linux_prom_translation prom_trans[512] __read_mostly;
447unsigned int prom_trans_ents __read_mostly;
448
449/* Exported for SMP bootup purposes. */ 500/* Exported for SMP bootup purposes. */
450unsigned long kern_locked_tte_data; 501unsigned long kern_locked_tte_data;
451 502
@@ -1875,62 +1926,6 @@ static unsigned long kern_large_tte(unsigned long paddr)
1875 return val | paddr; 1926 return val | paddr;
1876} 1927}
1877 1928
1878/*
1879 * Translate PROM's mapping we capture at boot time into physical address.
1880 * The second parameter is only set from prom_callback() invocations.
1881 */
1882unsigned long prom_virt_to_phys(unsigned long promva, int *error)
1883{
1884 unsigned long mask;
1885 int i;
1886
1887 mask = _PAGE_PADDR_4U;
1888 if (tlb_type == hypervisor)
1889 mask = _PAGE_PADDR_4V;
1890
1891 for (i = 0; i < prom_trans_ents; i++) {
1892 struct linux_prom_translation *p = &prom_trans[i];
1893
1894 if (promva >= p->virt &&
1895 promva < (p->virt + p->size)) {
1896 unsigned long base = p->data & mask;
1897
1898 if (error)
1899 *error = 0;
1900 return base + (promva & (8192 - 1));
1901 }
1902 }
1903 if (error)
1904 *error = 1;
1905 return 0UL;
1906}
1907
1908/* XXX We should kill off this ugly thing at so me point. XXX */
1909unsigned long sun4u_get_pte(unsigned long addr)
1910{
1911 pgd_t *pgdp;
1912 pud_t *pudp;
1913 pmd_t *pmdp;
1914 pte_t *ptep;
1915 unsigned long mask = _PAGE_PADDR_4U;
1916
1917 if (tlb_type == hypervisor)
1918 mask = _PAGE_PADDR_4V;
1919
1920 if (addr >= PAGE_OFFSET)
1921 return addr & mask;
1922
1923 if ((addr >= LOW_OBP_ADDRESS) && (addr < HI_OBP_ADDRESS))
1924 return prom_virt_to_phys(addr, NULL);
1925
1926 pgdp = pgd_offset_k(addr);
1927 pudp = pud_offset(pgdp, addr);
1928 pmdp = pmd_offset(pudp, addr);
1929 ptep = pte_offset_kernel(pmdp, addr);
1930
1931 return pte_val(*ptep) & mask;
1932}
1933
1934/* If not locked, zap it. */ 1929/* If not locked, zap it. */
1935void __flush_tlb_all(void) 1930void __flush_tlb_all(void)
1936{ 1931{
diff --git a/include/asm-sparc64/pgtable.h b/include/asm-sparc64/pgtable.h
index b12be7a869f6..fd46dd615b6f 100644
--- a/include/asm-sparc64/pgtable.h
+++ b/include/asm-sparc64/pgtable.h
@@ -737,20 +737,6 @@ extern unsigned long pte_file(pte_t);
737extern pte_t pgoff_to_pte(unsigned long); 737extern pte_t pgoff_to_pte(unsigned long);
738#define PTE_FILE_MAX_BITS (64UL - PAGE_SHIFT - 1UL) 738#define PTE_FILE_MAX_BITS (64UL - PAGE_SHIFT - 1UL)
739 739
740extern unsigned long prom_virt_to_phys(unsigned long, int *);
741
742extern unsigned long sun4u_get_pte(unsigned long);
743
744static inline unsigned long __get_phys(unsigned long addr)
745{
746 return sun4u_get_pte(addr);
747}
748
749static inline int __get_iospace(unsigned long addr)
750{
751 return ((sun4u_get_pte(addr) & 0xf0000000) >> 28);
752}
753
754extern unsigned long *sparc64_valid_addr_bitmap; 740extern unsigned long *sparc64_valid_addr_bitmap;
755 741
756/* Needs to be defined here and not in linux/mm.h, as it is arch dependent */ 742/* Needs to be defined here and not in linux/mm.h, as it is arch dependent */