2 * Copyright (C) 2012 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 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
44 #define INSTALLED_PACKAGES_MIN_SIZE 4
47 * Private data of an ietf_attr_installed_packages_t object.
49 struct private_ietf_attr_installed_packages_t
{
52 * Public members of ietf_attr_installed_packages_t
54 ietf_attr_installed_packages_t
public;
57 * Vendor-specific attribute type
72 * List of Installed Package entries
74 linked_list_t
*packages
;
85 struct package_entry_t
{
91 * Free a package entry
93 static void free_package_entry(package_entry_t
*entry
)
95 free(entry
->name
.ptr
);
96 free(entry
->version
.ptr
);
100 METHOD(pa_tnc_attr_t
, get_type
, pen_type_t
,
101 private_ietf_attr_installed_packages_t
*this)
106 METHOD(pa_tnc_attr_t
, get_value
, chunk_t
,
107 private_ietf_attr_installed_packages_t
*this)
112 METHOD(pa_tnc_attr_t
, get_noskip_flag
, bool,
113 private_ietf_attr_installed_packages_t
*this)
115 return this->noskip_flag
;
118 METHOD(pa_tnc_attr_t
, set_noskip_flag
,void,
119 private_ietf_attr_installed_packages_t
*this, bool noskip
)
121 this->noskip_flag
= noskip
;
124 METHOD(pa_tnc_attr_t
, build
, void,
125 private_ietf_attr_installed_packages_t
*this)
127 bio_writer_t
*writer
;
128 enumerator_t
*enumerator
;
129 package_entry_t
*entry
;
135 writer
= bio_writer_create(INSTALLED_PACKAGES_MIN_SIZE
);
136 writer
->write_uint16(writer
, 0x0000);
137 writer
->write_uint16(writer
, this->packages
->get_count(this->packages
));
139 enumerator
= this->packages
->create_enumerator(this->packages
);
140 while (enumerator
->enumerate(enumerator
, &entry
))
142 writer
->write_data8(writer
, entry
->name
);
143 writer
->write_data8(writer
, entry
->version
);
145 enumerator
->destroy(enumerator
);
147 this->value
= chunk_clone(writer
->get_buf(writer
));
148 writer
->destroy(writer
);
151 METHOD(pa_tnc_attr_t
, process
, status_t
,
152 private_ietf_attr_installed_packages_t
*this, u_int32_t
*offset
)
154 bio_reader_t
*reader
;
155 package_entry_t
*entry
;
156 status_t status
= FAILED
;
157 chunk_t name
, version
;
158 u_int16_t reserved
, count
;
163 if (this->value
.len
< INSTALLED_PACKAGES_MIN_SIZE
)
165 DBG1(DBG_TNC
, "insufficient data for IETF installed packages");
168 reader
= bio_reader_create(this->value
);
169 reader
->read_uint16(reader
, &reserved
);
170 reader
->read_uint16(reader
, &count
);
171 *offset
= INSTALLED_PACKAGES_MIN_SIZE
;
173 while (reader
->remaining(reader
))
175 if (!reader
->read_data8(reader
, &name
))
177 DBG1(DBG_TNC
, "insufficient data for IETF installed package name");
180 pos
= memchr(name
.ptr
, '\0', name
.len
);
183 DBG1(DBG_TNC
, "nul termination in IETF installed package name");
184 *offset
+= 1 + (pos
- name
.ptr
);
187 *offset
+= 1 + name
.len
;
189 if (!reader
->read_data8(reader
, &version
))
191 DBG1(DBG_TNC
, "insufficient data for IETF installed package version");
194 pos
= memchr(version
.ptr
, '\0', version
.len
);
197 DBG1(DBG_TNC
, "nul termination in IETF installed package version");
198 *offset
+= 1 + (pos
- version
.ptr
);
201 *offset
+= 1 + version
.len
;
203 entry
= malloc_thing(package_entry_t
);
204 entry
->name
= chunk_clone(name
);
205 entry
->version
= chunk_clone(version
);
206 this->packages
->insert_last(this->packages
, entry
);
209 if (count
!= this->packages
->get_count(this->packages
))
211 DBG1(DBG_TNC
, "IETF installed package count unequal to "
212 "number of included packages");
218 reader
->destroy(reader
);
222 METHOD(pa_tnc_attr_t
, get_ref
, pa_tnc_attr_t
*,
223 private_ietf_attr_installed_packages_t
*this)
226 return &this->public.pa_tnc_attribute
;
229 METHOD(pa_tnc_attr_t
, destroy
, void,
230 private_ietf_attr_installed_packages_t
*this)
232 if (ref_put(&this->ref
))
234 this->packages
->destroy_function(this->packages
, (void*)free_package_entry
);
235 free(this->value
.ptr
);
240 METHOD(ietf_attr_installed_packages_t
, add
, void,
241 private_ietf_attr_installed_packages_t
*this, chunk_t name
, chunk_t version
)
243 package_entry_t
*entry
;
245 /* restrict package name and package version number fields to 255 octets */
246 name
.len
= min(255, name
.len
);
247 version
.len
= min(255, version
.len
);
249 entry
= malloc_thing(package_entry_t
);
250 entry
->name
= chunk_clone(name
);
251 entry
->version
= chunk_clone(version
);
252 this->packages
->insert_last(this->packages
, entry
);
256 * Enumerate package filter entries
258 static bool package_filter(void *null
, package_entry_t
**entry
, chunk_t
*name
,
259 void *i2
, chunk_t
*version
)
261 *name
= (*entry
)->name
;
262 *version
= (*entry
)->version
;
266 METHOD(ietf_attr_installed_packages_t
, create_enumerator
, enumerator_t
*,
267 private_ietf_attr_installed_packages_t
*this)
269 return enumerator_create_filter(
270 this->packages
->create_enumerator(this->packages
),
271 (void*)package_filter
, NULL
, NULL
);
275 * Described in header.
277 pa_tnc_attr_t
*ietf_attr_installed_packages_create(void)
279 private_ietf_attr_installed_packages_t
*this;
283 .pa_tnc_attribute
= {
284 .get_type
= _get_type
,
285 .get_value
= _get_value
,
286 .get_noskip_flag
= _get_noskip_flag
,
287 .set_noskip_flag
= _set_noskip_flag
,
294 .create_enumerator
= _create_enumerator
,
296 .type
= { PEN_IETF
, IETF_ATTR_INSTALLED_PACKAGES
},
297 .packages
= linked_list_create(),
301 return &this->public.pa_tnc_attribute
;
305 * Described in header.
307 pa_tnc_attr_t
*ietf_attr_installed_packages_create_from_data(chunk_t data
)
309 private_ietf_attr_installed_packages_t
*this;
313 .pa_tnc_attribute
= {
314 .get_type
= _get_type
,
315 .get_value
= _get_value
,
316 .get_noskip_flag
= _get_noskip_flag
,
317 .set_noskip_flag
= _set_noskip_flag
,
324 .create_enumerator
= _create_enumerator
,
326 .type
= {PEN_IETF
, IETF_ATTR_INSTALLED_PACKAGES
},
327 .value
= chunk_clone(data
),
328 .packages
= linked_list_create(),
332 return &this->public.pa_tnc_attribute
;