2 * Copyright (C) 2012-2014 Andreas Steffen
3 * HSR Hochschule fuer Technik Rapperswil
5 * This program is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License as published by the
7 * Free Software Foundation; either version 2 of the License, or (at your
8 * option) any later version. See <http://www.fsf.org/copyleft/gpl.txt>.
10 * This program is distributed in the hope that it will be useful, but
11 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
12 * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
16 #include "ietf_attr_installed_packages.h"
20 #include <pa_tnc/pa_tnc_msg.h>
21 #include <bio/bio_writer.h>
22 #include <bio/bio_reader.h>
23 #include <collections/linked_list.h>
24 #include <utils/debug.h>
27 typedef struct private_ietf_attr_installed_packages_t private_ietf_attr_installed_packages_t
;
28 typedef struct package_entry_t package_entry_t
;
31 * PA-TNC Installed Packages Type (see section 4.2.7 of RFC 5792)
34 * 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
35 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
36 * | Reserved | Package Count |
37 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
38 * | Pkg Name Len | Package Name (Variable Length) |
39 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
40 * | Version Len | Package Version Number (Variable Length) |
41 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
45 * Private data of an ietf_attr_installed_packages_t object.
47 struct private_ietf_attr_installed_packages_t
{
50 * Public members of ietf_attr_installed_packages_t
52 ietf_attr_installed_packages_t
public;
55 * Vendor-specific attribute type
60 * Length of attribute value
65 * Attribute value or segment
75 * List of Installed Package entries
77 linked_list_t
*packages
;
88 struct package_entry_t
{
94 * Free a package entry
96 static void free_package_entry(package_entry_t
*entry
)
98 free(entry
->name
.ptr
);
99 free(entry
->version
.ptr
);
103 METHOD(pa_tnc_attr_t
, get_type
, pen_type_t
,
104 private_ietf_attr_installed_packages_t
*this)
109 METHOD(pa_tnc_attr_t
, get_value
, chunk_t
,
110 private_ietf_attr_installed_packages_t
*this)
115 METHOD(pa_tnc_attr_t
, get_noskip_flag
, bool,
116 private_ietf_attr_installed_packages_t
*this)
118 return this->noskip_flag
;
121 METHOD(pa_tnc_attr_t
, set_noskip_flag
,void,
122 private_ietf_attr_installed_packages_t
*this, bool noskip
)
124 this->noskip_flag
= noskip
;
127 METHOD(pa_tnc_attr_t
, build
, void,
128 private_ietf_attr_installed_packages_t
*this)
130 bio_writer_t
*writer
;
131 enumerator_t
*enumerator
;
132 package_entry_t
*entry
;
138 writer
= bio_writer_create(IETF_INSTALLED_PACKAGES_MIN_SIZE
);
139 writer
->write_uint16(writer
, 0x0000);
140 writer
->write_uint16(writer
, this->packages
->get_count(this->packages
));
142 enumerator
= this->packages
->create_enumerator(this->packages
);
143 while (enumerator
->enumerate(enumerator
, &entry
))
145 writer
->write_data8(writer
, entry
->name
);
146 writer
->write_data8(writer
, entry
->version
);
148 enumerator
->destroy(enumerator
);
150 this->value
= writer
->extract_buf(writer
);
151 this->length
= this->value
.len
;
152 writer
->destroy(writer
);
155 METHOD(pa_tnc_attr_t
, process
, status_t
,
156 private_ietf_attr_installed_packages_t
*this, u_int32_t
*offset
)
158 bio_reader_t
*reader
;
159 package_entry_t
*entry
;
160 status_t status
= FAILED
;
161 chunk_t name
, version
;
162 u_int16_t reserved
, count
;
167 if (this->value
.len
< this->length
)
171 if (this->value
.len
< IETF_INSTALLED_PACKAGES_MIN_SIZE
)
173 DBG1(DBG_TNC
, "insufficient data for IETF installed packages");
176 reader
= bio_reader_create(this->value
);
177 reader
->read_uint16(reader
, &reserved
);
178 reader
->read_uint16(reader
, &count
);
179 *offset
= IETF_INSTALLED_PACKAGES_MIN_SIZE
;
181 while (reader
->remaining(reader
))
183 if (!reader
->read_data8(reader
, &name
))
185 DBG1(DBG_TNC
, "insufficient data for IETF installed package name");
188 pos
= memchr(name
.ptr
, '\0', name
.len
);
191 DBG1(DBG_TNC
, "nul termination in IETF installed package name");
192 *offset
+= 1 + (pos
- name
.ptr
);
195 *offset
+= 1 + name
.len
;
197 if (!reader
->read_data8(reader
, &version
))
199 DBG1(DBG_TNC
, "insufficient data for IETF installed package version");
202 pos
= memchr(version
.ptr
, '\0', version
.len
);
205 DBG1(DBG_TNC
, "nul termination in IETF installed package version");
206 *offset
+= 1 + (pos
- version
.ptr
);
209 *offset
+= 1 + version
.len
;
211 entry
= malloc_thing(package_entry_t
);
212 entry
->name
= chunk_clone(name
);
213 entry
->version
= chunk_clone(version
);
214 this->packages
->insert_last(this->packages
, entry
);
217 if (count
!= this->packages
->get_count(this->packages
))
219 DBG1(DBG_TNC
, "IETF installed package count unequal to "
220 "number of included packages");
226 reader
->destroy(reader
);
230 METHOD(pa_tnc_attr_t
, add_segment
, void,
231 private_ietf_attr_installed_packages_t
*this, chunk_t segment
)
233 this->value
= chunk_cat("mc", this->value
, segment
);
236 METHOD(pa_tnc_attr_t
, get_ref
, pa_tnc_attr_t
*,
237 private_ietf_attr_installed_packages_t
*this)
240 return &this->public.pa_tnc_attribute
;
243 METHOD(pa_tnc_attr_t
, destroy
, void,
244 private_ietf_attr_installed_packages_t
*this)
246 if (ref_put(&this->ref
))
248 this->packages
->destroy_function(this->packages
, (void*)free_package_entry
);
249 free(this->value
.ptr
);
254 METHOD(ietf_attr_installed_packages_t
, add
, void,
255 private_ietf_attr_installed_packages_t
*this, chunk_t name
, chunk_t version
)
257 package_entry_t
*entry
;
259 /* restrict package name and package version number fields to 255 octets */
260 name
.len
= min(255, name
.len
);
261 version
.len
= min(255, version
.len
);
263 entry
= malloc_thing(package_entry_t
);
264 entry
->name
= chunk_clone(name
);
265 entry
->version
= chunk_clone(version
);
266 this->packages
->insert_last(this->packages
, entry
);
270 * Enumerate package filter entries
272 static bool package_filter(void *null
, package_entry_t
**entry
, chunk_t
*name
,
273 void *i2
, chunk_t
*version
)
275 *name
= (*entry
)->name
;
276 *version
= (*entry
)->version
;
280 METHOD(ietf_attr_installed_packages_t
, create_enumerator
, enumerator_t
*,
281 private_ietf_attr_installed_packages_t
*this)
283 return enumerator_create_filter(
284 this->packages
->create_enumerator(this->packages
),
285 (void*)package_filter
, NULL
, NULL
);
289 * Described in header.
291 pa_tnc_attr_t
*ietf_attr_installed_packages_create(void)
293 private_ietf_attr_installed_packages_t
*this;
297 .pa_tnc_attribute
= {
298 .get_type
= _get_type
,
299 .get_value
= _get_value
,
300 .get_noskip_flag
= _get_noskip_flag
,
301 .set_noskip_flag
= _set_noskip_flag
,
304 .add_segment
= _add_segment
,
309 .create_enumerator
= _create_enumerator
,
311 .type
= { PEN_IETF
, IETF_ATTR_INSTALLED_PACKAGES
},
312 .packages
= linked_list_create(),
316 return &this->public.pa_tnc_attribute
;
320 * Described in header. .length = length,
323 pa_tnc_attr_t
*ietf_attr_installed_packages_create_from_data(size_t length
,
326 private_ietf_attr_installed_packages_t
*this;
330 .pa_tnc_attribute
= {
331 .get_type
= _get_type
,
332 .get_value
= _get_value
,
333 .get_noskip_flag
= _get_noskip_flag
,
334 .set_noskip_flag
= _set_noskip_flag
,
337 .add_segment
= _add_segment
,
342 .create_enumerator
= _create_enumerator
,
344 .type
= {PEN_IETF
, IETF_ATTR_INSTALLED_PACKAGES
},
346 .value
= chunk_clone(data
),
347 .packages
= linked_list_create(),
351 return &this->public.pa_tnc_attribute
;