annotate uprogs/printf.c @ 0:ed10291ff195

first commit
author mir3636
date Sun, 06 Jan 2019 19:27:03 +0900
parents
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
ed10291ff195 first commit
mir3636
parents:
diff changeset
1 #include "types.h"
ed10291ff195 first commit
mir3636
parents:
diff changeset
2 #include "stat.h"
ed10291ff195 first commit
mir3636
parents:
diff changeset
3 #include "user.h"
ed10291ff195 first commit
mir3636
parents:
diff changeset
4
ed10291ff195 first commit
mir3636
parents:
diff changeset
5 static void
ed10291ff195 first commit
mir3636
parents:
diff changeset
6 putc(int fd, char c)
ed10291ff195 first commit
mir3636
parents:
diff changeset
7 {
ed10291ff195 first commit
mir3636
parents:
diff changeset
8 write(fd, &c, 1);
ed10291ff195 first commit
mir3636
parents:
diff changeset
9 }
ed10291ff195 first commit
mir3636
parents:
diff changeset
10
ed10291ff195 first commit
mir3636
parents:
diff changeset
11 u32 div(u32 n, u32 d) // long division
ed10291ff195 first commit
mir3636
parents:
diff changeset
12 {
ed10291ff195 first commit
mir3636
parents:
diff changeset
13 u32 q=0, r=0;
ed10291ff195 first commit
mir3636
parents:
diff changeset
14 int i;
ed10291ff195 first commit
mir3636
parents:
diff changeset
15
ed10291ff195 first commit
mir3636
parents:
diff changeset
16 for(i=31;i>=0;i--){
ed10291ff195 first commit
mir3636
parents:
diff changeset
17 r = r << 1;
ed10291ff195 first commit
mir3636
parents:
diff changeset
18 r = r | ((n >> i) & 1);
ed10291ff195 first commit
mir3636
parents:
diff changeset
19 if(r >= d) {
ed10291ff195 first commit
mir3636
parents:
diff changeset
20 r = r - d;
ed10291ff195 first commit
mir3636
parents:
diff changeset
21 q = q | (1 << i);
ed10291ff195 first commit
mir3636
parents:
diff changeset
22 }
ed10291ff195 first commit
mir3636
parents:
diff changeset
23 }
ed10291ff195 first commit
mir3636
parents:
diff changeset
24 return q;
ed10291ff195 first commit
mir3636
parents:
diff changeset
25 }
ed10291ff195 first commit
mir3636
parents:
diff changeset
26
ed10291ff195 first commit
mir3636
parents:
diff changeset
27 static void
ed10291ff195 first commit
mir3636
parents:
diff changeset
28 printint(int fd, int xx, int base, int sgn)
ed10291ff195 first commit
mir3636
parents:
diff changeset
29 {
ed10291ff195 first commit
mir3636
parents:
diff changeset
30 static char digits[] = "0123456789ABCDEF";
ed10291ff195 first commit
mir3636
parents:
diff changeset
31 char buf[16];
ed10291ff195 first commit
mir3636
parents:
diff changeset
32 int i, neg;
ed10291ff195 first commit
mir3636
parents:
diff changeset
33 uint x, y, b;
ed10291ff195 first commit
mir3636
parents:
diff changeset
34
ed10291ff195 first commit
mir3636
parents:
diff changeset
35 neg = 0;
ed10291ff195 first commit
mir3636
parents:
diff changeset
36 if(sgn && xx < 0){
ed10291ff195 first commit
mir3636
parents:
diff changeset
37 neg = 1;
ed10291ff195 first commit
mir3636
parents:
diff changeset
38 x = -xx;
ed10291ff195 first commit
mir3636
parents:
diff changeset
39 } else {
ed10291ff195 first commit
mir3636
parents:
diff changeset
40 x = xx;
ed10291ff195 first commit
mir3636
parents:
diff changeset
41 }
ed10291ff195 first commit
mir3636
parents:
diff changeset
42
ed10291ff195 first commit
mir3636
parents:
diff changeset
43 b = base;
ed10291ff195 first commit
mir3636
parents:
diff changeset
44 i = 0;
ed10291ff195 first commit
mir3636
parents:
diff changeset
45 do{
ed10291ff195 first commit
mir3636
parents:
diff changeset
46 y = div(x, b);
ed10291ff195 first commit
mir3636
parents:
diff changeset
47 buf[i++] = digits[x - y * b];
ed10291ff195 first commit
mir3636
parents:
diff changeset
48 }while((x = y) != 0);
ed10291ff195 first commit
mir3636
parents:
diff changeset
49 if(neg)
ed10291ff195 first commit
mir3636
parents:
diff changeset
50 buf[i++] = '-';
ed10291ff195 first commit
mir3636
parents:
diff changeset
51
ed10291ff195 first commit
mir3636
parents:
diff changeset
52 while(--i >= 0)
ed10291ff195 first commit
mir3636
parents:
diff changeset
53 putc(fd, buf[i]);
ed10291ff195 first commit
mir3636
parents:
diff changeset
54 }
ed10291ff195 first commit
mir3636
parents:
diff changeset
55
ed10291ff195 first commit
mir3636
parents:
diff changeset
56 // Print to the given fd. Only understands %d, %x, %p, %s.
ed10291ff195 first commit
mir3636
parents:
diff changeset
57 void
ed10291ff195 first commit
mir3636
parents:
diff changeset
58 printf(int fd, char *fmt, ...)
ed10291ff195 first commit
mir3636
parents:
diff changeset
59 {
ed10291ff195 first commit
mir3636
parents:
diff changeset
60 char *s;
ed10291ff195 first commit
mir3636
parents:
diff changeset
61 int c, i, state;
ed10291ff195 first commit
mir3636
parents:
diff changeset
62 uint *ap;
ed10291ff195 first commit
mir3636
parents:
diff changeset
63
ed10291ff195 first commit
mir3636
parents:
diff changeset
64 state = 0;
ed10291ff195 first commit
mir3636
parents:
diff changeset
65 ap = (uint*)(void*)&fmt + 1;
ed10291ff195 first commit
mir3636
parents:
diff changeset
66 for(i = 0; fmt[i]; i++){
ed10291ff195 first commit
mir3636
parents:
diff changeset
67 c = fmt[i] & 0xff;
ed10291ff195 first commit
mir3636
parents:
diff changeset
68 if(state == 0){
ed10291ff195 first commit
mir3636
parents:
diff changeset
69 if(c == '%'){
ed10291ff195 first commit
mir3636
parents:
diff changeset
70 state = '%';
ed10291ff195 first commit
mir3636
parents:
diff changeset
71 } else {
ed10291ff195 first commit
mir3636
parents:
diff changeset
72 putc(fd, c);
ed10291ff195 first commit
mir3636
parents:
diff changeset
73 }
ed10291ff195 first commit
mir3636
parents:
diff changeset
74 } else if(state == '%'){
ed10291ff195 first commit
mir3636
parents:
diff changeset
75 if(c == 'd'){
ed10291ff195 first commit
mir3636
parents:
diff changeset
76 printint(fd, *ap, 10, 1);
ed10291ff195 first commit
mir3636
parents:
diff changeset
77 ap++;
ed10291ff195 first commit
mir3636
parents:
diff changeset
78 } else if(c == 'x' || c == 'p'){
ed10291ff195 first commit
mir3636
parents:
diff changeset
79 printint(fd, *ap, 16, 0);
ed10291ff195 first commit
mir3636
parents:
diff changeset
80 ap++;
ed10291ff195 first commit
mir3636
parents:
diff changeset
81 } else if(c == 's'){
ed10291ff195 first commit
mir3636
parents:
diff changeset
82 s = (char*)*ap;
ed10291ff195 first commit
mir3636
parents:
diff changeset
83 ap++;
ed10291ff195 first commit
mir3636
parents:
diff changeset
84 if(s == 0)
ed10291ff195 first commit
mir3636
parents:
diff changeset
85 s = "(null)";
ed10291ff195 first commit
mir3636
parents:
diff changeset
86 while(*s != 0){
ed10291ff195 first commit
mir3636
parents:
diff changeset
87 putc(fd, *s);
ed10291ff195 first commit
mir3636
parents:
diff changeset
88 s++;
ed10291ff195 first commit
mir3636
parents:
diff changeset
89 }
ed10291ff195 first commit
mir3636
parents:
diff changeset
90 } else if(c == 'c'){
ed10291ff195 first commit
mir3636
parents:
diff changeset
91 putc(fd, *ap);
ed10291ff195 first commit
mir3636
parents:
diff changeset
92 ap++;
ed10291ff195 first commit
mir3636
parents:
diff changeset
93 } else if(c == '%'){
ed10291ff195 first commit
mir3636
parents:
diff changeset
94 putc(fd, c);
ed10291ff195 first commit
mir3636
parents:
diff changeset
95 } else {
ed10291ff195 first commit
mir3636
parents:
diff changeset
96 // Unknown % sequence. Print it to draw attention.
ed10291ff195 first commit
mir3636
parents:
diff changeset
97 putc(fd, '%');
ed10291ff195 first commit
mir3636
parents:
diff changeset
98 putc(fd, c);
ed10291ff195 first commit
mir3636
parents:
diff changeset
99 }
ed10291ff195 first commit
mir3636
parents:
diff changeset
100 state = 0;
ed10291ff195 first commit
mir3636
parents:
diff changeset
101 }
ed10291ff195 first commit
mir3636
parents:
diff changeset
102 }
ed10291ff195 first commit
mir3636
parents:
diff changeset
103 }