ppc: Use config file to enable RTAS
[openbios.git] / arch / ppc / pearpc / methods.c
blobf505b6ceafcf65ef8c451e194964f939609f54e6
1 /*
2 * Creation Date: <2004/08/28 18:38:22 greg>
3 * Time-stamp: <2004/08/28 18:38:22 greg>
5 * <methods.c>
7 * Misc device node methods
9 * Copyright (C) 2004 Greg Watson
11 * Based on MOL specific code which is
13 * Copyright (C) 2003, 2004 Samuel Rydh (samuel@ibrium.se)
15 * This program is free software; you can redistribute it and/or
16 * modify it under the terms of the GNU General Public License
17 * version 2
21 #include "config.h"
22 #include "libopenbios/bindings.h"
23 #include "libc/string.h"
24 #include "pearpc/pearpc.h"
25 #include "libopenbios/ofmem.h"
27 /************************************************************************/
28 /* RTAS (run-time abstraction services) */
29 /************************************************************************/
31 #ifdef CONFIG_RTAS
32 DECLARE_NODE( rtas, INSTALL_OPEN, 0, "+/rtas" );
34 /* ( physbase -- rtas_callback ) */
35 static void
36 rtas_instantiate( void )
38 int physbase = POP();
39 int s=0x1000, size = (int)of_rtas_end - (int)of_rtas_start;
40 unsigned long virt;
42 while( s < size )
43 s += 0x1000;
44 virt = ofmem_claim_virt( 0, s, 0x1000 );
45 ofmem_map( physbase, virt, s, -1 );
46 memcpy( (char*)virt, of_rtas_start, size );
48 printk("RTAS instantiated at %08x\n", physbase );
49 flush_icache_range( (char*)virt, (char*)virt + size );
51 PUSH( physbase );
54 NODE_METHODS( rtas ) = {
55 { "instantiate", rtas_instantiate },
56 { "instantiate-rtas", rtas_instantiate },
58 #endif
61 /************************************************************************/
62 /* stdout */
63 /************************************************************************/
65 DECLARE_NODE( video_stdout, INSTALL_OPEN, 0, "Tdisplay" );
67 /* ( addr len -- actual ) */
68 static void
69 stdout_write( void )
71 int len = POP();
72 char *addr = (char*)POP();
74 printk( "%s", s );
75 //vfd_draw_str( s );
76 console_draw_fstr(addr, len);
78 PUSH( len );
81 NODE_METHODS( video_stdout ) = {
82 { "write", stdout_write },
86 /************************************************************************/
87 /* tty */
88 /************************************************************************/
90 DECLARE_NODE( tty, INSTALL_OPEN, 0, "/packages/terminal-emulator" );
92 /* ( addr len -- actual ) */
93 static void
94 tty_read( void )
96 int ch, len = POP();
97 char *p = (char*)POP();
98 int ret=0;
100 if( len > 0 ) {
101 ret = 1;
102 ch = getchar();
103 if( ch >= 0 ) {
104 *p = ch;
105 } else {
106 ret = 0;
109 PUSH( ret );
112 /* ( addr len -- actual ) */
113 static void
114 tty_write( void )
116 int i, len = POP();
117 char *p = (char*)POP();
118 for( i=0; i<len; i++ )
119 putchar( *p++ );
120 RET( len );
123 NODE_METHODS( tty ) = {
124 { "read", tty_read },
125 { "write", tty_write },
128 /************************************************************************/
129 /* client interface 'quiesce' */
130 /************************************************************************/
132 DECLARE_NODE( ciface, 0, 0, "/packages/client-iface" );
134 /* ( -- ) */
135 static void
136 ciface_quiesce( unsigned long args[], unsigned long ret[] )
138 #if 0
139 unsigned long msr;
140 /* This seems to be the correct thing to do - but I'm not sure */
141 asm volatile("mfmsr %0" : "=r" (msr) : );
142 msr &= ~(MSR_IR | MSR_DR);
143 asm volatile("mtmsr %0" :: "r" (msr) );
144 #endif
145 printk("=============================================================\n\n");
148 /* ( -- ms ) */
149 static void
150 ciface_milliseconds( unsigned long args[], unsigned long ret[] )
152 extern unsigned long get_timer_freq();
153 static unsigned long mticks=0, usecs=0;
154 unsigned long t;
156 asm volatile("mftb %0" : "=r" (t) : );
157 if( mticks )
158 usecs += get_timer_freq() / 1000000 * ( t-mticks );
159 mticks = t;
161 PUSH( usecs/1000 );
165 NODE_METHODS( ciface ) = {
166 { "quiesce", ciface_quiesce },
167 { "milliseconds", ciface_milliseconds },
171 /************************************************************************/
172 /* MMU/memory methods */
173 /************************************************************************/
175 DECLARE_NODE( memory, INSTALL_OPEN, 0, "/memory" );
176 DECLARE_NODE( mmu, INSTALL_OPEN, 0, "/cpu@0" );
177 DECLARE_NODE( mmu_ciface, 0, 0, "/packages/client-iface" );
180 /* ( phys size align --- base ) */
181 static void
182 mem_claim( void )
184 ucell align = POP();
185 ucell size = POP();
186 ucell phys = POP();
187 ucell ret = ofmem_claim_phys( phys, size, align );
189 if( ret == (ucell)-1 ) {
190 printk("MEM: claim failure\n");
191 throw( -13 );
192 return;
194 PUSH( ret );
197 /* ( phys size --- ) */
198 static void
199 mem_release( void )
201 POP(); POP();
204 /* ( phys size align --- base ) */
205 static void
206 mmu_claim( void )
208 ucell align = POP();
209 ucell size = POP();
210 ucell phys = POP();
211 ucell ret = ofmem_claim_virt( phys, size, align );
213 if( ret == -1 ) {
214 printk("MMU: CLAIM failure\n");
215 throw( -13 );
216 return;
218 PUSH( ret );
221 /* ( phys size --- ) */
222 static void
223 mmu_release( void )
225 POP(); POP();
228 /* ( phys virt size mode -- [ret???] ) */
229 static void
230 mmu_map( void )
232 ucell mode = POP();
233 ucell size = POP();
234 ucell virt = POP();
235 ucell phys = POP();
236 ucell ret;
238 /* printk("mmu_map: %x %x %x %x\n", phys, virt, size, mode ); */
239 ret = ofmem_map( phys, virt, size, mode );
241 if( ret ) {
242 printk("MMU: map failure\n");
243 throw( -13 );
244 return;
248 /* ( virt size -- ) */
249 static void
250 mmu_unmap( void )
252 POP(); POP();
255 /* ( virt -- false | phys mode true ) */
256 static void
257 mmu_translate( void )
259 ucell mode;
260 ucell virt = POP();
261 ucell phys = ofmem_translate( virt, &mode );
263 if( phys == -1 ) {
264 PUSH( 0 );
265 } else {
266 PUSH( phys );
267 PUSH( mode );
268 PUSH( -1 );
272 /* ( virt size align -- baseaddr|-1 ) */
273 static void
274 ciface_claim( void )
276 ucell align = POP();
277 ucell size = POP();
278 ucell virt = POP();
279 ucell ret = ofmem_claim( virt, size, align );
281 /* printk("ciface_claim: %08x %08x %x\n", virt, size, align ); */
282 PUSH( ret );
285 /* ( virt size -- ) */
286 static void
287 ciface_release( void )
289 POP();
290 POP();
294 NODE_METHODS( memory ) = {
295 { "claim", mem_claim },
296 { "release", mem_release },
299 NODE_METHODS( mmu ) = {
300 { "claim", mmu_claim },
301 { "release", mmu_release },
302 { "map", mmu_map },
303 { "unmap", mmu_unmap },
304 { "translate", mmu_translate },
307 NODE_METHODS( mmu_ciface ) = {
308 { "cif-claim", ciface_claim },
309 { "cif-release", ciface_release },
313 /************************************************************************/
314 /* init */
315 /************************************************************************/
317 void
318 node_methods_init( void )
320 #ifdef CONFIG_RTAS
321 REGISTER_NODE( rtas );
322 #endif
323 REGISTER_NODE( video_stdout );
324 REGISTER_NODE( ciface );
325 REGISTER_NODE( memory );
326 REGISTER_NODE( mmu );
327 REGISTER_NODE( mmu_ciface );
328 REGISTER_NODE( tty );