1b2441318SGreg Kroah-Hartman /* SPDX-License-Identifier: GPL-2.0 */ 2a439fe51SSam Ravnborg #ifndef __MMU_H 3a439fe51SSam Ravnborg #define __MMU_H 4a439fe51SSam Ravnborg 5a439fe51SSam Ravnborg #include <linux/const.h> 6a439fe51SSam Ravnborg #include <asm/page.h> 7a439fe51SSam Ravnborg #include <asm/hypervisor.h> 8a439fe51SSam Ravnborg 9a439fe51SSam Ravnborg #define CTX_NR_BITS 13 10a439fe51SSam Ravnborg 11a439fe51SSam Ravnborg #define TAG_CONTEXT_BITS ((_AC(1,UL) << CTX_NR_BITS) - _AC(1,UL)) 12a439fe51SSam Ravnborg 13a439fe51SSam Ravnborg /* UltraSPARC-III+ and later have a feature whereby you can 14a439fe51SSam Ravnborg * select what page size the various Data-TLB instances in the 15a439fe51SSam Ravnborg * chip. In order to gracefully support this, we put the version 16a439fe51SSam Ravnborg * field in a spot outside of the areas of the context register 17a439fe51SSam Ravnborg * where this parameter is specified. 18a439fe51SSam Ravnborg */ 19a439fe51SSam Ravnborg #define CTX_VERSION_SHIFT 22 20a439fe51SSam Ravnborg #define CTX_VERSION_MASK ((~0UL) << CTX_VERSION_SHIFT) 21a439fe51SSam Ravnborg 22a439fe51SSam Ravnborg #define CTX_PGSZ_8KB _AC(0x0,UL) 23a439fe51SSam Ravnborg #define CTX_PGSZ_64KB _AC(0x1,UL) 24a439fe51SSam Ravnborg #define CTX_PGSZ_512KB _AC(0x2,UL) 25a439fe51SSam Ravnborg #define CTX_PGSZ_4MB _AC(0x3,UL) 26a439fe51SSam Ravnborg #define CTX_PGSZ_BITS _AC(0x7,UL) 27a439fe51SSam Ravnborg #define CTX_PGSZ0_NUC_SHIFT 61 28a439fe51SSam Ravnborg #define CTX_PGSZ1_NUC_SHIFT 58 29a439fe51SSam Ravnborg #define CTX_PGSZ0_SHIFT 16 30a439fe51SSam Ravnborg #define CTX_PGSZ1_SHIFT 19 31a439fe51SSam Ravnborg #define CTX_PGSZ_MASK ((CTX_PGSZ_BITS << CTX_PGSZ0_SHIFT) | \ 32a439fe51SSam Ravnborg (CTX_PGSZ_BITS << CTX_PGSZ1_SHIFT)) 33a439fe51SSam Ravnborg 34a439fe51SSam Ravnborg #define CTX_PGSZ_BASE CTX_PGSZ_8KB 35a439fe51SSam Ravnborg #define CTX_PGSZ_HUGE CTX_PGSZ_4MB 36a439fe51SSam Ravnborg #define CTX_PGSZ_KERN CTX_PGSZ_4MB 37a439fe51SSam Ravnborg 38a439fe51SSam Ravnborg /* Thus, when running on UltraSPARC-III+ and later, we use the following 39a439fe51SSam Ravnborg * PRIMARY_CONTEXT register values for the kernel context. 40a439fe51SSam Ravnborg */ 41a439fe51SSam Ravnborg #define CTX_CHEETAH_PLUS_NUC \ 42a439fe51SSam Ravnborg ((CTX_PGSZ_KERN << CTX_PGSZ0_NUC_SHIFT) | \ 43a439fe51SSam Ravnborg (CTX_PGSZ_BASE << CTX_PGSZ1_NUC_SHIFT)) 44a439fe51SSam Ravnborg 45a439fe51SSam Ravnborg #define CTX_CHEETAH_PLUS_CTX0 \ 46a439fe51SSam Ravnborg ((CTX_PGSZ_KERN << CTX_PGSZ0_SHIFT) | \ 47a439fe51SSam Ravnborg (CTX_PGSZ_BASE << CTX_PGSZ1_SHIFT)) 48a439fe51SSam Ravnborg 49a439fe51SSam Ravnborg /* If you want "the TLB context number" use CTX_NR_MASK. If you 50a439fe51SSam Ravnborg * want "the bits I program into the context registers" use 51a439fe51SSam Ravnborg * CTX_HW_MASK. 52a439fe51SSam Ravnborg */ 53a439fe51SSam Ravnborg #define CTX_NR_MASK TAG_CONTEXT_BITS 54a439fe51SSam Ravnborg #define CTX_HW_MASK (CTX_NR_MASK | CTX_PGSZ_MASK) 55a439fe51SSam Ravnborg 56c4415235SPavel Tatashin #define CTX_FIRST_VERSION BIT(CTX_VERSION_SHIFT) 57a439fe51SSam Ravnborg #define CTX_VALID(__ctx) \ 58a439fe51SSam Ravnborg (!(((__ctx.sparc64_ctx_val) ^ tlb_context_cache) & CTX_VERSION_MASK)) 59a439fe51SSam Ravnborg #define CTX_HWBITS(__ctx) ((__ctx.sparc64_ctx_val) & CTX_HW_MASK) 60a439fe51SSam Ravnborg #define CTX_NRBITS(__ctx) ((__ctx.sparc64_ctx_val) & CTX_NR_MASK) 61a439fe51SSam Ravnborg 62a439fe51SSam Ravnborg #ifndef __ASSEMBLY__ 63a439fe51SSam Ravnborg 64a439fe51SSam Ravnborg #define TSB_ENTRY_ALIGNMENT 16 65a439fe51SSam Ravnborg 66a439fe51SSam Ravnborg struct tsb { 67a439fe51SSam Ravnborg unsigned long tag; 68a439fe51SSam Ravnborg unsigned long pte; 69a439fe51SSam Ravnborg } __attribute__((aligned(TSB_ENTRY_ALIGNMENT))); 70a439fe51SSam Ravnborg 71f05a6865SSam Ravnborg void __tsb_insert(unsigned long ent, unsigned long tag, unsigned long pte); 72f05a6865SSam Ravnborg void tsb_flush(unsigned long ent, unsigned long tag); 73f05a6865SSam Ravnborg void tsb_init(struct tsb *tsb, unsigned long size); 74a439fe51SSam Ravnborg 75a439fe51SSam Ravnborg struct tsb_config { 76a439fe51SSam Ravnborg struct tsb *tsb; 77a439fe51SSam Ravnborg unsigned long tsb_rss_limit; 78a439fe51SSam Ravnborg unsigned long tsb_nentries; 79a439fe51SSam Ravnborg unsigned long tsb_reg_val; 80a439fe51SSam Ravnborg unsigned long tsb_map_vaddr; 81a439fe51SSam Ravnborg unsigned long tsb_map_pte; 82a439fe51SSam Ravnborg }; 83a439fe51SSam Ravnborg 84a439fe51SSam Ravnborg #define MM_TSB_BASE 0 85a439fe51SSam Ravnborg 869e695d2eSDavid Miller #if defined(CONFIG_HUGETLB_PAGE) || defined(CONFIG_TRANSPARENT_HUGEPAGE) 87a439fe51SSam Ravnborg #define MM_TSB_HUGE 1 88a439fe51SSam Ravnborg #define MM_NUM_TSBS 2 89a439fe51SSam Ravnborg #else 90a439fe51SSam Ravnborg #define MM_NUM_TSBS 1 91a439fe51SSam Ravnborg #endif 92a439fe51SSam Ravnborg 93*74a04967SKhalid Aziz /* ADI tags are stored when a page is swapped out and the storage for 94*74a04967SKhalid Aziz * tags is allocated dynamically. There is a tag storage descriptor 95*74a04967SKhalid Aziz * associated with each set of tag storage pages. Tag storage descriptors 96*74a04967SKhalid Aziz * are allocated dynamically. Since kernel will allocate a full page for 97*74a04967SKhalid Aziz * each tag storage descriptor, we can store up to 98*74a04967SKhalid Aziz * PAGE_SIZE/sizeof(tag storage descriptor) descriptors on that page. 99*74a04967SKhalid Aziz */ 100*74a04967SKhalid Aziz typedef struct { 101*74a04967SKhalid Aziz unsigned long start; /* Start address for this tag storage */ 102*74a04967SKhalid Aziz unsigned long end; /* Last address for tag storage */ 103*74a04967SKhalid Aziz unsigned char *tags; /* Where the tags are */ 104*74a04967SKhalid Aziz unsigned long tag_users; /* number of references to descriptor */ 105*74a04967SKhalid Aziz } tag_storage_desc_t; 106*74a04967SKhalid Aziz 107a439fe51SSam Ravnborg typedef struct { 108a439fe51SSam Ravnborg spinlock_t lock; 109a439fe51SSam Ravnborg unsigned long sparc64_ctx_val; 110af1b1a9bSMike Kravetz unsigned long hugetlb_pte_count; 111af1b1a9bSMike Kravetz unsigned long thp_pte_count; 112a439fe51SSam Ravnborg struct tsb_config tsb_block[MM_NUM_TSBS]; 113a439fe51SSam Ravnborg struct hv_tsb_descr tsb_descr[MM_NUM_TSBS]; 1149a08862aSNagarathnam Muthusamy void *vdso; 115*74a04967SKhalid Aziz bool adi; 116*74a04967SKhalid Aziz tag_storage_desc_t *tag_store; 117*74a04967SKhalid Aziz spinlock_t tag_lock; 118a439fe51SSam Ravnborg } mm_context_t; 119a439fe51SSam Ravnborg 120a439fe51SSam Ravnborg #endif /* !__ASSEMBLY__ */ 121a439fe51SSam Ravnborg 122a439fe51SSam Ravnborg #define TSB_CONFIG_TSB 0x00 123a439fe51SSam Ravnborg #define TSB_CONFIG_RSS_LIMIT 0x08 124a439fe51SSam Ravnborg #define TSB_CONFIG_NENTRIES 0x10 125a439fe51SSam Ravnborg #define TSB_CONFIG_REG_VAL 0x18 126a439fe51SSam Ravnborg #define TSB_CONFIG_MAP_VADDR 0x20 127a439fe51SSam Ravnborg #define TSB_CONFIG_MAP_PTE 0x28 128a439fe51SSam Ravnborg 129a439fe51SSam Ravnborg #endif /* __MMU_H */ 130