summaryrefslogtreecommitdiffstats
path: root/qemu/tests/tcg/hello-mips.c
blob: f8256730dd108310fb13c99ae93cce73a6d61ea0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
/*
* MIPS o32 Linux syscall example
*
* http://www.linux-mips.org/wiki/RISC/os
* http://www.linux-mips.org/wiki/MIPSABIHistory
* http://www.linux.com/howtos/Assembly-HOWTO/mips.shtml
*
* mipsel-linux-gcc -nostdlib -mno-abicalls -fno-PIC -mabi=32 \
*                  -O2 -static -o hello-mips hello-mips.c
*
*/
#define __NR_SYSCALL_BASE	4000
#define __NR_exit			(__NR_SYSCALL_BASE+  1)
#define __NR_write			(__NR_SYSCALL_BASE+  4)

static inline void exit1(int status)
{
    register unsigned long __a0 asm("$4") = (unsigned long) status;

    __asm__ __volatile__ (
        "	.set push	\n"
        "	.set noreorder	\n"
        "	li	$2, %0	\n"
        "	syscall		\n"
        "	.set pop	"
        :
	: "i" (__NR_exit), "r" (__a0)
	: "$2", "$8", "$9", "$10", "$11", "$12", "$13", "$14", "$15", "$24",
	  "memory");
}

static inline int write(int fd, const char *buf, int len)
{
    register unsigned long __a0 asm("$4") = (unsigned long) fd;
    register unsigned long __a1 asm("$5") = (unsigned long) buf;
    register unsigned long __a2 asm("$6") = (unsigned long) len;
    register unsigned long __a3 asm("$7");
    unsigned long __v0;

    __asm__ __volatile__ (
        "	.set push	\n"
        "	.set noreorder	\n"
        "	li	$2, %2	\n"
        "	syscall		\n"
        "	move	%0, $2	\n"
        "	.set pop	"
        : "=r" (__v0), "=r" (__a3)
        : "i" (__NR_write), "r" (__a0), "r" (__a1), "r" (__a2)
	: "$2", "$8", "$9", "$10", "$11", "$12", "$13", "$14", "$15", "$24",
	  "memory");

/*    if (__a3 == 0) */
        return (int) __v0;
/*
    errno = __v0;
    return -1;
 */
}

void __start(void)
{
    write (1, "Hello, World!\n", 14);
    exit1 (42);
}