00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025 #define _GNU_SOURCE
00026 #include <unistd.h>
00027 #include <fcntl.h>
00028 #include <string.h>
00029 #include <sys/mman.h>
00030 #include <sys/stat.h>
00031 #include <arpa/inet.h>
00032 #include <yyast/yyast.h>
00033
00034 typedef union {
00035 int64_t i;
00036 uint64_t u;
00037 double d;
00038 char c[8];
00039 } type64_t;
00040
00041 void indent(unsigned int level)
00042 {
00043 unsigned int i;
00044
00045 for (i = 0; i < level; i++) {
00046 fprintf(stdout, " ");
00047 }
00048 }
00049
00050 void print_name(ya_name_t name)
00051 {
00052 fprintf(stdout, "'%c%c%c%c%c%c%c%c'",
00053 (char)(name >> 56) & 0xff,
00054 (char)(name >> 48) & 0xff,
00055 (char)(name >> 40) & 0xff,
00056 (char)(name >> 32) & 0xff,
00057 (char)(name >> 24) & 0xff,
00058 (char)(name >> 16) & 0xff,
00059 (char)(name >> 8) & 0xff,
00060 (char)(name ) & 0xff
00061 );
00062 }
00063
00064 off_t node_decode(char *buf, size_t buf_size, unsigned int level)
00065 {
00066 if (buf_size < sizeof (ya_node_t)) {
00067 fprintf(stdout, "!error size to small to decode header.\n");
00068 exit(1);
00069 }
00070
00071 ya_node_t *node = (ya_node_t *)buf;
00072 ya_name_t name = ntohll(node->name);
00073 ya_type_t type = node->type;
00074 ya_position_t position = {.file = ntohl(node->position.file), .line = ntohl(node->position.line), .column = ntohl(node->position.column)};
00075 size_t inner_size = ntohll(node->size) - sizeof (ya_node_t);
00076 off_t inner_offset = 0;
00077 type64_t t64;
00078 char *s;
00079
00080 if (inner_size > buf_size) {
00081
00082 fprintf(stdout, "!error inner_size (%llu) larger than buffer_size (%llu)\n",
00083 (unsigned long long)inner_size,
00084 (unsigned long long)buf_size
00085 );
00086 exit(1);
00087 }
00088
00089 if (position.file != UINT32_MAX) {
00090 fprintf(stdout, "%2lu:%4lu:%3lu", (long)position.file, (long)position.line, (long)position.column);
00091 } else {
00092 fprintf(stdout, " ");
00093 }
00094
00095 fprintf(stdout, " (%8llu)", (unsigned long long)inner_size);
00096
00097 indent(level);
00098 fprintf(stdout, " ");
00099 print_name(name);
00100
00101 switch (type) {
00102 case YA_NODE_TYPE_POSITIVE_INTEGER:
00103 if (inner_size == sizeof (t64)) {
00104 memcpy(t64.c, node->data, sizeof (t64));
00105 t64.u = ntohll(t64.u);
00106 fprintf(stdout, " +%lli\n", (long long int)t64.i);
00107 } else {
00108 fprintf(stdout, " +i%i\n", (int)inner_size);
00109 }
00110 break;
00111 case YA_NODE_TYPE_NEGATIVE_INTEGER:
00112 if (inner_size == sizeof (t64)) {
00113 memcpy(t64.c, node->data, sizeof (t64));
00114 t64.u = ntohll(t64.u);
00115 fprintf(stdout, " -%llu\n", (long long unsigned)t64.u);
00116 } else {
00117 fprintf(stdout, " -i%i\n", (int)inner_size);
00118 }
00119 break;
00120 case YA_NODE_TYPE_BINARY_FLOAT:
00121 if (inner_size == sizeof (t64)) {
00122 memcpy(t64.c, node->data, sizeof (t64));
00123 t64.u = ntohll(t64.u);
00124 fprintf(stdout, " %lf\n", t64.d);
00125 } else {
00126 fprintf(stdout, " bf%i\n", (int)inner_size);
00127 }
00128 break;
00129 case YA_NODE_TYPE_TEXT:
00130 s = strndup(node->data, inner_size);
00131 fprintf(stdout, " \"%s\"\n", s);
00132 free(s);
00133 break;
00134 case YA_NODE_TYPE_NULL:
00135 fprintf(stdout, " pass\n");
00136 break;
00137 case YA_NODE_TYPE_BRANCH:
00138
00139 fprintf(stdout, "\n");
00140 while (inner_offset < inner_size) {
00141
00142 inner_offset+= node_decode(&node->data[inner_offset], inner_size - inner_offset, level + 1);
00143 }
00144 break;
00145 default:
00146 fprintf(stderr, " *unknown*\n");
00147 break;
00148 }
00149
00150
00151 return ya_align64(ntohll(node->size));
00152 }
00153
00154 int main(int argc, char *argv[])
00155 {
00156 int fd;
00157 struct stat fd_st;
00158 char *buf;
00159 size_t buf_size;
00160
00161 if (argc != 2) {
00162 fprintf(stderr, "Expect 1 filename as argument.\n");
00163 exit(1);
00164 }
00165
00166 if ((fd = open(argv[1], O_RDONLY)) == -1) {
00167 perror("Failed to open file.");
00168 exit(1);
00169 }
00170
00171 if (fstat(fd, &fd_st) == -1) {
00172 perror("Could not stat the file.");
00173 exit(1);
00174 }
00175 buf_size = fd_st.st_size;
00176
00177 if ((buf = mmap(0, buf_size, PROT_READ, MAP_FILE | MAP_SHARED, fd, 0)) == MAP_FAILED) {
00178 perror("Could not map the file in memory.");
00179 exit(1);
00180 }
00181
00182
00183 (void)node_decode(buf, buf_size, 0);
00184
00185 if (munmap(buf, buf_size) == -1) {
00186 perror("Could not unmap the file from memory.");
00187 exit(1);
00188 }
00189
00190 if (close(fd) == -1) {
00191 perror("Close the file.");
00192 exit(1);
00193 }
00194 }
00195