2 * Copyright (C) 2011 Sansar Choinyambuu
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 "imv_attestation_state.h"
18 #include <utils/lexparser.h>
19 #include <utils/linked_list.h>
22 typedef struct private_imv_attestation_state_t private_imv_attestation_state_t
;
23 typedef struct file_meas_request_t file_meas_request_t
;
26 * PTS File/Directory Measurement request entry
28 struct file_meas_request_t
{
35 * Private data of an imv_attestation_state_t object.
37 struct private_imv_attestation_state_t
{
40 * Public members of imv_attestation_state_t
42 imv_attestation_state_t
public;
47 TNC_ConnectionID connection_id
;
50 * TNCCS connection state
52 TNC_ConnectionState state
;
55 * Does the TNCCS connection support long message types?
60 * Does the TNCCS connection support exclusive delivery?
65 * Maximum PA-TNC message size for this TNCCS connection
67 u_int32_t max_msg_len
;
70 * IMV Attestation handshake state
72 imv_attestation_handshake_state_t handshake_state
;
75 * IMV action recommendation
77 TNC_IMV_Action_Recommendation rec
;
80 * IMV evaluation result
82 TNC_IMV_Evaluation_Result eval
;
85 * File Measurement Request counter
87 u_int16_t file_meas_request_counter
;
90 * List of PTS File/Directory Measurement requests
92 linked_list_t
*file_meas_requests
;
95 * List of Functional Components
97 linked_list_t
*components
;
107 bool measurement_error
;
111 typedef struct entry_t entry_t
;
114 * Define an internal reason string entry
122 * Table of multi-lingual reason string entries
124 static entry_t reasons
[] = {
125 { "en", "IMV Attestation: Incorrect/pending file measurement/component"
126 " evidence or invalid TPM Quote signature received" },
127 { "mn", "IMV Attestation: Буруу/хүлээгдэж байгаа файл/компонент хэмжилт "
128 "эсвэл буруу TPM Quote гарын үсэг" },
129 { "de", "IMV Attestation: Falsche/Fehlende Dateimessung/Komponenten Beweis "
130 "oder ungültige TPM Quote Unterschrift ist erhalten" },
133 METHOD(imv_state_t
, get_connection_id
, TNC_ConnectionID
,
134 private_imv_attestation_state_t
*this)
136 return this->connection_id
;
139 METHOD(imv_state_t
, has_long
, bool,
140 private_imv_attestation_state_t
*this)
142 return this->has_long
;
145 METHOD(imv_state_t
, has_excl
, bool,
146 private_imv_attestation_state_t
*this)
148 return this->has_excl
;
151 METHOD(imv_state_t
, set_flags
, void,
152 private_imv_attestation_state_t
*this, bool has_long
, bool has_excl
)
154 this->has_long
= has_long
;
155 this->has_excl
= has_excl
;
158 METHOD(imv_state_t
, set_max_msg_len
, void,
159 private_imv_attestation_state_t
*this, u_int32_t max_msg_len
)
161 this->max_msg_len
= max_msg_len
;
164 METHOD(imv_state_t
, get_max_msg_len
, u_int32_t
,
165 private_imv_attestation_state_t
*this)
167 return this->max_msg_len
;
170 METHOD(imv_state_t
, change_state
, void,
171 private_imv_attestation_state_t
*this, TNC_ConnectionState new_state
)
173 this->state
= new_state
;
176 METHOD(imv_state_t
, get_recommendation
, void,
177 private_imv_attestation_state_t
*this, TNC_IMV_Action_Recommendation
*rec
,
178 TNC_IMV_Evaluation_Result
*eval
)
184 METHOD(imv_state_t
, set_recommendation
, void,
185 private_imv_attestation_state_t
*this, TNC_IMV_Action_Recommendation rec
,
186 TNC_IMV_Evaluation_Result eval
)
192 METHOD(imv_state_t
, get_reason_string
, bool,
193 private_imv_attestation_state_t
*this, chunk_t preferred_language
,
194 chunk_t
*reason_string
, chunk_t
*reason_language
)
196 chunk_t pref_lang
, lang
;
200 while (eat_whitespace(&preferred_language
))
202 if (!extract_token(&pref_lang
, ',', &preferred_language
))
204 /* last entry in a comma-separated list or single entry */
205 pref_lang
= preferred_language
;
208 /* eat trailing whitespace */
209 pos
= pref_lang
.ptr
+ pref_lang
.len
- 1;
210 while (pref_lang
.len
&& *pos
-- == ' ')
215 for (i
= 0 ; i
< countof(reasons
); i
++)
217 lang
= chunk_create(reasons
[i
].lang
, strlen(reasons
[i
].lang
));
218 if (chunk_equals(lang
, pref_lang
))
220 *reason_language
= lang
;
221 *reason_string
= chunk_create(reasons
[i
].string
,
222 strlen(reasons
[i
].string
));
228 /* no preferred language match found - use the default language */
229 *reason_string
= chunk_create(reasons
[0].string
,
230 strlen(reasons
[0].string
));
231 *reason_language
= chunk_create(reasons
[0].lang
,
232 strlen(reasons
[0].lang
));
236 METHOD(imv_state_t
, destroy
, void,
237 private_imv_attestation_state_t
*this)
239 this->file_meas_requests
->destroy_function(this->file_meas_requests
, free
);
240 this->components
->destroy_offset(this->components
,
241 offsetof(pts_component_t
, destroy
));
242 this->pts
->destroy(this->pts
);
246 METHOD(imv_attestation_state_t
, get_handshake_state
,
247 imv_attestation_handshake_state_t
, private_imv_attestation_state_t
*this)
249 return this->handshake_state
;
252 METHOD(imv_attestation_state_t
, set_handshake_state
, void,
253 private_imv_attestation_state_t
*this,
254 imv_attestation_handshake_state_t new_state
)
256 this->handshake_state
= new_state
;
259 METHOD(imv_attestation_state_t
, get_pts
, pts_t
*,
260 private_imv_attestation_state_t
*this)
265 METHOD(imv_attestation_state_t
, add_file_meas_request
, u_int16_t
,
266 private_imv_attestation_state_t
*this, int file_id
, bool is_dir
)
268 file_meas_request_t
*request
;
270 request
= malloc_thing(file_meas_request_t
);
271 request
->id
= ++this->file_meas_request_counter
;
272 request
->file_id
= file_id
;
273 request
->is_dir
= is_dir
;
274 this->file_meas_requests
->insert_last(this->file_meas_requests
, request
);
276 return this->file_meas_request_counter
;
279 METHOD(imv_attestation_state_t
, check_off_file_meas_request
, bool,
280 private_imv_attestation_state_t
*this, u_int16_t id
, int *file_id
,
283 enumerator_t
*enumerator
;
284 file_meas_request_t
*request
;
287 enumerator
= this->file_meas_requests
->create_enumerator(this->file_meas_requests
);
288 while (enumerator
->enumerate(enumerator
, &request
))
290 if (request
->id
== id
)
293 *file_id
= request
->file_id
;
294 *is_dir
= request
->is_dir
;
295 this->file_meas_requests
->remove_at(this->file_meas_requests
, enumerator
);
300 enumerator
->destroy(enumerator
);
304 METHOD(imv_attestation_state_t
, get_file_meas_request_count
, int,
305 private_imv_attestation_state_t
*this)
307 return this->file_meas_requests
->get_count(this->file_meas_requests
);
310 METHOD(imv_attestation_state_t
, add_component
, void,
311 private_imv_attestation_state_t
*this, pts_component_t
*entry
)
313 this->components
->insert_last(this->components
, entry
);
316 METHOD(imv_attestation_state_t
, get_component
, pts_component_t
*,
317 private_imv_attestation_state_t
*this, pts_comp_func_name_t
*name
)
319 enumerator_t
*enumerator
;
320 pts_component_t
*entry
, *found
= NULL
;
322 enumerator
= this->components
->create_enumerator(this->components
);
323 while (enumerator
->enumerate(enumerator
, &entry
))
325 if (name
->equals(name
, entry
->get_comp_func_name(entry
)))
331 enumerator
->destroy(enumerator
);
335 METHOD(imv_attestation_state_t
, get_measurement_error
, bool,
336 private_imv_attestation_state_t
*this)
338 return this->measurement_error
;
341 METHOD(imv_attestation_state_t
, set_measurement_error
, void,
342 private_imv_attestation_state_t
*this)
344 this->measurement_error
= TRUE
;
347 METHOD(imv_attestation_state_t
, finalize_components
, void,
348 private_imv_attestation_state_t
*this)
350 pts_component_t
*entry
;
352 while (this->components
->remove_last(this->components
,
353 (void**)&entry
) == SUCCESS
)
355 if (!entry
->finalize(entry
))
357 _set_measurement_error(this);
359 entry
->destroy(entry
);
364 * Described in header.
366 imv_state_t
*imv_attestation_state_create(TNC_ConnectionID connection_id
)
368 private_imv_attestation_state_t
*this;
374 .get_connection_id
= _get_connection_id
,
375 .has_long
= _has_long
,
376 .has_excl
= _has_excl
,
377 .set_flags
= _set_flags
,
378 .set_max_msg_len
= _set_max_msg_len
,
379 .get_max_msg_len
= _get_max_msg_len
,
380 .change_state
= _change_state
,
381 .get_recommendation
= _get_recommendation
,
382 .set_recommendation
= _set_recommendation
,
383 .get_reason_string
= _get_reason_string
,
386 .get_handshake_state
= _get_handshake_state
,
387 .set_handshake_state
= _set_handshake_state
,
389 .add_file_meas_request
= _add_file_meas_request
,
390 .check_off_file_meas_request
= _check_off_file_meas_request
,
391 .get_file_meas_request_count
= _get_file_meas_request_count
,
392 .add_component
= _add_component
,
393 .get_component
= _get_component
,
394 .finalize_components
= _finalize_components
,
395 .get_measurement_error
= _get_measurement_error
,
396 .set_measurement_error
= _set_measurement_error
,
398 .connection_id
= connection_id
,
399 .state
= TNC_CONNECTION_STATE_CREATE
,
400 .handshake_state
= IMV_ATTESTATION_STATE_INIT
,
401 .rec
= TNC_IMV_ACTION_RECOMMENDATION_NO_RECOMMENDATION
,
402 .eval
= TNC_IMV_EVALUATION_RESULT_DONT_KNOW
,
403 .file_meas_requests
= linked_list_create(),
404 .components
= linked_list_create(),
405 .pts
= pts_create(FALSE
),
408 platform_info
= lib
->settings
->get_str(lib
->settings
,
409 "libimcv.plugins.imv-attestation.platform_info", NULL
);
412 this->pts
->set_platform_info(this->pts
, platform_info
);
415 return &this->public.interface
;