pack.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. #include "stdinc.h"
  2. #include "dat.h"
  3. #include "fns.h"
  4. #include "error.h"
  5. /*
  6. * integer conversion routines
  7. */
  8. #define U8GET(p) ((p)[0])
  9. #define U16GET(p) (((p)[0]<<8)|(p)[1])
  10. #define U32GET(p) (((p)[0]<<24)|((p)[1]<<16)|((p)[2]<<8)|(p)[3])
  11. #define U48GET(p) (((uvlong)U16GET(p)<<32)|(uvlong)U32GET((p)+2))
  12. #define U64GET(p) (((uvlong)U32GET(p)<<32)|(uvlong)U32GET((p)+4))
  13. #define U8PUT(p,v) (p)[0]=(v)
  14. #define U16PUT(p,v) (p)[0]=(v)>>8;(p)[1]=(v)
  15. #define U32PUT(p,v) (p)[0]=(v)>>24;(p)[1]=(v)>>16;(p)[2]=(v)>>8;(p)[3]=(v)
  16. #define U48PUT(p,v,t32) t32=(v)>>32;U16PUT(p,t32);t32=(v);U32PUT((p)+2,t32)
  17. #define U64PUT(p,v,t32) t32=(v)>>32;U32PUT(p,t32);t32=(v);U32PUT((p)+4,t32)
  18. void
  19. headerPack(Header *h, uchar *p)
  20. {
  21. memset(p, 0, HeaderSize);
  22. U32PUT(p, HeaderMagic);
  23. U16PUT(p+4, HeaderVersion);
  24. U16PUT(p+6, h->blockSize);
  25. U32PUT(p+8, h->super);
  26. U32PUT(p+12, h->label);
  27. U32PUT(p+16, h->data);
  28. U32PUT(p+20, h->end);
  29. }
  30. int
  31. headerUnpack(Header *h, uchar *p)
  32. {
  33. if(U32GET(p) != HeaderMagic){
  34. vtSetError("vac header bad magic");
  35. return 0;
  36. }
  37. h->version = U16GET(p+4);
  38. if(h->version != HeaderVersion){
  39. vtSetError("vac header bad version");
  40. return 0;
  41. }
  42. h->blockSize = U16GET(p+6);
  43. h->super = U32GET(p+8);
  44. h->label = U32GET(p+12);
  45. h->data = U32GET(p+16);
  46. h->end = U32GET(p+20);
  47. return 1;
  48. }
  49. void
  50. labelPack(Label *l, uchar *p, int i)
  51. {
  52. p += i*LabelSize;
  53. U8PUT(p, l->state);
  54. U8PUT(p+1, l->type);
  55. U32PUT(p+2, l->epoch);
  56. U32PUT(p+6, l->epochClose);
  57. U32PUT(p+10, l->tag);
  58. }
  59. int
  60. labelUnpack(Label *l, uchar *p, int i)
  61. {
  62. p += i*LabelSize;
  63. l->state = p[0];
  64. l->type = p[1];
  65. l->epoch = U32GET(p+2);
  66. l->epochClose = U32GET(p+6);
  67. l->tag = U32GET(p+10);
  68. if(l->type > BtMax){
  69. Bad:
  70. vtSetError(EBadLabel);
  71. fprint(2, "labelUnpack %.2ux %.2ux %.8ux %.8ux %.8ux\n",
  72. l->state, l->type, l->epoch, l->epochClose, l->tag);
  73. return 0;
  74. }
  75. if(l->state != BsBad && l->state != BsFree){
  76. if(!(l->state&BsAlloc))
  77. goto Bad;
  78. if(l->state&~BsMask)
  79. goto Bad;
  80. if(l->state&BsClosed){
  81. if(l->epochClose == ~(u32int)0)
  82. goto Bad;
  83. }else{
  84. if(l->epochClose != ~(u32int)0)
  85. goto Bad;
  86. }
  87. }
  88. return 1;
  89. }
  90. u32int
  91. globalToLocal(uchar score[VtScoreSize])
  92. {
  93. int i;
  94. for(i=0; i<VtScoreSize-4; i++)
  95. if(score[i] != 0)
  96. return NilBlock;
  97. return U32GET(score+VtScoreSize-4);
  98. }
  99. void
  100. localToGlobal(u32int addr, uchar score[VtScoreSize])
  101. {
  102. memset(score, 0, VtScoreSize-4);
  103. U32PUT(score+VtScoreSize-4, addr);
  104. }
  105. void
  106. entryPack(Entry *e, uchar *p, int index)
  107. {
  108. ulong t32;
  109. int flags;
  110. p += index * VtEntrySize;
  111. U32PUT(p, e->gen);
  112. U16PUT(p+4, e->psize);
  113. U16PUT(p+6, e->dsize);
  114. flags = e->flags | ((e->depth << VtEntryDepthShift) & VtEntryDepthMask);
  115. U8PUT(p+8, flags);
  116. memset(p+9, 0, 5);
  117. U48PUT(p+14, e->size, t32);
  118. if(flags & VtEntryLocal){
  119. if(globalToLocal(e->score) == NilBlock)
  120. abort();
  121. memset(p+20, 0, 7);
  122. U8PUT(p+27, e->archive);
  123. U32PUT(p+28, e->snap);
  124. U32PUT(p+32, e->tag);
  125. memmove(p+36, e->score+16, 4);
  126. }else
  127. memmove(p+20, e->score, VtScoreSize);
  128. }
  129. int
  130. entryUnpack(Entry *e, uchar *p, int index)
  131. {
  132. p += index * VtEntrySize;
  133. e->gen = U32GET(p);
  134. e->psize = U16GET(p+4);
  135. e->dsize = U16GET(p+6);
  136. e->flags = U8GET(p+8);
  137. e->depth = (e->flags & VtEntryDepthMask) >> VtEntryDepthShift;
  138. e->flags &= ~VtEntryDepthMask;
  139. e->size = U48GET(p+14);
  140. if(e->flags & VtEntryLocal){
  141. e->archive = p[27];
  142. e->snap = U32GET(p+28);
  143. e->tag = U32GET(p+32);
  144. memset(e->score, 0, 16);
  145. memmove(e->score+16, p+36, 4);
  146. }else{
  147. e->archive = 0;
  148. e->snap = 0;
  149. e->tag = 0;
  150. memmove(e->score, p+20, VtScoreSize);
  151. }
  152. return 1;
  153. }
  154. int
  155. entryType(Entry *e)
  156. {
  157. return (((e->flags & VtEntryDir) != 0) << 3) | e->depth;
  158. }
  159. void
  160. superPack(Super *s, uchar *p)
  161. {
  162. u32int t32;
  163. memset(p, 0, SuperSize);
  164. U32PUT(p, SuperMagic);
  165. assert(s->version == SuperVersion);
  166. U16PUT(p+4, s->version);
  167. U32PUT(p+6, s->epochLow);
  168. U32PUT(p+10, s->epochHigh);
  169. U64PUT(p+14, s->qid, t32);
  170. U32PUT(p+22, s->active);
  171. U32PUT(p+26, s->next);
  172. U32PUT(p+30, s->current);
  173. memmove(p+34, s->last, VtScoreSize);
  174. memmove(p+54, s->name, sizeof(s->name));
  175. }
  176. int
  177. superUnpack(Super *s, uchar *p)
  178. {
  179. memset(s, 0, sizeof(*s));
  180. if(U32GET(p) != SuperMagic)
  181. goto Err;
  182. s->version = U16GET(p+4);
  183. if(s->version != SuperVersion)
  184. goto Err;
  185. s->epochLow = U32GET(p+6);
  186. s->epochHigh = U32GET(p+10);
  187. s->qid = U64GET(p+14);
  188. if(s->epochLow == 0 || s->epochLow > s->epochHigh || s->qid == 0)
  189. goto Err;
  190. s->active = U32GET(p+22);
  191. s->next = U32GET(p+26);
  192. s->current = U32GET(p+30);
  193. memmove(s->last, p+34, VtScoreSize);
  194. memmove(s->name, p+54, sizeof(s->name));
  195. s->name[sizeof(s->name)-1] = 0;
  196. return 1;
  197. Err:
  198. memset(s, 0, sizeof(*s));
  199. vtSetError(EBadSuper);
  200. return 0;
  201. }