73a671baae26d526b3cf152401fc3f88c6f588ce
[virt-mem.git] / dmesg / virt_dmesg.ml
1 (* Memory info for virtual domains.
2    (C) Copyright 2008 Richard W.M. Jones, Red Hat Inc.
3    http://libvirt.org/
4
5    This program is free software; you can redistribute it and/or modify
6    it under the terms of the GNU General Public License as published by
7    the Free Software Foundation; either version 2 of the License, or
8    (at your option) any later version.
9
10    This program is distributed in the hope that it will be useful,
11    but WITHOUT ANY WARRANTY; without even the implied warranty of
12    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13    GNU General Public License for more details.
14
15    You should have received a copy of the GNU General Public License
16    along with this program; if not, write to the Free Software
17    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18  *)
19
20 open Printf
21
22 open Virt_mem_gettext.Gettext
23 open Virt_mem_utils
24 open Virt_mem_types
25 open Virt_mem_mmap
26
27 let run debug ({ domname = domname; mem = mem }, ksymmap, _) =
28   try
29     (* I don't know why but this symbol doesn't exist in 2.6.9
30      * even in kallsyms.  Hence this won't work with that kernel.
31      * It's possible we can fall back to memory scanning. XXX
32      *)
33     let log_buf = Ksymmap.find "log_buf" ksymmap in
34     let log_buf = follow_pointer mem log_buf in
35     let log_buf_len = Ksymmap.find "log_buf_len" ksymmap in
36     let log_buf_len = Int64.of_int32 (get_C_int mem log_buf_len) in
37     (*  let log_start = Ksymmap.find "log_start" ksymmap in
38         let log_start = get_C_long mem log_start in *)
39     let log_end = Ksymmap.find "log_end" ksymmap in
40     let log_end = get_C_long mem log_end in
41     (*  let con_start = Ksymmap.find "con_start" ksymmap in
42         let con_start = get_C_long mem con_start in *)
43     let logged_chars = Ksymmap.find "logged_chars" ksymmap in
44     let logged_chars = get_C_long mem logged_chars in
45
46     (* This is basically the same algorithm from printk.c:do_syslog
47      * type=3, translated into OCaml.  Unlike the kernel version
48      * however we don't copy the buffer backwards.
49      *)
50     let get_log_buf idx =
51       let addr = log_buf +^ (idx &^ (log_buf_len -^ 1L)) in
52       Char.chr (get_byte mem addr)
53     in
54
55     let count = log_buf_len in
56     let count = if count > logged_chars then logged_chars else count in
57     let limit = log_end in
58
59     let rec loop i =
60       if i >= 0L then (
61         let j = limit-^1L-^i in
62         if j +^ log_buf_len >= log_end then (
63           let c = get_log_buf j in
64           printf "%c" c;
65           loop (i-^1L)
66         )
67       )
68     in
69     loop (count-^1L)
70   with
71     Not_found ->
72       eprintf (f_"%s: could not find kernel log buffer in kernel image\n")
73         domname
74
75 let summary = s_"display kernel messages"
76 let description = s_"\
77 virt-dmesg prints the kernel messages for virtual machines running
78 under libvirt.  The output is similar to the ordinary dmesg command
79 run inside the virtual machine."
80
81 let () = Virt_mem.register "dmesg" summary description ~run