2 * cleankey.c - Routines to look for common key problems and clean them up.
4 * Copyright 2004,2012 Jonathan McDowell <noodles@earth.li>
6 * This program is free software: you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the Free
8 * Software Foundation; version 2 of the License.
10 * This program is distributed in the hope that it will be useful, but WITHOUT
11 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
15 * You should have received a copy of the GNU General Public License along with
16 * this program; if not, write to the Free Software Foundation, Inc., 51
17 * Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
26 #include "keystructs.h"
30 #include "onak-conf.h"
34 * dedupuids - Merge duplicate uids on a key.
35 * @key: The key to de-dup uids on.
37 * This function attempts to merge duplicate IDs on a key. It returns 0
38 * if the key is unchanged, otherwise the number of dups merged.
40 int dedupuids(struct openpgp_publickey *key)
42 struct openpgp_signedpacket_list *curuid = NULL;
43 struct openpgp_signedpacket_list *dup = NULL;
44 struct openpgp_signedpacket_list *tmp = NULL;
47 log_assert(key != NULL);
49 while (curuid != NULL) {
50 dup = find_signed_packet(curuid->next, curuid->packet);
52 logthing(LOGTHING_INFO, "Found duplicate uid: %.*s",
53 curuid->packet->length,
54 curuid->packet->data);
56 merge_packet_sigs(curuid, dup);
58 * Remove the duplicate uid.
61 while (tmp != NULL && tmp->next != dup) {
64 log_assert(tmp != NULL);
65 tmp->next = dup->next;
67 free_signedpacket_list(dup);
69 dup = find_signed_packet(curuid->next, curuid->packet);
71 curuid = curuid->next;
78 * dedupsubkeys - Merge duplicate subkeys on a key.
79 * @key: The key to de-dup subkeys on.
81 * This function attempts to merge duplicate subkeys on a key. It returns
82 * 0 if the key is unchanged, otherwise the number of dups merged.
84 int dedupsubkeys(struct openpgp_publickey *key)
86 struct openpgp_signedpacket_list *cursubkey = NULL;
87 struct openpgp_signedpacket_list *dup = NULL;
88 struct openpgp_signedpacket_list *tmp = NULL;
92 log_assert(key != NULL);
93 cursubkey = key->subkeys;
94 while (cursubkey != NULL) {
95 dup = find_signed_packet(cursubkey->next, cursubkey->packet);
97 get_packetid(cursubkey->packet, &subkeyid);
98 logthing(LOGTHING_INFO,
99 "Found duplicate subkey: 0x%016" PRIX64,
102 merge_packet_sigs(cursubkey, dup);
104 * Remove the duplicate uid.
107 while (tmp != NULL && tmp->next != dup) {
110 log_assert(tmp != NULL);
111 tmp->next = dup->next;
113 free_signedpacket_list(dup);
115 dup = find_signed_packet(cursubkey->next,
118 cursubkey = cursubkey->next;
125 * check_sighashes - Check that sig hashes are correct.
126 * @key - the check to check the sig hashes of.
128 * Given an OpenPGP key confirm that all of the sigs on it have the
129 * appropriate 2 octet hash beginning, as stored as part of the sig.
130 * This is a simple way to remove junk sigs and, for example, catches
131 * subkey sig corruption as produced by old pksd implementations.
132 * Any sig that has an incorrect hash is removed from the key. If the
133 * hash cannot be checked (eg we don't support that hash type) we err
134 * on the side of caution and keep it.
136 int clean_sighashes(struct openpgp_publickey *key,
137 struct openpgp_packet *sigdata,
138 struct openpgp_packet_list **sigs)
140 struct openpgp_packet_list *tmpsig;
143 while (*sigs != NULL) {
144 if (check_packet_sighash(key, sigdata, (*sigs)->packet) == 0) {
146 *sigs = (*sigs)->next;
148 free_packet_list(tmpsig);
151 sigs = &(*sigs)->next;
158 int clean_list_sighashes(struct openpgp_publickey *key,
159 struct openpgp_signedpacket_list *siglist)
163 while (siglist != NULL) {
164 removed += clean_sighashes(key, siglist->packet,
166 siglist = siglist->next;
172 int clean_key_sighashes(struct openpgp_publickey *key)
176 removed = clean_sighashes(key, NULL, &key->sigs);
177 removed += clean_list_sighashes(key, key->uids);
178 removed += clean_list_sighashes(key, key->subkeys);
184 * cleankeys - Apply all available cleaning options on a list of keys.
185 * @keys: The list of keys to clean.
187 * Applies all the cleaning options we can (eg duplicate key ids) to a
188 * list of keys. Returns 0 if no changes were made, otherwise the number
191 int cleankeys(struct openpgp_publickey *keys)
193 int changed = 0, count;
195 while (keys != NULL) {
196 count = dedupuids(keys);
197 count += dedupsubkeys(keys);
198 if (config.check_sighash) {
199 count += clean_key_sighashes(keys);