File:  [Qemu by Fabrice Bellard] / qemu / roms / openbios / arch / ppc / mol / methods.c
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs
Tue Apr 24 19:19:39 2018 UTC (8 years, 1 month ago) by root
Branches: qemu, MAIN
CVS tags: qemu1101, qemu1001, HEAD
qemu 1.0.1

/*
 *   Creation Date: <2003/10/18 13:24:29 samuel>
 *   Time-stamp: <2004/03/27 02:00:30 samuel>
 *
 *	<methods.c>
 *
 *	Misc device node methods
 *
 *   Copyright (C) 2003, 2004 Samuel Rydh ([email protected])
 *
 *   This program is free software; you can redistribute it and/or
 *   modify it under the terms of the GNU General Public License
 *   version 2
 *
 */

#include "config.h"
#include "libopenbios/bindings.h"
#include "libc/string.h"
#include "mol/mol.h"
#include "libopenbios/ofmem.h"
#include "mol/prom.h"
#include "osi_calls.h"
#include "kbd_sh.h"

/************************************************************************/
/*	Power Management						*/
/************************************************************************/

DECLARE_NODE( powermgt, INSTALL_OPEN, 0, "/pci/pci-bridge/mac-io/power-mgt" );

/* ( -- ) */
static void
set_hybernot_flag( void )
{
}

NODE_METHODS( powermgt ) = {
	{ "set-hybernot-flag",	set_hybernot_flag	},
};


/************************************************************************/
/*	RTAS (run-time abstraction services)				*/
/************************************************************************/

DECLARE_NODE( rtas, INSTALL_OPEN, 0, "+/rtas" );

/* ( physbase -- rtas_callback ) */
static void
rtas_instantiate( void )
{
	int physbase = POP();
	int s=0x1000, size = (int)of_rtas_end - (int)of_rtas_start;
	unsigned long virt;

	while( s < size )
		s += 0x1000;
	virt = ofmem_claim_virt( 0, s, 0x1000 );
	ofmem_map( physbase, virt, s, -1 );
	memcpy( (char*)virt, of_rtas_start, size );

	printk("RTAS instantiated at %08x\n", physbase );
	flush_icache_range( (char*)virt, (char*)virt + size );

	PUSH( physbase );
}

NODE_METHODS( rtas ) = {
	{ "instantiate",	rtas_instantiate },
	{ "instantiate-rtas",	rtas_instantiate },
};



/************************************************************************/
/*	stdout								*/
/************************************************************************/

DECLARE_NODE( video_stdout, INSTALL_OPEN, 0, "Tdisplay" );

/* ( addr len -- actual ) */
static void
stdout_write( void )
{
	int len = POP();
	char *addr = (char*)POP();

	/* printk( "%s", s ); */
        console_draw_fstr(addr, len);

	PUSH( len );
}

NODE_METHODS( video_stdout ) = {
	{ "write",	stdout_write	},
};


/************************************************************************/
/*	tty								*/
/************************************************************************/

DECLARE_NODE( tty, INSTALL_OPEN, 0, "+/mol/mol-tty" );

/* ( addr len -- actual ) */
static void
tty_read( void )
{
	int ch, len = POP();
	char *p = (char*)POP();
	int ret=0;

	if( len > 0 ) {
		ret = 1;
		ch = OSI_TTYGetc();
		if( ch >= 0 ) {
			*p = ch;
		} else {
			ret = 0;
			OSI_USleep(1);
		}
	}
	PUSH( ret );
}

/* ( addr len -- actual ) */
static void
tty_write( void )
{
	int i, len = POP();
	char *p = (char*)POP();
	for( i=0; i<len; i++ )
		OSI_TTYPutc( *p++ );
	RET( len );
}

NODE_METHODS( tty ) = {
	{ "read",	tty_read	},
	{ "write",	tty_write	},
};


/************************************************************************/
/*	keyboard							*/
/************************************************************************/

typedef struct {
	int	cntrl;
	int	shift;
	int	meta;
	int	alt;
	int	save_key;
	char 	keytable[32];
} kbd_state_t;

