/* * Copyright (C) 2014 Andreas Steffen * HSR Hochschule fuer Technik Rapperswil * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation; either version 2 of the License, or (at your * option) any later version. See . * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * for more details. */ #include "tcg_seg_attr_seg_env.h" #include #include #include #include typedef struct private_tcg_seg_attr_seg_env_t private_tcg_seg_attr_seg_env_t; /** * Attribute Segment Envelope * see TCG IF-M Segmentation Specification * * 1 2 3 * 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ * |M|S| Reserved | Base Attribute ID | * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ * | Segment Value (Variable Length) | * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ */ /** * Private data of an tcg_seg_attr_seg_env_t object. */ struct private_tcg_seg_attr_seg_env_t { /** * Public members of tcg_seg_attr_seg_env_t */ tcg_seg_attr_seg_env_t public; /** * Vendor-specific attribute type */ pen_type_t type; /** * Length of attribute value */ size_t length; /** * Attribute value or segment */ chunk_t value; /** * Noskip flag */ bool noskip_flag; /** * PA-TNC segmentation flags */ uint8_t flags; /** * Base Attribute ID */ uint32_t base_attr_id; /** * Attribute value */ chunk_t segment; /** * Reference count */ refcount_t ref; }; METHOD(pa_tnc_attr_t, get_type, pen_type_t, private_tcg_seg_attr_seg_env_t *this) { return this->type; } METHOD(pa_tnc_attr_t, get_value, chunk_t, private_tcg_seg_attr_seg_env_t *this) { return this->value; } METHOD(pa_tnc_attr_t, get_noskip_flag, bool, private_tcg_seg_attr_seg_env_t *this) { return this->noskip_flag; } METHOD(pa_tnc_attr_t, set_noskip_flag,void, private_tcg_seg_attr_seg_env_t *this, bool noskip) { this->noskip_flag = noskip; } METHOD(pa_tnc_attr_t, build, void, private_tcg_seg_attr_seg_env_t *this) { /* constructor already allocated and built value */ this->length = this->value.len; return; } METHOD(pa_tnc_attr_t, process, status_t, private_tcg_seg_attr_seg_env_t *this, uint32_t *offset) { bio_reader_t *reader; *offset = 0; if (this->value.len < this->length) { DBG1(DBG_TNC, "segmentation not allowed for %N/%N", pen_names, PEN_TCG, tcg_attr_names, this->type.type); return FAILED; } if (this->value.len < TCG_SEG_ATTR_SEG_ENV_HEADER) { DBG1(DBG_TNC, "insufficient data for %N/%N", pen_names, PEN_TCG, tcg_attr_names, this->type.type); return FAILED; } reader = bio_reader_create(this->value); reader->read_uint8 (reader, &this->flags); reader->read_uint24(reader, &this->base_attr_id); reader->destroy(reader); return SUCCESS; } METHOD(pa_tnc_attr_t, add_segment, void, private_tcg_seg_attr_seg_env_t *this, chunk_t segment) { /* no segments are expected */ } METHOD(pa_tnc_attr_t, get_ref, pa_tnc_attr_t*, private_tcg_seg_attr_seg_env_t *this) { ref_get(&this->ref); return &this->public.pa_tnc_attribute; } METHOD(pa_tnc_attr_t, destroy, void, private_tcg_seg_attr_seg_env_t *this) { if (ref_put(&this->ref)) { free(this->value.ptr); free(this); } } METHOD(tcg_seg_attr_seg_env_t, get_segment, chunk_t, private_tcg_seg_attr_seg_env_t *this, uint8_t *flags) { if (flags) { *flags = this->flags; } return chunk_skip(this->value, TCG_SEG_ATTR_SEG_ENV_HEADER); } METHOD(tcg_seg_attr_seg_env_t, get_base_attr_id, uint32_t, private_tcg_seg_attr_seg_env_t *this) { return this->base_attr_id; } /** * Described in header. */ pa_tnc_attr_t* tcg_seg_attr_seg_env_create(chunk_t segment, uint8_t flags, uint32_t base_attr_id) { private_tcg_seg_attr_seg_env_t *this; INIT(this, .public = { .pa_tnc_attribute = { .get_type = _get_type, .get_value = _get_value, .get_noskip_flag = _get_noskip_flag, .set_noskip_flag = _set_noskip_flag, .build = _build, .process = _process, .add_segment = _add_segment, .get_ref = _get_ref, .destroy = _destroy, }, .get_base_attr_id = _get_base_attr_id, .get_segment = _get_segment, }, .type = { PEN_TCG, TCG_SEG_ATTR_SEG_ENV }, .flags = flags, .base_attr_id = base_attr_id, .value = chunk_alloc(TCG_SEG_ATTR_SEG_ENV_HEADER + segment.len), .ref = 1, ); htoun32(this->value.ptr, base_attr_id); *this->value.ptr = flags; memcpy(this->value.ptr + TCG_SEG_ATTR_SEG_ENV_HEADER, segment.ptr, segment.len); return &this->public.pa_tnc_attribute; } /** * Described in header. */ pa_tnc_attr_t *tcg_seg_attr_seg_env_create_from_data(size_t length, chunk_t data) { private_tcg_seg_attr_seg_env_t *this; INIT(this, .public = { .pa_tnc_attribute = { .get_type = _get_type, .get_value = _get_value, .get_noskip_flag = _get_noskip_flag, .set_noskip_flag = _set_noskip_flag, .build = _build, .process = _process, .add_segment = _add_segment, .get_ref = _get_ref, .destroy = _destroy, }, .get_base_attr_id = _get_base_attr_id, .get_segment = _get_segment, }, .type = { PEN_TCG, TCG_SEG_ATTR_SEG_ENV }, .length = length, .value = chunk_clone(data), .ref = 1, ); return &this->public.pa_tnc_attribute; }