pencal.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. #include <u.h>
  2. #include <libc.h>
  3. #include <draw.h>
  4. int debug = 0;
  5. struct Cal {
  6. double scalex;
  7. double transx;
  8. double scaley;
  9. double transy;
  10. } cal;
  11. /* Calibrate the pen */
  12. Point pts[5];
  13. Point pens[5];
  14. Point up = { 0, -10};
  15. Point down = { 0, 10};
  16. Point left = {-10, 0};
  17. Point right = { 10, 0};
  18. Point
  19. pen2scr(Point p)
  20. {
  21. return Pt(p.x * cal.scalex / 65536.0 + cal.transx,
  22. p.y * cal.scaley / 65536.0 + cal.transy);
  23. }
  24. Point
  25. scr2pen(Point p)
  26. {
  27. return Pt((p.x - cal.transx) * 65536.0 / cal.scalex,
  28. (p.y + cal.transy) * 65536.0 / cal.scaley);
  29. }
  30. void
  31. cross(Point p) {
  32. draw(screen, screen->r, display->white, nil, ZP);
  33. line(screen, addpt(p, up), addpt(p, down), Endsquare, Endsquare, 0, display->black, ZP);
  34. line(screen, addpt(p, left), addpt(p, right), Endsquare, Endsquare, 0, display->black, ZP);
  35. flushimage(display, 1);
  36. }
  37. Point
  38. getmouse(int mouse) {
  39. static char buf[50];
  40. int x, y, b, n;
  41. Point p;
  42. n = 0;
  43. p.x = p.y = 0;
  44. do{
  45. if (read(mouse, buf, 48) != 48)
  46. sysfatal("read: %r");
  47. if (debug > 1) fprint(2, "%s\n", buf);
  48. if (buf[0] == 'r') {
  49. b = 1;
  50. continue;
  51. }
  52. if (buf[0] != 'm')
  53. sysfatal("mouse message: %s", buf);
  54. x = strtol(buf+1, nil, 10);
  55. y = strtol(buf+13, nil, 10);
  56. b = strtol(buf+25, nil, 10);
  57. p.x = (n * p.x + x)/(n+1);
  58. p.y = (n * p.y + y)/(n+1);
  59. n++;
  60. }while (b & 0x7);
  61. return p;
  62. }
  63. void
  64. main(int argc, char **argv) {
  65. int i, mouse, mousectl, wctl, ntries;
  66. Rectangle r, oldsize;
  67. ARGBEGIN{
  68. case 'd':
  69. debug = 1;
  70. break;
  71. }ARGEND;
  72. if((mouse = open("/dev/mouse", OREAD)) < 0
  73. && (mouse = open("#m/mouse", OREAD)) < 0)
  74. sysfatal("#m/mouse: %r");
  75. mousectl = open("#m/mousectl", ORDWR);
  76. if(mousectl < 0)
  77. sysfatal("#m/mousectl: %r");
  78. if(initdraw(nil, nil, "calibrate") < 0)
  79. sysfatal("initdraw: %r\n");
  80. wctl = -1;
  81. for(ntries = 0; ntries < 3; ntries++){
  82. r = insetrect(display->image->r, 2);
  83. if(wctl < 0)
  84. wctl = open("/dev/wctl", ORDWR);
  85. if(wctl >= 0){
  86. char buf[4*12+1];
  87. buf[48] = 0;
  88. oldsize = screen->r;
  89. if(fprint(wctl, "resize -r %R", r) <= 0)
  90. sysfatal("write /dev/wctl, %r");
  91. if(getwindow(display, Refbackup) < 0)
  92. sysfatal("getwindow");
  93. if(debug) fprint(2, "resize: %R\n", screen->r);
  94. if(read(mousectl, buf, 48) != 48)
  95. sysfatal("read mousectl: %r");
  96. if(debug > 1) fprint(2, "mousectl %s\n", buf);
  97. if(buf[0] != 'c')
  98. sysfatal("mousectl message: %s", buf);
  99. cal.scalex = strtol(buf+ 1, nil, 10);
  100. cal.scaley = strtol(buf+13, nil, 10);
  101. cal.transx = strtol(buf+25, nil, 10);
  102. cal.transy = strtol(buf+37, nil, 10);
  103. }else{
  104. fprint(mousectl, "calibrate");
  105. cal.scalex = 1<<16;
  106. cal.scaley = 1<<16;
  107. cal.transx = 0;
  108. cal.transy = 0;
  109. }
  110. fprint(2, "calibrate %ld %ld %ld %ld (old)\n",
  111. (long)cal.scalex, (long)cal.scaley, (long)cal.transx, (long)cal.transy);
  112. if(debug)
  113. fprint(2, "screen coords %R\n", screen->r);
  114. pts[0] = addpt(Pt( 16, 16), screen->r.min);
  115. pts[1] = addpt(Pt( 16, -16), Pt(screen->r.min.x, screen->r.max.y));
  116. pts[2] = addpt(Pt(-16, 16), Pt(screen->r.max.x, screen->r.min.y));
  117. pts[3] = addpt(Pt(-16, -16), screen->r.max);
  118. pts[4] = Pt((screen->r.max.x + screen->r.min.x)/2,
  119. (screen->r.max.y + screen->r.min.y)/2);;
  120. for(i = 0; i < 4; i++){
  121. cross(pts[i]);
  122. pens[i] = scr2pen(getmouse(mouse));
  123. if (debug) fprint(2, "%P\n", pens[i]);
  124. }
  125. cal.scalex = (pts[2].x + pts[3].x - pts[0].x - pts[1].x) * 65536.0 /
  126. (pens[2].x + pens[3].x - pens[0].x - pens[1].x);
  127. cal.scaley = (pts[1].y + pts[3].y - pts[0].y - pts[2].y) * 65536.0 /
  128. (pens[1].y + pens[3].y - pens[0].y - pens[2].y);
  129. cal.transx = pts[0].x - (pens[0].x*cal.scalex) / 65536.0;
  130. cal.transy = pts[0].y - (pens[0].y*cal.scaley) / 65536.0;
  131. if(debug)
  132. fprint(2, "scale [%f, %f], trans [%f, %f]\n",
  133. cal.scalex, cal.scaley, cal.transx, cal.transy);
  134. fprint(mousectl, "calibrate %ld %ld %ld %ld",
  135. (long)cal.scalex, (long)cal.scaley, (long)cal.transx, (long)cal.transy);
  136. cross(pts[4]);
  137. pens[4] = getmouse(mouse);
  138. draw(screen, screen->r, display->white, nil, ZP);
  139. flushimage(display, 1);
  140. if(abs(pts[4].x-pens[4].x) <= 4 && abs(pts[4].y-pens[4].y) <= 4){
  141. fprint(2, "Calibration ok: %P -> %P\n", pts[4], pens[4]);
  142. break;
  143. }
  144. fprint(2, "Calibration inaccurate: %P -> %P\n", pts[4], pens[4]);
  145. }
  146. print("calibrate='%ld %ld %ld %ld'\n",
  147. (long)cal.scalex, (long)cal.scaley, (long)cal.transx, (long)cal.transy);
  148. if(debug)
  149. for(i = 0; i < 4; i++){
  150. cross(pts[i]);
  151. pens[i] = getmouse(mouse);
  152. if (debug) fprint(2, "%P\n", pens[i]);
  153. }
  154. if(wctl >= 0)
  155. fprint(wctl, "resize -r %R", oldsize);
  156. }