static const unsigned char adb_ascii_table[128] =
	/* 0x00 */	"asdfhgzxcv`bqwer"
	/* 0x10 */	"yt123465=97-80]o"
	/* 0x20 */	"u[ip\nlj'k;\\,/nm."
	/* 0x30 */	"\t <\b \e          "
	/* 0x40 */	" . * +     /  - "
	/* 0x50 */	" =01234567 89   "
	/* 0x60 */	"                "
	/* 0x70 */	"                ";

static const unsigned char adb_shift_table[128] =
	/* 0x00 */	"ASDFHGZXCV~BQWER"
	/* 0x10 */	"YT!@#$^%+(&_*)}O"
	/* 0x20 */	"U{IP\nLJ\"K:|<?NM>"
	/* 0x30 */	"\t <\b \e          "
	/* 0x40 */	" . * +     /  - "
	/* 0x50 */	" =01234567 89   "
	/* 0x60 */	"                "
	/* 0x70 */	"                ";

DECLARE_NODE( kbd, INSTALL_OPEN, sizeof(kbd_state_t),
      "/psuedo-hid/keyboard",
      "/mol/mol-keyboard",
      "/mol/keyboard"
);

/* ( -- keymap ) (?) */
/* should return a pointer to an array with 32 bytes (256 bits) */
static void
kbd_get_key_map( kbd_state_t *ks )
{
	/* printk("met_kbd_get_key_map\n"); */

	/* keytable[5] = 0x40; */
	PUSH( (int)ks->keytable );
}

/* ( buf len --- actlen ) */
static void
kbd_read( kbd_state_t *ks )
{
	int ret=0, len = POP();
	char *p = (char*)POP();
	int key;

	if( !p || !len ) {
		PUSH( -1 );
		return;
	}

	if( ks->save_key ) {
		*p = ks->save_key;
		ks->save_key = 0;
		RET( 1 );
	}
	OSI_USleep(1);	/* be nice */

	for( ; (key=OSI_GetAdbKey()) >= 0 ; ) {
		int code = (key & 0x7f);
		int down = !(key & 0x80);

		if( code == 0x36 /* ctrl */ ) {
			ks->cntrl = down;
			continue;
		}
		if( code == 0x38 /* shift */ || code == 0x7b) {
			ks->shift = down;
			continue;
		}
		if( code == 0x37 /* command */ ) {
			ks->meta = down;
			continue;
		}
		if( code == 0x3a /* alt */ ) {
			ks->alt = down;
			continue;
		}
		if( !down )
			continue;

		ret = 1;
		if( ks->shift )
			key = adb_shift_table[ key & 0x7f ];
		else
			key = adb_ascii_table[ key & 0x7f ];

		if( ks->meta ) {
			ks->save_key = key;
			key = 27;
		} else if( ks->cntrl ) {
			key = key - 'a' + 1;
		}
		*p = key;
		if( !*p )
			*p = 'x';
		break;
	}
	PUSH( ret );
}

NODE_METHODS( kbd ) = {
	{ "read",		kbd_read		},
	{ "get-key-map",	kbd_get_key_map		},
};


/************************************************************************/
/*	client interface 'quiesce'					*/
/************************************************************************/

DECLARE_NODE( ciface, 0, 0, "/packages/client-iface" );

/* ( -- ) */
static void
ciface_quiesce( unsigned long args[], unsigned long ret[] )
{
#if 0
	unsigned long msr;
	/* This seems to be the correct thing to do - but I'm not sure */
	asm volatile("mfmsr %0" : "=r" (msr) : );
	msr &= ~(MSR_IR | MSR_DR);
	asm volatile("mtmsr %0" :: "r" (msr) );
#endif
	printk("=============================================================\n\n");
	prom_close();

	OSI_KbdCntrl( kKbdCntrlSuspend );
}

/* ( -- ms ) */
static void
ciface_milliseconds( unsigned long args[], unsigned long ret[] )
{
	static unsigned long mticks=0, usecs=0;
	unsigned long t;

	asm volatile("mftb %0" : "=r" (t) : );
	if( mticks )
		usecs += OSI_MticksToUsecs( t-mticks );
	mticks = t;

	PUSH( usecs/1000 );
}


