1 /*- 2 * Copyright (c) 2009-2010 The FreeBSD Foundation 3 * All rights reserved. 4 * 5 * This software was developed by Pawel Jakub Dawidek under sponsorship from 6 * the FreeBSD Foundation. 7 * 8 * Redistribution and use in source and binary forms, with or without 9 * modification, are permitted provided that the following conditions 10 * are met: 11 * 1. Redistributions of source code must retain the above copyright 12 * notice, this list of conditions and the following disclaimer. 13 * 2. Redistributions in binary form must reproduce the above copyright 14 * notice, this list of conditions and the following disclaimer in the 15 * documentation and/or other materials provided with the distribution. 16 * 17 * THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS ``AS IS'' AND 18 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 20 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE 21 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 23 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 25 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 26 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 27 * SUCH DAMAGE. 28 */ 29 30 #include <sys/cdefs.h> 31 __FBSDID("$FreeBSD$"); 32 33 #include <sys/endian.h> 34 35 #include <assert.h> 36 #include <errno.h> 37 #include <string.h> 38 #include <strings.h> 39 40 #ifdef HAVE_CRYPTO 41 #include <openssl/sha.h> 42 #endif 43 44 #include <hast.h> 45 #include <ebuf.h> 46 #include <nv.h> 47 #include <pjdlog.h> 48 #include <proto.h> 49 50 #include "hast_proto.h" 51 52 struct hast_main_header { 53 /* Protocol version. */ 54 uint8_t version; 55 /* Size of nv headers. */ 56 uint32_t size; 57 } __packed; 58 59 typedef int hps_send_t(struct hast_resource *, struct nv *nv, void **, size_t *, bool *); 60 typedef int hps_recv_t(struct hast_resource *, struct nv *nv, void **, size_t *, bool *); 61 62 struct hast_pipe_stage { 63 const char *hps_name; 64 hps_send_t *hps_send; 65 hps_recv_t *hps_recv; 66 }; 67 68 static int compression_send(struct hast_resource *res, struct nv *nv, 69 void **datap, size_t *sizep, bool *freedatap); 70 static int compression_recv(struct hast_resource *res, struct nv *nv, 71 void **datap, size_t *sizep, bool *freedatap); 72 #ifdef HAVE_CRYPTO 73 static int checksum_send(struct hast_resource *res, struct nv *nv, 74 void **datap, size_t *sizep, bool *freedatap); 75 static int checksum_recv(struct hast_resource *res, struct nv *nv, 76 void **datap, size_t *sizep, bool *freedatap); 77 #endif 78 79 static struct hast_pipe_stage pipeline[] = { 80 { "compression", compression_send, compression_recv }, 81 #ifdef HAVE_CRYPTO 82 { "checksum", checksum_send, checksum_recv } 83 #endif 84 }; 85 86 static int 87 compression_send(struct hast_resource *res, struct nv *nv, void **datap, 88 size_t *sizep, bool *freedatap) 89 { 90 unsigned char *newbuf; 91 92 res = res; /* TODO */ 93 94 /* 95 * TODO: For now we emulate compression. 96 * At 80% probability we succeed to compress data, which means we 97 * allocate new buffer, copy the data over set *freedatap to true. 98 */ 99 100 if (arc4random_uniform(100) < 80) { 101 uint32_t *origsize; 102 103 /* 104 * Compression succeeded (but we will grow by 4 bytes, not 105 * shrink for now). 106 */ 107 newbuf = malloc(sizeof(uint32_t) + *sizep); 108 if (newbuf == NULL) 109 return (-1); 110 origsize = (void *)newbuf; 111 *origsize = htole32((uint32_t)*sizep); 112 nv_add_string(nv, "null", "compression"); 113 if (nv_error(nv) != 0) { 114 free(newbuf); 115 errno = nv_error(nv); 116 return (-1); 117 } 118 bcopy(*datap, newbuf + sizeof(uint32_t), *sizep); 119 if (*freedatap) 120 free(*datap); 121 *freedatap = true; 122 *datap = newbuf; 123 *sizep = sizeof(uint32_t) + *sizep; 124 } else { 125 /* 126 * Compression failed, so we leave everything as it was. 127 * It is not critical for compression to succeed. 128 */ 129 } 130 131 return (0); 132 } 133 134 static int 135 compression_recv(struct hast_resource *res, struct nv *nv, void **datap, 136 size_t *sizep, bool *freedatap) 137 { 138 unsigned char *newbuf; 139 const char *algo; 140 size_t origsize; 141 142 res = res; /* TODO */ 143 144 /* 145 * TODO: For now we emulate compression. 146 */ 147 148 algo = nv_get_string(nv, "compression"); 149 if (algo == NULL) 150 return (0); /* No compression. */ 151 if (strcmp(algo, "null") != 0) { 152 pjdlog_error("Unknown compression algorithm '%s'.", algo); 153 return (-1); /* Unknown compression algorithm. */ 154 } 155 156 origsize = le32toh(*(uint32_t *)*datap); 157 newbuf = malloc(origsize); 158 if (newbuf == NULL) 159 return (-1); 160 bcopy((unsigned char *)*datap + sizeof(uint32_t), newbuf, origsize); 161 if (*freedatap) 162 free(*datap); 163 *freedatap = true; 164 *datap = newbuf; 165 *sizep = origsize; 166 167 return (0); 168 } 169 170 #ifdef HAVE_CRYPTO 171 static int 172 checksum_send(struct hast_resource *res, struct nv *nv, void **datap, 173 size_t *sizep, bool *freedatap __unused) 174 { 175 unsigned char hash[SHA256_DIGEST_LENGTH]; 176 SHA256_CTX ctx; 177 178 res = res; /* TODO */ 179 180 SHA256_Init(&ctx); 181 SHA256_Update(&ctx, *datap, *sizep); 182 SHA256_Final(hash, &ctx); 183 184 nv_add_string(nv, "sha256", "checksum"); 185 nv_add_uint8_array(nv, hash, sizeof(hash), "hash"); 186 187 return (0); 188 } 189 190 static int 191 checksum_recv(struct hast_resource *res, struct nv *nv, void **datap, 192 size_t *sizep, bool *freedatap __unused) 193 { 194 unsigned char chash[SHA256_DIGEST_LENGTH]; 195 const unsigned char *rhash; 196 SHA256_CTX ctx; 197 const char *algo; 198 size_t size; 199 200 res = res; /* TODO */ 201 202 algo = nv_get_string(nv, "checksum"); 203 if (algo == NULL) 204 return (0); /* No checksum. */ 205 if (strcmp(algo, "sha256") != 0) { 206 pjdlog_error("Unknown checksum algorithm '%s'.", algo); 207 return (-1); /* Unknown checksum algorithm. */ 208 } 209 rhash = nv_get_uint8_array(nv, &size, "hash"); 210 if (rhash == NULL) { 211 pjdlog_error("Checksum algorithm is present, but hash is missing."); 212 return (-1); /* Hash not found. */ 213 } 214 if (size != sizeof(chash)) { 215 pjdlog_error("Invalid hash size (%zu) for %s, should be %zu.", 216 size, algo, sizeof(chash)); 217 return (-1); /* Different hash size. */ 218 } 219 220 SHA256_Init(&ctx); 221 SHA256_Update(&ctx, *datap, *sizep); 222 SHA256_Final(chash, &ctx); 223 224 if (bcmp(rhash, chash, sizeof(chash)) != 0) { 225 pjdlog_error("Hash mismatch."); 226 return (-1); /* Hash mismatch. */ 227 } 228 229 return (0); 230 } 231 #endif /* HAVE_CRYPTO */ 232 233 /* 234 * Send the given nv structure via conn. 235 * We keep headers in nv structure and pass data in separate argument. 236 * There can be no data at all (data is NULL then). 237 */ 238 int 239 hast_proto_send(struct hast_resource *res, struct proto_conn *conn, 240 struct nv *nv, const void *data, size_t size) 241 { 242 struct hast_main_header hdr; 243 struct ebuf *eb; 244 bool freedata; 245 void *dptr, *hptr; 246 size_t hsize; 247 int ret; 248 249 dptr = (void *)(uintptr_t)data; 250 freedata = false; 251 ret = -1; 252 253 if (data != NULL) { 254 if (false) { 255 unsigned int ii; 256 257 for (ii = 0; ii < sizeof(pipeline) / sizeof(pipeline[0]); 258 ii++) { 259 ret = pipeline[ii].hps_send(res, nv, &dptr, &size, 260 &freedata); 261 if (ret == -1) 262 goto end; 263 } 264 ret = -1; 265 } 266 nv_add_uint32(nv, size, "size"); 267 if (nv_error(nv) != 0) { 268 errno = nv_error(nv); 269 goto end; 270 } 271 } 272 273 eb = nv_hton(nv); 274 if (eb == NULL) 275 goto end; 276 277 hdr.version = HAST_PROTO_VERSION; 278 hdr.size = htole32((uint32_t)ebuf_size(eb)); 279 if (ebuf_add_head(eb, &hdr, sizeof(hdr)) < 0) 280 goto end; 281 282 hptr = ebuf_data(eb, &hsize); 283 if (proto_send(conn, hptr, hsize) < 0) 284 goto end; 285 if (data != NULL && proto_send(conn, dptr, size) < 0) 286 goto end; 287 288 ret = 0; 289 end: 290 if (freedata) 291 free(dptr); 292 return (ret); 293 } 294 295 int 296 hast_proto_recv_hdr(struct proto_conn *conn, struct nv **nvp) 297 { 298 struct hast_main_header hdr; 299 struct nv *nv; 300 struct ebuf *eb; 301 void *hptr; 302 303 eb = NULL; 304 nv = NULL; 305 306 if (proto_recv(conn, &hdr, sizeof(hdr)) < 0) 307 goto fail; 308 309 if (hdr.version != HAST_PROTO_VERSION) { 310 errno = ERPCMISMATCH; 311 goto fail; 312 } 313 314 hdr.size = le32toh(hdr.size); 315 316 eb = ebuf_alloc(hdr.size); 317 if (eb == NULL) 318 goto fail; 319 if (ebuf_add_tail(eb, NULL, hdr.size) < 0) 320 goto fail; 321 hptr = ebuf_data(eb, NULL); 322 assert(hptr != NULL); 323 if (proto_recv(conn, hptr, hdr.size) < 0) 324 goto fail; 325 nv = nv_ntoh(eb); 326 if (nv == NULL) 327 goto fail; 328 329 *nvp = nv; 330 return (0); 331 fail: 332 if (nv != NULL) 333 nv_free(nv); 334 else if (eb != NULL) 335 ebuf_free(eb); 336 return (-1); 337 } 338 339 int 340 hast_proto_recv_data(struct hast_resource *res, struct proto_conn *conn, 341 struct nv *nv, void *data, size_t size) 342 { 343 unsigned int ii; 344 bool freedata; 345 size_t dsize; 346 void *dptr; 347 int ret; 348 349 assert(data != NULL); 350 assert(size > 0); 351 352 ret = -1; 353 freedata = false; 354 dptr = data; 355 356 dsize = nv_get_uint32(nv, "size"); 357 if (dsize == 0) 358 (void)nv_set_error(nv, 0); 359 else { 360 if (proto_recv(conn, data, dsize) < 0) 361 goto end; 362 if (false) { 363 for (ii = sizeof(pipeline) / sizeof(pipeline[0]); ii > 0; 364 ii--) { 365 assert(!"to be verified"); 366 ret = pipeline[ii - 1].hps_recv(res, nv, &dptr, 367 &dsize, &freedata); 368 if (ret == -1) 369 goto end; 370 } 371 ret = -1; 372 if (dsize < size) 373 goto end; 374 /* TODO: 'size' doesn't seem right here. It is maximum data size. */ 375 if (dptr != data) 376 bcopy(dptr, data, dsize); 377 } 378 } 379 380 ret = 0; 381 end: 382 if (ret < 0) printf("%s:%u %s\n", __func__, __LINE__, strerror(errno)); 383 if (freedata) 384 free(dptr); 385 return (ret); 386 } 387 388 int 389 hast_proto_recv(struct hast_resource *res, struct proto_conn *conn, 390 struct nv **nvp, void *data, size_t size) 391 { 392 struct nv *nv; 393 size_t dsize; 394 int ret; 395 396 ret = hast_proto_recv_hdr(conn, &nv); 397 if (ret < 0) 398 return (ret); 399 dsize = nv_get_uint32(nv, "size"); 400 if (dsize == 0) 401 (void)nv_set_error(nv, 0); 402 else 403 ret = hast_proto_recv_data(res, conn, nv, data, size); 404 if (ret < 0) 405 nv_free(nv); 406 else 407 *nvp = nv; 408 return (ret); 409 } 410