load.c 1.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172
  1. #include <u.h>
  2. #include <libc.h>
  3. #include <draw.h>
  4. #include <memdraw.h>
  5. int
  6. loadmemimage(Memimage *i, Rectangle r, uchar *data, int ndata)
  7. {
  8. int y, l, lpart, rpart, mx, m, mr;
  9. uchar *q;
  10. if(!rectinrect(r, i->r))
  11. return -1;
  12. l = bytesperline(r, i->depth);
  13. if(ndata < l*Dy(r))
  14. return -1;
  15. ndata = l*Dy(r);
  16. q = byteaddr(i, r.min);
  17. mx = 7/i->depth;
  18. lpart = (r.min.x & mx) * i->depth;
  19. rpart = (r.max.x & mx) * i->depth;
  20. m = 0xFF >> lpart;
  21. /* may need to do bit insertion on edges */
  22. if(l == 1){ /* all in one byte */
  23. if(rpart)
  24. m ^= 0xFF >> rpart;
  25. for(y=r.min.y; y<r.max.y; y++){
  26. *q ^= (*data^*q) & m;
  27. q += i->width*sizeof(ulong);
  28. data++;
  29. }
  30. return ndata;
  31. }
  32. if(lpart==0 && rpart==0){ /* easy case */
  33. for(y=r.min.y; y<r.max.y; y++){
  34. memmove(q, data, l);
  35. q += i->width*sizeof(ulong);
  36. data += l;
  37. }
  38. return ndata;
  39. }
  40. mr = 0xFF ^ (0xFF >> rpart);
  41. if(lpart!=0 && rpart==0){
  42. for(y=r.min.y; y<r.max.y; y++){
  43. *q ^= (*data^*q) & m;
  44. if(l > 1)
  45. memmove(q+1, data+1, l-1);
  46. q += i->width*sizeof(ulong);
  47. data += l;
  48. }
  49. return ndata;
  50. }
  51. if(lpart==0 && rpart!=0){
  52. for(y=r.min.y; y<r.max.y; y++){
  53. if(l > 1)
  54. memmove(q, data, l-1);
  55. q[l-1] ^= (data[l-1]^q[l-1]) & mr;
  56. q += i->width*sizeof(ulong);
  57. data += l;
  58. }
  59. return ndata;
  60. }
  61. for(y=r.min.y; y<r.max.y; y++){
  62. *q ^= (*data^*q) & m;
  63. if(l > 2)
  64. memmove(q+1, data+1, l-2);
  65. q[l-1] ^= (data[l-1]^q[l-1]) & mr;
  66. q += i->width*sizeof(ulong);
  67. data += l;
  68. }
  69. return ndata;
  70. }