NODE_METHODS( ciface ) = {
	{ "quiesce",		ciface_quiesce		},
	{ "milliseconds",	ciface_milliseconds	},
};


/************************************************************************/
/*	MMU/memory methods						*/
/************************************************************************/

DECLARE_NODE( memory, INSTALL_OPEN, 0, "/memory" );
DECLARE_NODE( mmu, INSTALL_OPEN, 0, "/cpus/@0" );
DECLARE_NODE( mmu_ciface, 0, 0, "/packages/client-iface" );


/* ( phys size align --- base ) */
static void
mem_claim( void )
{
	ucell align = POP();
	ucell size = POP();
	ucell phys = POP();
	ucell ret = ofmem_claim_phys( phys, size, align );

	if( ret == -1 ) {
		printk("MEM: claim failure\n");
		throw( -13 );
		return;
	}
	PUSH( ret );
}

/* ( phys size --- ) */
static void
mem_release( void )
{
	POP(); POP();
}

/* ( phys size align --- base ) */
static void
mmu_claim( void )
{
	ucell align = POP();
	ucell size = POP();
	ucell phys = POP();
	ucell ret = ofmem_claim_virt( phys, size, align );

	if( ret == -1 ) {
		printk("MMU: CLAIM failure\n");
		throw( -13 );
		return;
	}
	PUSH( ret );
}

/* ( phys size --- ) */
static void
mmu_release( void )
{
	POP(); POP();
}

/* ( phys virt size mode -- [ret???] ) */
static void
mmu_map( void )
{
	ucell mode = POP();
	ucell size = POP();
	ucell virt = POP();
	ucell phys = POP();
	ucell ret;

	/* printk("mmu_map: %x %x %x %x\n", phys, virt, size, mode ); */
	ret = ofmem_map( phys, virt, size, mode );

	if( ret ) {
		printk("MMU: map failure\n");
		throw( -13 );
		return;
	}
}

/* ( virt size -- ) */
static void
mmu_unmap( void )
{
	POP(); POP();
}

/* ( virt -- false | phys mode true ) */
static void
mmu_translate( void )
{
	ucell mode;
	ucell virt = POP();
	ucell phys = ofmem_translate( virt, &mode );

	if( phys == -1 ) {
		PUSH( 0 );
	} else {
		PUSH( phys );
		PUSH( mode );
		PUSH( -1 );
	}
}

/* ( virt size align -- baseaddr|-1 ) */
static void
ciface_claim( void )
{
	ucell align = POP();
	ucell size = POP();
	ucell virt = POP();
	ucell ret = ofmem_claim( virt, size, align );

	/* printk("ciface_claim: %08x %08x %x\n", virt, size, align ); */
	PUSH( ret );
}

/* ( virt size -- ) */
static void
ciface_release( void )
{
	POP();
	POP();
}


NODE_METHODS( memory ) = {
	{ "claim",		mem_claim		},
	{ "release",		mem_release		},
};

NODE_METHODS( mmu ) = {
	{ "claim",		mmu_claim		},
	{ "release",		mmu_release		},
	{ "map",		mmu_map			},
	{ "unmap",		mmu_unmap		},
	{ "translate",		mmu_translate		},
};

NODE_METHODS( mmu_ciface ) = {
	{ "cif-claim",		ciface_claim		},
	{ "cif-release",	ciface_release		},
};


/************************************************************************/
/*	init								*/
/************************************************************************/

void
node_methods_init( void )
{
	REGISTER_NODE( rtas );
	REGISTER_NODE( powermgt );
	REGISTER_NODE( kbd );
	REGISTER_NODE( video_stdout );
	REGISTER_NODE( ciface );
	REGISTER_NODE( memory );
	REGISTER_NODE( mmu );
	REGISTER_NODE( mmu_ciface );

	if( OSI_CallAvailable(OSI_TTY_GETC) )
		REGISTER_NODE( tty );

	OSI_KbdCntrl( kKbdCntrlActivate );
}

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.