162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * linux/arch/m68k/sun3/dvma.c
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Written by Sam Creasey
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * Sun3 IOMMU routines used for dvma accesses.
862306a36Sopenharmony_ci *
962306a36Sopenharmony_ci */
1062306a36Sopenharmony_ci
1162306a36Sopenharmony_ci#include <linux/init.h>
1262306a36Sopenharmony_ci#include <linux/kernel.h>
1362306a36Sopenharmony_ci#include <linux/mm.h>
1462306a36Sopenharmony_ci#include <linux/memblock.h>
1562306a36Sopenharmony_ci#include <linux/list.h>
1662306a36Sopenharmony_ci#include <asm/page.h>
1762306a36Sopenharmony_ci#include <asm/sun3mmu.h>
1862306a36Sopenharmony_ci#include <asm/dvma.h>
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_ci
2162306a36Sopenharmony_cistatic unsigned long ptelist[120];
2262306a36Sopenharmony_ci
2362306a36Sopenharmony_cistatic unsigned long dvma_page(unsigned long kaddr, unsigned long vaddr)
2462306a36Sopenharmony_ci{
2562306a36Sopenharmony_ci	unsigned long pte;
2662306a36Sopenharmony_ci	unsigned long j;
2762306a36Sopenharmony_ci	pte_t ptep;
2862306a36Sopenharmony_ci
2962306a36Sopenharmony_ci	j = *(volatile unsigned long *)kaddr;
3062306a36Sopenharmony_ci	*(volatile unsigned long *)kaddr = j;
3162306a36Sopenharmony_ci
3262306a36Sopenharmony_ci	ptep = pfn_pte(virt_to_pfn((void *)kaddr), PAGE_KERNEL);
3362306a36Sopenharmony_ci	pte = pte_val(ptep);
3462306a36Sopenharmony_ci//	pr_info("dvma_remap: addr %lx -> %lx pte %08lx\n", kaddr, vaddr, pte);
3562306a36Sopenharmony_ci	if(ptelist[(vaddr & 0xff000) >> PAGE_SHIFT] != pte) {
3662306a36Sopenharmony_ci		sun3_put_pte(vaddr, pte);
3762306a36Sopenharmony_ci		ptelist[(vaddr & 0xff000) >> PAGE_SHIFT] = pte;
3862306a36Sopenharmony_ci	}
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_ci	return (vaddr + (kaddr & ~PAGE_MASK));
4162306a36Sopenharmony_ci
4262306a36Sopenharmony_ci}
4362306a36Sopenharmony_ci
4462306a36Sopenharmony_ciint dvma_map_iommu(unsigned long kaddr, unsigned long baddr,
4562306a36Sopenharmony_ci			      int len)
4662306a36Sopenharmony_ci{
4762306a36Sopenharmony_ci
4862306a36Sopenharmony_ci	unsigned long end;
4962306a36Sopenharmony_ci	unsigned long vaddr;
5062306a36Sopenharmony_ci
5162306a36Sopenharmony_ci	vaddr = dvma_btov(baddr);
5262306a36Sopenharmony_ci
5362306a36Sopenharmony_ci	end = vaddr + len;
5462306a36Sopenharmony_ci
5562306a36Sopenharmony_ci	while(vaddr < end) {
5662306a36Sopenharmony_ci		dvma_page(kaddr, vaddr);
5762306a36Sopenharmony_ci		kaddr += PAGE_SIZE;
5862306a36Sopenharmony_ci		vaddr += PAGE_SIZE;
5962306a36Sopenharmony_ci	}
6062306a36Sopenharmony_ci
6162306a36Sopenharmony_ci	return 0;
6262306a36Sopenharmony_ci
6362306a36Sopenharmony_ci}
6462306a36Sopenharmony_ci
6562306a36Sopenharmony_civoid __init sun3_dvma_init(void)
6662306a36Sopenharmony_ci{
6762306a36Sopenharmony_ci	memset(ptelist, 0, sizeof(ptelist));
6862306a36Sopenharmony_ci}
69