00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025 #include <stdlib.h>
00026 #include <errno.h>
00027 #include "segment.h"
00028 #include "type_limits.h"
00029 #include "debug.h"
00030
00031 struct segment {
00032 void *data;
00033 off_t start;
00034 off_t size;
00035 segment_data_usage_func data_usage_func;
00036 };
00037
00038
00039
00040
00041
00042
00043
00044
00045
00046
00047
00048
00049
00050 int segment_new(segment_t **seg, void *data, off_t start, off_t size,
00051 segment_data_usage_func data_usage_func)
00052 {
00053 segment_t *segp = NULL;
00054
00055 segp = (segment_t *) malloc(sizeof(segment_t));
00056
00057 if (segp == NULL)
00058 return_error(ENOMEM);
00059
00060
00061 segp->data_usage_func = NULL;
00062
00063 int err = segment_set_data(segp, data, data_usage_func);
00064 if (err)
00065 goto fail;
00066
00067 err = segment_set_range(segp, start, size);
00068 if (err)
00069 goto fail;
00070
00071 *seg = segp;
00072
00073 return 0;
00074
00075 fail:
00076 free(segp);
00077 return_error(err);
00078 }
00079
00080
00081
00082
00083
00084
00085
00086
00087
00088 int segment_copy(segment_t *seg, segment_t **seg_copy)
00089 {
00090 if (seg == NULL || seg_copy == NULL)
00091 return_error(EINVAL);
00092
00093 int err = segment_new(seg_copy, seg->data, seg->start, seg->size,
00094 seg->data_usage_func);
00095
00096 if (err)
00097 return_error(err);
00098
00099 return 0;
00100 }
00101
00102
00103
00104
00105
00106
00107
00108
00109 int segment_free(segment_t *seg)
00110 {
00111
00112 if (seg->data_usage_func != NULL)
00113 (*seg->data_usage_func)(seg->data, -1);
00114
00115 free(seg);
00116
00117 return 0;
00118 }
00119
00120
00121
00122
00123
00124
00125
00126
00127 int segment_clear(segment_t *seg)
00128 {
00129 if (seg == NULL)
00130 return_error(EINVAL);
00131
00132 seg->start = -1;
00133 seg->size = 0;
00134
00135 return 0;
00136 }
00137
00138
00139
00140
00141
00142
00143
00144
00145
00146
00147
00148
00149
00150
00151
00152 int segment_split(segment_t *seg, segment_t **seg1, off_t split_index)
00153 {
00154 int err = 0;
00155
00156 if (seg == NULL || seg1 == NULL)
00157 return_error(EINVAL);
00158
00159 *seg1 = NULL;
00160
00161 off_t size = seg->size;
00162 off_t start = seg->start;
00163 void *data = seg->data;
00164
00165
00166 if (split_index >= size)
00167 return_error(EINVAL);
00168
00169 err = segment_new(seg1, data, start + split_index, size - split_index,
00170 seg->data_usage_func);
00171
00172 if (err)
00173 return_error(err);
00174
00175
00176 if (split_index == 0)
00177 segment_clear(seg);
00178 else {
00179 err = segment_set_range(seg, start, split_index);
00180 if (err)
00181 goto fail;
00182 }
00183
00184 return 0;
00185
00186 fail:
00187 segment_free(*seg1);
00188 *seg1 = NULL;
00189
00190 return_error(err);
00191 }
00192
00193
00194
00195
00196
00197
00198
00199
00200
00201
00202
00203
00204
00205 int segment_merge(segment_t *seg, segment_t *seg1)
00206 {
00207 if (seg == NULL || seg1 == NULL)
00208 return_error(EINVAL);
00209
00210
00211 if (seg->data != seg1->data)
00212 return_error(EINVAL);
00213
00214
00215 if (__MAX(off_t) - seg->size < seg1->size)
00216 return_error(EOVERFLOW);
00217
00218 off_t new_size = seg->size + seg1->size;
00219
00220
00221 if (__MAX(off_t) - seg->start < new_size - 1 * (new_size != 0))
00222 return_error(EOVERFLOW);
00223
00224
00225 if (seg->start + seg->size != seg1->start)
00226 return_error(EINVAL);
00227
00228 seg->size = new_size;
00229
00230 return 0;
00231 }
00232
00233
00234
00235
00236
00237
00238
00239
00240
00241 int segment_get_data(segment_t *seg, void **data)
00242 {
00243 if (seg == NULL || data == NULL)
00244 return_error(EINVAL);
00245
00246 *data = seg->data;
00247
00248 return 0;
00249 }
00250
00251
00252
00253
00254
00255
00256
00257
00258
00259 int segment_get_start(segment_t *seg, off_t *start)
00260 {
00261 if (seg == NULL || start == NULL)
00262 return_error(EINVAL);
00263
00264 *start = seg->start;
00265
00266 return 0;
00267 }
00268
00269
00270
00271
00272
00273
00274
00275
00276
00277 int segment_get_size(segment_t *seg, off_t *size)
00278 {
00279 if (seg == NULL || size == NULL)
00280 return_error(EINVAL);
00281
00282 *size = seg->size;
00283
00284 return 0;
00285 }
00286
00287
00288
00289
00290
00291
00292
00293
00294
00295
00296
00297 int segment_set_data(segment_t *seg, void *data,
00298 segment_data_usage_func data_usage_func)
00299 {
00300 if (seg == NULL)
00301 return_error(EINVAL);
00302
00303
00304 if (data_usage_func != NULL) {
00305 int err = (*data_usage_func)(data, 1);
00306 if (err)
00307 return_error(err);
00308 }
00309
00310
00311 if (seg->data_usage_func != NULL) {
00312 int err = (*seg->data_usage_func)(seg->data, -1);
00313 if (err) {
00314
00315 (*data_usage_func)(data, 0);
00316 return_error(err);
00317 }
00318 }
00319
00320 seg->data = data;
00321 seg->data_usage_func = data_usage_func;
00322
00323 return 0;
00324 }
00325
00326
00327
00328
00329
00330
00331
00332
00333
00334
00335 int segment_set_range(segment_t *seg, off_t start, off_t size)
00336 {
00337 if (seg == NULL || start < 0 || size < 0)
00338 return_error(EINVAL);
00339
00340
00341 if (__MAX(off_t) - start < size - 1 * (size != 0))
00342 return_error(EOVERFLOW);
00343
00344 seg->start = start;
00345 seg->size = size;
00346
00347 return 0;
00348 }
00349