diff options
Diffstat (limited to 'arch/ia64/mm/tlb.c')
-rw-r--r-- | arch/ia64/mm/tlb.c | 46 |
1 files changed, 41 insertions, 5 deletions
diff --git a/arch/ia64/mm/tlb.c b/arch/ia64/mm/tlb.c index d41d6076ed03..1a8948fd0029 100644 --- a/arch/ia64/mm/tlb.c +++ b/arch/ia64/mm/tlb.c | |||
@@ -120,19 +120,45 @@ static int need_ptcg_sem = 1; | |||
120 | static int toolatetochangeptcgsem = 0; | 120 | static int toolatetochangeptcgsem = 0; |
121 | 121 | ||
122 | /* | 122 | /* |
123 | * Kernel parameter "nptcg=" overrides max number of concurrent global TLB | ||
124 | * purges which is reported from either PAL or SAL PALO. | ||
125 | * | ||
126 | * We don't have sanity checking for nptcg value. It's the user's responsibility | ||
127 | * for valid nptcg value on the platform. Otherwise, kernel may hang in some | ||
128 | * cases. | ||
129 | */ | ||
130 | static int __init | ||
131 | set_nptcg(char *str) | ||
132 | { | ||
133 | int value = 0; | ||
134 | |||
135 | get_option(&str, &value); | ||
136 | setup_ptcg_sem(value, NPTCG_FROM_KERNEL_PARAMETER); | ||
137 | |||
138 | return 1; | ||
139 | } | ||
140 | |||
141 | __setup("nptcg=", set_nptcg); | ||
142 | |||
143 | /* | ||
123 | * Maximum number of simultaneous ptc.g purges in the system can | 144 | * Maximum number of simultaneous ptc.g purges in the system can |
124 | * be defined by PAL_VM_SUMMARY (in which case we should take | 145 | * be defined by PAL_VM_SUMMARY (in which case we should take |
125 | * the smallest value for any cpu in the system) or by the PAL | 146 | * the smallest value for any cpu in the system) or by the PAL |
126 | * override table (in which case we should ignore the value from | 147 | * override table (in which case we should ignore the value from |
127 | * PAL_VM_SUMMARY). | 148 | * PAL_VM_SUMMARY). |
128 | * | 149 | * |
150 | * Kernel parameter "nptcg=" overrides maximum number of simultanesous ptc.g | ||
151 | * purges defined in either PAL_VM_SUMMARY or PAL override table. In this case, | ||
152 | * we should ignore the value from either PAL_VM_SUMMARY or PAL override table. | ||
153 | * | ||
129 | * Complicating the logic here is the fact that num_possible_cpus() | 154 | * Complicating the logic here is the fact that num_possible_cpus() |
130 | * isn't fully setup until we start bringing cpus online. | 155 | * isn't fully setup until we start bringing cpus online. |
131 | */ | 156 | */ |
132 | void | 157 | void |
133 | setup_ptcg_sem(int max_purges, int from_palo) | 158 | setup_ptcg_sem(int max_purges, int nptcg_from) |
134 | { | 159 | { |
135 | static int have_palo; | 160 | static int kp_override; |
161 | static int palo_override; | ||
136 | static int firstcpu = 1; | 162 | static int firstcpu = 1; |
137 | 163 | ||
138 | if (toolatetochangeptcgsem) { | 164 | if (toolatetochangeptcgsem) { |
@@ -140,8 +166,18 @@ setup_ptcg_sem(int max_purges, int from_palo) | |||
140 | return; | 166 | return; |
141 | } | 167 | } |
142 | 168 | ||
143 | if (from_palo) { | 169 | if (nptcg_from == NPTCG_FROM_KERNEL_PARAMETER) { |
144 | have_palo = 1; | 170 | kp_override = 1; |
171 | nptcg = max_purges; | ||
172 | goto resetsema; | ||
173 | } | ||
174 | if (kp_override) { | ||
175 | need_ptcg_sem = num_possible_cpus() > nptcg; | ||
176 | return; | ||
177 | } | ||
178 | |||
179 | if (nptcg_from == NPTCG_FROM_PALO) { | ||
180 | palo_override = 1; | ||
145 | 181 | ||
146 | /* In PALO max_purges == 0 really means it! */ | 182 | /* In PALO max_purges == 0 really means it! */ |
147 | if (max_purges == 0) | 183 | if (max_purges == 0) |
@@ -153,7 +189,7 @@ setup_ptcg_sem(int max_purges, int from_palo) | |||
153 | } | 189 | } |
154 | goto resetsema; | 190 | goto resetsema; |
155 | } | 191 | } |
156 | if (have_palo) { | 192 | if (palo_override) { |
157 | if (nptcg != PALO_MAX_TLB_PURGES) | 193 | if (nptcg != PALO_MAX_TLB_PURGES) |
158 | need_ptcg_sem = (num_possible_cpus() > nptcg); | 194 | need_ptcg_sem = (num_possible_cpus() > nptcg); |
159 | return; | 195 | return; |