]> Joshua Wise's Git repositories - netwatch.git/blob - net/http/fs.c
Improve correctness of backtrace library.
[netwatch.git] / net / http / fs.c
1 /*
2  * Copyright (c) 2001-2003 Swedish Institute of Computer Science.
3  * All rights reserved. 
4  * 
5  * Redistribution and use in source and binary forms, with or without modification, 
6  * are permitted provided that the following conditions are met:
7  *
8  * 1. Redistributions of source code must retain the above copyright notice,
9  *    this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright notice,
11  *    this list of conditions and the following disclaimer in the documentation
12  *    and/or other materials provided with the distribution.
13  * 3. The name of the author may not be used to endorse or promote products
14  *    derived from this software without specific prior written permission. 
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED 
17  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF 
18  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT 
19  * SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 
20  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT 
21  * OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 
22  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 
23  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING 
24  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY 
25  * OF SUCH DAMAGE.
26  *
27  * This file is part of the lwIP TCP/IP stack.
28  * 
29  * Author: Adam Dunkels <adam@sics.se>
30  *
31  */
32 #include "lwip/def.h"
33 #include "fs.h"
34 #include "fsdata.h"
35 #include "fsdata.c"
36 #include <io.h>
37 #include <minilib.h>
38 #include <paging.h>
39 #include <demap.h>
40 #include <output.h>
41 #include <state.h>
42
43 static char http_output_buffer[1280];
44
45 /*-----------------------------------------------------------------------------------*/
46
47 void handle_regs(struct fs_file *file)
48 {
49   int i;
50   int len;
51
52   len = snprintf(http_output_buffer, sizeof(http_output_buffer), "<html><pre>");
53
54   for (i = 0; i < state_num_regs(); i++) {
55     len += state_dump_reg(http_output_buffer + len, sizeof(http_output_buffer) - len, i);
56   }
57
58   file->len = len;
59   file->data = http_output_buffer;
60 }
61
62 #define LEFT (sizeof(http_output_buffer) - len)
63
64 void handle_backtrace(struct fs_file *file)
65 {
66   int i = 10;
67   int len;
68   void *pebp, *peip;
69   uint64_t bp, next;
70
71   int longmode = (get_operating_mode() == LONG_64BIT);
72
73   char * buf = http_output_buffer;
74   
75   strcpy(buf, "<html><head><title>Backtrace</title></head><body><tt><pre>");
76   len = strlen(buf);
77
78   bp = state_get_reg(longmode ? STATE_REG_RIP : STATE_REG_EIP);
79
80   if (longmode)
81     len += snprintf(buf + len, LEFT, "0x%08x%08x, from\n", (uint32_t)(bp >> 32), (uint32_t)bp);
82   else
83     len += snprintf(buf + len, LEFT, "0x%08x, from\n", (uint32_t)bp);
84   
85   bp = state_get_reg(longmode ? STATE_REG_RBP : STATE_REG_EBP);
86   
87
88   outputf("Backtrace:");
89   while ((peip = demap(bp+(longmode?8:4))) != 0x0 && i--)
90   {
91     if (longmode) {
92       uint64_t rip;
93       rip = *(uint64_t *)peip;
94       len += snprintf(buf + len, LEFT, "0x%08x%08x, from\n", (uint32_t)(rip >> 32), (uint32_t)rip);
95     } else {
96       len += snprintf(buf + len, LEFT, "0x%08x, from\n", *(uint32_t *)peip);
97     }
98     outputf("   EIP: %08x", *(uint32_t *)peip);
99
100     pebp = demap(bp);
101     if (!pebp || (bp & 3))
102     {
103       len += snprintf(buf + len, LEFT, "&lt;unreadable frame&gt;\n");
104       break;
105     }
106     outputf("   *EBP: %08x", *(uint32_t *)pebp);
107
108     if (longmode)
109       next = *(uint64_t *)pebp;
110     else
111       next = *(uint32_t *)pebp;
112
113     if (bp >= next && next)
114     {
115       len += snprintf(buf + len, LEFT, "&lt;recursive frame&gt;\n");
116       break;
117     }
118
119     bp = next;
120   }
121
122   if (i == -1)
123     len += snprintf(buf + len, LEFT, "...\n");
124   else if (bp && !peip)
125     len += snprintf(buf + len, LEFT, "&lt;unreadable fp @ %08x&gt;\n", (uint32_t) bp);
126   else
127     len += snprintf(buf + len, LEFT, "&lt;root&gt;");
128
129   len += snprintf(buf + len, LEFT, "</pre></tt></body></html>");
130   
131   file->data = buf;
132   file->len = len;
133 }
134
135 void handle_reboot(struct fs_file *file)
136 {
137   outb(0xCF9, 0x4);
138   file->data = "So long!";
139   file->len = 8;
140 }
141
142
143 /*-----------------------------------------------------------------------------------*/
144 int
145 fs_open(const char *name, struct fs_file *file)
146 {
147   const struct fsdata_file *f;
148   
149   /* /registers.html is CGI */
150   if (!strcmp(name, "/registers.html"))
151   {
152     handle_regs(file);
153     return 1;
154   }
155   if (!strcmp(name, "/backtrace.html"))
156   {
157     handle_backtrace(file);
158     return 1;
159   }
160   if (!strcmp(name, "/reboot"))
161   {
162     handle_reboot(file);
163     return 1;
164   }
165
166   for(f = FS_ROOT;
167       f != NULL;
168       f = f->next) {
169     if (!strcmp(name, (const char*)f->name)) {
170       file->data = f->data;
171       file->len = f->len-1;
172       return 1;
173     }
174   }
175   file->data = "You clown...";
176   file->len = 9;
177   return 0;
178 }
179 /*-----------------------------------------------------------------------------------*/
This page took 0.031928 seconds and 4 git commands to generate.