readimage.c 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. #include "lib9.h"
  2. #include "kernel.h"
  3. #include "draw.h"
  4. Image*
  5. readimage(Display *d, int fd, int dolock)
  6. {
  7. char hdr[5*12+1];
  8. int dy;
  9. int new;
  10. uint l, n;
  11. int m, j, chunk;
  12. int miny, maxy;
  13. Rectangle r;
  14. int ldepth;
  15. ulong chan;
  16. uchar *tmp;
  17. Image *i;
  18. if(libreadn(fd, hdr, 11) != 11) {
  19. kwerrstr("readimage: short header");
  20. return nil;
  21. }
  22. if(memcmp(hdr, "compressed\n", 11) == 0)
  23. return creadimage(d, fd, dolock);
  24. dolock &= 1;
  25. if(libreadn(fd, hdr+11, 5*12-11) != 5*12-11) {
  26. kwerrstr("readimage: short header");
  27. return nil;
  28. }
  29. chunk = d->bufsize - 32; /* a little room for header */
  30. /*
  31. * distinguish new channel descriptor from old ldepth.
  32. * channel descriptors have letters as well as numbers,
  33. * while ldepths are a single digit formatted as %-11d.
  34. */
  35. new = 0;
  36. for(m=0; m<10; m++){
  37. if(hdr[m] != ' '){
  38. new = 1;
  39. break;
  40. }
  41. }
  42. if(hdr[11] != ' '){
  43. kwerrstr("readimage: bad format");
  44. return nil;
  45. }
  46. if(new){
  47. hdr[11] = '\0';
  48. if((chan = strtochan(hdr)) == 0){
  49. kwerrstr("readimage: bad channel string %s", hdr);
  50. return nil;
  51. }
  52. }else{
  53. ldepth = ((int)hdr[10])-'0';
  54. if(ldepth<0 || ldepth>3){
  55. kwerrstr("readimage: bad ldepth %d", ldepth);
  56. return nil;
  57. }
  58. chan = drawld2chan[ldepth];
  59. }
  60. r.min.x = atoi(hdr+1*12);
  61. r.min.y = atoi(hdr+2*12);
  62. r.max.x = atoi(hdr+3*12);
  63. r.max.y = atoi(hdr+4*12);
  64. if(r.min.x>r.max.x || r.min.y>r.max.y){
  65. kwerrstr("readimage: bad rectangle");
  66. return nil;
  67. }
  68. miny = r.min.y;
  69. maxy = r.max.y;
  70. l = bytesperline(r, chantodepth(chan));
  71. if(dolock)
  72. dolock = lockdisplay(d);
  73. i = allocimage(d, r, chan, 0, -1);
  74. if(dolock)
  75. unlockdisplay(d);
  76. if(i == nil)
  77. return nil;
  78. tmp = malloc(chunk);
  79. if(tmp == nil)
  80. goto Err;
  81. while(maxy > miny){
  82. dy = maxy - miny;
  83. if(dy*l > chunk)
  84. dy = chunk/l;
  85. if(dy <= 0){
  86. kwerrstr("readimage: image too wide for buffer");
  87. goto Err;
  88. }
  89. n = dy*l;
  90. m = libreadn(fd, tmp, n);
  91. if(m != n){
  92. kwerrstr("readimage: read count %d not %d: %r", m, n);
  93. Err:
  94. if(dolock)
  95. lockdisplay(d);
  96. Err1:
  97. freeimage(i);
  98. if(dolock)
  99. unlockdisplay(d);
  100. free(tmp);
  101. return nil;
  102. }
  103. if(!new) /* an old image: must flip all the bits */
  104. for(j=0; j<chunk; j++)
  105. tmp[j] ^= 0xFF;
  106. if(dolock)
  107. lockdisplay(d);
  108. if(loadimage(i, Rect(r.min.x, miny, r.max.x, miny+dy), tmp, chunk) <= 0)
  109. goto Err1;
  110. if(dolock)
  111. unlockdisplay(d);
  112. miny += dy;
  113. }
  114. free(tmp);
  115. return i;
  116. }