00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021 #include <string.h>
00022
00023 #include "relfile.h"
00024 #include "tools.h"
00025
00026 #define MAX_RELOC_ITEM_LEN 8
00027
00028
00029
00030
00031 void ht_reloc_file::init(ht_streamfile *s, bool os)
00032 {
00033 ht_layer_streamfile::init(s, os);
00034 relocs = new ht_stree();
00035
00036 ((ht_stree*)relocs)->init(compare_keys_uint);
00037 enabled = true;
00038 }
00039
00040 void ht_reloc_file::done()
00041 {
00042 relocs->destroy();
00043 delete relocs;
00044 ht_layer_streamfile::done();
00045 }
00046
00047 void ht_reloc_file::finalize()
00048 {
00049
00050 }
00051
00052 int ht_reloc_file::vcntl(UINT cmd, va_list vargs)
00053 {
00054 switch (cmd) {
00055 case FCNTL_GET_RELOC: {
00056 bool *e = va_arg(vargs, bool*);
00057 *e = enabled;
00058 return 0;
00059 }
00060 case FCNTL_SET_RELOC: {
00061 enabled = (bool)(va_arg(vargs, int));
00062 return 0;
00063 }
00064 }
00065 return ht_layer_streamfile::vcntl(cmd, vargs);
00066 }
00067
00068 void ht_reloc_file::insert_reloc(FILEOFS o, Object *reloc)
00069 {
00070 relocs->insert(new ht_data_uint(o), reloc);
00071 }
00072
00073 UINT ht_reloc_file::read(void *buf, UINT size)
00074 {
00075 FILEOFS o = tell();
00076
00077 UINT ret = ht_layer_streamfile::read(buf, size), c = ret;
00078 if (enabled) {
00079 ht_data_uint q;
00080 Object *r;
00081 ht_data_uint *k = &q;
00082 if ((MAX_RELOC_ITEM_LEN+1) <= o)
00083 k->value = o - (MAX_RELOC_ITEM_LEN+1);
00084 else
00085 k = NULL;
00086
00087
00088
00089
00090
00091 while ((k = (ht_data_uint*)relocs->enum_next(&r, k))) {
00092
00093 byte b[MAX_RELOC_ITEM_LEN];
00094
00095
00096 if (k->value >= o+c) break;
00097
00098
00099
00100 UINT s = (k->value < o) ? o - k->value : 0;
00101
00102
00103 UINT e = (k->value > o) ? k->value - o : 0;
00104
00105
00106
00107 UINT l = (k->value + sizeof b > o+c) ?
00108 k->value + sizeof b - o - c : 0;
00109 UINT mm = MIN(sizeof b - l, sizeof b - s);
00110
00111
00112
00113
00114 memset(b, 0, sizeof b);
00115
00116
00117 assert(mm+s <= sizeof b);
00118
00119
00120 memmove(b+s, ((byte*)buf)+e, mm);
00121
00122 reloc_apply(r, b);
00123
00124 memmove(((byte*)buf)+e, b+s, mm);
00125 }
00126 }
00127 return ret;
00128 }
00129
00130 UINT ht_reloc_file::write(const void *buf, UINT size)
00131 {
00132
00133 FILEOFS o;
00134 if (enabled) {
00135 o = tell();
00136 UINT c = size;
00137 ht_data_uint q;
00138 Object *r;
00139 ht_data_uint *k = &q;
00140 if ((MAX_RELOC_ITEM_LEN+1) <= o)
00141 k->value = o - (MAX_RELOC_ITEM_LEN+1);
00142 else
00143 k = NULL;
00144
00145 while ((k = (ht_data_uint*)relocs->enum_next(&r, k))) {
00146 byte b[MAX_RELOC_ITEM_LEN];
00147 if (k->value >= o+c) break;
00148
00149 UINT s = (k->value < o) ? o - k->value : 0;
00150 UINT e = (k->value > o) ? k->value - o : 0;
00151
00152 UINT l = (k->value+sizeof b > o+c) ?
00153 k->value + sizeof b - o - c : 0;
00154
00155 memset(b, 0, sizeof b);
00156 UINT mm = MIN(sizeof b - l, sizeof b - s);
00157
00158 assert(mm+s <= sizeof b);
00159 memmove(b+s, ((byte*)buf)+e, mm);
00160
00161 if (!reloc_unapply(r, b)) ;
00162
00163 memmove(((byte*)buf)+e, b+s, mm);
00164 }
00165 }
00166 return ht_layer_streamfile::write(buf, size);
00167 }