天天看點

【XEN學習筆記】XEN添加和調用Hypercall

sploving的文章,怕下次找不到,先轉來自己這裡

原文位址:http://blog.csdn.net/sploving/archive/2009/10/10/4651260.aspx

在linux系統中添加新的系統調用,一般需要三個步驟:

1.  注冊新的系統調用号

2. 更新系統調用表

3.  添加新函數

在xen中添加一個 hypercall,類似于在linux中添加一個系統調. 基本上也是上面幾個步驟。

現在舉個具體的例子:

比如我們要在xen中添加一個列印消息的hypercall,參數有一個,類型為char*, 代表我們要列印的消息. 函數原型為:

do_print_string(char* message)
           

1. 首先注冊一個hypercall調用号。

xen/include/public/xen.h

#define __HYPERVISOR_kexec_op             37//原有超級調用
#define __HYPERVISOR_print_string         38//新增超級調用号為38
           

2.更新系統調用表

/xen/arch/x86/x86_32/entry.S

ENTRY(hypercall_table)
        
   .long do_kexec_op
   .long do_print_string

ENTRY(hypercall_args_table)

   .byte 2 /* do_kexec_op          */
   .byte 1 /* do_print_string      */要寫在對應的位置
           

3. 定義函數頭檔案

/xen/include/asm-x86/hypercall.h

extern int
do_kexec(
    unsigned long op, unsigned arg1, XEN_GUEST_HANDLE(void) uarg);

extern int
do_print_string(char * message);//處理函數聲明
           

4.定義函數(函數定義在合适的檔案中,這個例子采用mm.c)

/xen/arch/x86/mm.c

int do_print_string(char * message)
{
    if(message)
      printk("The message is :\n%s\n", message);
    else printk("no message!\n");
    return 1;
}
           

OK.

重新編譯安裝

make dist

make install

重新制作img

mkinitrd -v -f initrd-2.6.18.8-xen.img 2.6.18.8-xen

重新開機

然後編輯一個.c file, 在使用者空間測試新的hypercall.如下:

(xen提供了privcmd這個驅動檔案,進而在3環(使用者空間)可以利用ioctl來調用hypercall)

#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <fcntl.h>
#include <string.h>
#include <xenctrl.h>
#include <xen/sys/privcmd.h>
int main(int argc, char *argv[])
{
    int fd, ret;
    char * message;
    if (argc != 2) {
        printf("please put one parameter!\n");
        return -1;
    }
    message = (char *) malloc(sizeof(char) * (strlen(argv[1])+1));
    strcpy(message, argv[1]);
    privcmd_hypercall_t hcall = {
        __HYPERVISOR_print_string,
        {message, 0, 0, 0, 0}
    };
    fd = open("/proc/xen/privcmd", O_RDWR);
    if (fd < 0) {
        perror("open");
        exit(1);
    } else
    printf("fd = %d\n", fd);
    ret = ioctl(fd, IOCTL_PRIVCMD_HYPERCALL, &hcall);
    printf("ret = %d\n", ret);
}
           

也可以采用加載子產品的形式,在核心空間之間調用hypercall來測試。

編譯該檔案,并測試如下:

gcc -o a hypercall_test.c
./a Hello!
           

檢視日志檔案,檢測是否新的hypercall安裝成功:(linux下的log 一般在/var/log/mesages,而xen下的日志采用xm dm指令檢視)

繼續閱讀