]> git.saurik.com Git - apple/xnu.git/blame - osfmk/i386/machine_routines.c
xnu-517.3.15.tar.gz
[apple/xnu.git] / osfmk / i386 / machine_routines.c
CommitLineData
1c79356b
A
1/*
2 * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
3 *
4 * @APPLE_LICENSE_HEADER_START@
5 *
43866e37 6 * Copyright (c) 1999-2003 Apple Computer, Inc. All Rights Reserved.
1c79356b 7 *
43866e37
A
8 * This file contains Original Code and/or Modifications of Original Code
9 * as defined in and that are subject to the Apple Public Source License
10 * Version 2.0 (the 'License'). You may not use this file except in
11 * compliance with the License. Please obtain a copy of the License at
12 * http://www.opensource.apple.com/apsl/ and read it before using this
13 * file.
14 *
15 * The Original Code and all software distributed under the License are
16 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
1c79356b
A
17 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
18 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
43866e37
A
19 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
20 * Please see the License for the specific language governing rights and
21 * limitations under the License.
1c79356b
A
22 *
23 * @APPLE_LICENSE_HEADER_END@
24 */
25#include <i386/machine_routines.h>
26#include <i386/io_map_entries.h>
55e303ae
A
27#include <i386/cpuid.h>
28#include <i386/fpu.h>
29#include <kern/processor.h>
1c79356b 30#include <kern/cpu_data.h>
9bccf70c 31#include <kern/thread_act.h>
55e303ae
A
32#include <i386/machine_cpu.h>
33#include <i386/mp.h>
34#include <i386/mp_events.h>
35
36static int max_cpus_initialized = 0;
37
38#define MAX_CPUS_SET 0x1
39#define MAX_CPUS_WAIT 0x2
1c79356b
A
40
41/* IO memory map services */
42
43/* Map memory map IO space */
44vm_offset_t ml_io_map(
45 vm_offset_t phys_addr,
46 vm_size_t size)
47{
48 return(io_map(phys_addr,size));
49}
50
51/* boot memory allocation */
52vm_offset_t ml_static_malloc(
53 vm_size_t size)
54{
55 return((vm_offset_t)NULL);
56}
57
58vm_offset_t
59ml_static_ptovirt(
60 vm_offset_t paddr)
61{
62 return phystokv(paddr);
63}
64
65void
66ml_static_mfree(
67 vm_offset_t vaddr,
68 vm_size_t size)
69{
70 return;
71}
72
73/* virtual to physical on wired pages */
74vm_offset_t ml_vtophys(
75 vm_offset_t vaddr)
76{
77 return kvtophys(vaddr);
78}
79
80/* Interrupt handling */
81
55e303ae
A
82/* Initialize Interrupts */
83void ml_init_interrupt(void)
84{
85 (void) ml_set_interrupts_enabled(TRUE);
86}
87
1c79356b
A
88/* Get Interrupts Enabled */
89boolean_t ml_get_interrupts_enabled(void)
90{
91 unsigned long flags;
92
93 __asm__ volatile("pushf; popl %0" : "=r" (flags));
94 return (flags & EFL_IF) != 0;
95}
96
97/* Set Interrupts Enabled */
98boolean_t ml_set_interrupts_enabled(boolean_t enable)
99{
100 unsigned long flags;
101
102 __asm__ volatile("pushf; popl %0" : "=r" (flags));
103
104 if (enable)
105 __asm__ volatile("sti");
106 else
107 __asm__ volatile("cli");
108
109 return (flags & EFL_IF) != 0;
110}
111
112/* Check if running at interrupt context */
113boolean_t ml_at_interrupt_context(void)
114{
115 return get_interrupt_level() != 0;
116}
117
118/* Generate a fake interrupt */
119void ml_cause_interrupt(void)
120{
121 panic("ml_cause_interrupt not defined yet on Intel");
122}
123
d52fe63f
A
124void ml_thread_policy(
125 thread_t thread,
126 unsigned policy_id,
127 unsigned policy_info)
128{
55e303ae
A
129 if (policy_id == MACHINE_GROUP)
130 thread_bind(thread, master_processor);
131
132 if (policy_info & MACHINE_NETWORK_WORKLOOP) {
133 spl_t s = splsched();
134
135 thread_lock(thread);
136
137 set_priority(thread, thread->priority + 1);
138
139 thread_unlock(thread);
140 splx(s);
141 }
d52fe63f
A
142}
143
1c79356b
A
144/* Initialize Interrupts */
145void ml_install_interrupt_handler(
146 void *nub,
147 int source,
148 void *target,
149 IOInterruptHandler handler,
150 void *refCon)
151{
152 boolean_t current_state;
153
154 current_state = ml_get_interrupts_enabled();
155
156 PE_install_interrupt_handler(nub, source, target,
157 (IOInterruptHandler) handler, refCon);
158
159 (void) ml_set_interrupts_enabled(current_state);
55e303ae
A
160
161 initialize_screen(0, kPEAcquireScreen);
162}
163
164void
165machine_idle(void)
166{
167 DBGLOG(cpu_handle, cpu_number(), MP_IDLE);
168 __asm__ volatile("sti; hlt": : :"memory");
169 __asm__ volatile("cli");
170 DBGLOG(cpu_handle, cpu_number(), MP_UNIDLE);
1c79356b
A
171}
172
173void
174machine_signal_idle(
175 processor_t processor)
176{
55e303ae
A
177 cpu_interrupt(processor->slot_num);
178}
179
180kern_return_t
181ml_processor_register(
182 cpu_id_t cpu_id,
183 uint32_t lapic_id,
184 processor_t *processor,
185 ipi_handler_t *ipi_handler,
186 boolean_t boot_cpu)
187{
188 kern_return_t ret;
189 int target_cpu;
190
191 if (cpu_register(&target_cpu) != KERN_SUCCESS)
192 return KERN_FAILURE;
193
194 assert((boot_cpu && (target_cpu == 0)) ||
195 (!boot_cpu && (target_cpu != 0)));
196
197 lapic_cpu_map(lapic_id, target_cpu);
198 cpu_data[target_cpu].cpu_id = cpu_id;
199 cpu_data[target_cpu].cpu_phys_number = lapic_id;
200 *processor = cpu_to_processor(target_cpu);
201 *ipi_handler = NULL;
202
203 return KERN_SUCCESS;
1c79356b
A
204}
205
43866e37
A
206void
207ml_cpu_get_info(ml_cpu_info_t *cpu_info)
208{
55e303ae
A
209 boolean_t os_supports_sse;
210 i386_cpu_info_t *cpuid_infop;
211
212 if (cpu_info == NULL)
213 return;
214
215 /*
216 * Are we supporting XMM/SSE/SSE2?
217 * As distinct from whether the cpu has these capabilities.
218 */
219 os_supports_sse = get_cr4() & CR4_XMM;
220 if ((cpuid_features() & CPUID_FEATURE_SSE2) && os_supports_sse)
221 cpu_info->vector_unit = 4;
222 else if ((cpuid_features() & CPUID_FEATURE_SSE) && os_supports_sse)
223 cpu_info->vector_unit = 3;
224 else if (cpuid_features() & CPUID_FEATURE_MMX)
225 cpu_info->vector_unit = 2;
226 else
227 cpu_info->vector_unit = 0;
228
229 cpuid_infop = cpuid_info();
230
231 cpu_info->cache_line_size = cpuid_infop->cache_linesize;
232
233 cpu_info->l1_icache_size = cpuid_infop->cache_size[L1I];
234 cpu_info->l1_dcache_size = cpuid_infop->cache_size[L1D];
235
236 cpu_info->l2_settings = 1;
237 cpu_info->l2_cache_size = cpuid_infop->cache_size[L2U];
238
239 /* XXX No L3 */
240 cpu_info->l3_settings = 0;
241 cpu_info->l3_cache_size = 0xFFFFFFFF;
43866e37
A
242}
243
244void
245ml_init_max_cpus(unsigned long max_cpus)
246{
55e303ae
A
247 boolean_t current_state;
248
249 current_state = ml_set_interrupts_enabled(FALSE);
250 if (max_cpus_initialized != MAX_CPUS_SET) {
251 if (max_cpus > 0 && max_cpus < NCPUS)
252 machine_info.max_cpus = max_cpus;
253 if (max_cpus_initialized == MAX_CPUS_WAIT)
254 wakeup((event_t)&max_cpus_initialized);
255 max_cpus_initialized = MAX_CPUS_SET;
256 }
257 (void) ml_set_interrupts_enabled(current_state);
43866e37
A
258}
259
260int
261ml_get_max_cpus(void)
262{
55e303ae 263 boolean_t current_state;
43866e37 264
55e303ae
A
265 current_state = ml_set_interrupts_enabled(FALSE);
266 if (max_cpus_initialized != MAX_CPUS_SET) {
267 max_cpus_initialized = MAX_CPUS_WAIT;
268 assert_wait((event_t)&max_cpus_initialized, THREAD_UNINT);
269 (void)thread_block(THREAD_CONTINUE_NULL);
270 }
271 (void) ml_set_interrupts_enabled(current_state);
272 return(machine_info.max_cpus);
43866e37
A
273}
274
1c79356b
A
275/* Stubs for pc tracing mechanism */
276
277int *pc_trace_buf;
278int pc_trace_cnt = 0;
279
280int
281set_be_bit()
282{
283 return(0);
284}
285
286int
287clr_be_bit()
288{
289 return(0);
290}
291
292int
293be_tracing()
294{
295 return(0);
296}
9bccf70c
A
297
298#undef current_act
299thread_act_t
300current_act(void)
301{
55e303ae 302 return(current_act_fast());
9bccf70c 303}
55e303ae
A
304
305#undef current_thread
306thread_t
307current_thread(void)
308{
309 return(current_act_fast());
310}