Commit 980c9e19 authored by Masahiro Yamada's avatar Masahiro Yamada

kconfig: convert linked list of files to hash table

Currently, a linked list is used to keep track of all the Kconfig
files that have ever been parsed. Every time the "source" statement
is encountered, the linked list is traversed to check if the file has
been opened before. This prevents the same file from being recorded
in include/config/auto.conf.cmd again.

Given 1500+ Kconfig files parsed, a hashtable is now a more optimal
data structure.

By the way, you may wonder why we check this in the first place.
It matters only when the same file is included multiple times.
In old days, such a use case was forbidden, but commit f094f8a1
("kconfig: allow multiple inclusion of the same file") provided a bit
more flexibility. Of course, it is almost hypothetical...
Signed-off-by: default avatarMasahiro Yamada <masahiroy@kernel.org>
parent 7c4aa901
...@@ -7,6 +7,8 @@ ...@@ -7,6 +7,8 @@
#include <stdarg.h> #include <stdarg.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
#include "hashtable.h"
#include "lkc.h" #include "lkc.h"
unsigned int strhash(const char *s) unsigned int strhash(const char *s)
...@@ -19,32 +21,32 @@ unsigned int strhash(const char *s) ...@@ -19,32 +21,32 @@ unsigned int strhash(const char *s)
return hash; return hash;
} }
/* hash table of all parsed Kconfig files */
static HASHTABLE_DEFINE(file_hashtable, 1U << 11);
struct file { struct file {
struct file *next; struct hlist_node node;
char name[]; char name[];
}; };
static struct file *file_list;
/* file already present in list? If not add it */ /* file already present in list? If not add it */
const char *file_lookup(const char *name) const char *file_lookup(const char *name)
{ {
struct file *file; struct file *file;
size_t len; size_t len;
int hash = strhash(name);
for (file = file_list; file; file = file->next) { hash_for_each_possible(file_hashtable, file, node, hash)
if (!strcmp(name, file->name)) { if (!strcmp(name, file->name))
return file->name; return file->name;
}
}
len = strlen(name); len = strlen(name);
file = xmalloc(sizeof(*file) + len + 1); file = xmalloc(sizeof(*file) + len + 1);
memset(file, 0, sizeof(*file)); memset(file, 0, sizeof(*file));
memcpy(file->name, name, len); memcpy(file->name, name, len);
file->name[len] = '\0'; file->name[len] = '\0';
file->next = file_list;
file_list = file; hash_add(file_hashtable, &file->node, hash);
str_printf(&autoconf_cmd, "\t%s \\\n", name); str_printf(&autoconf_cmd, "\t%s \\\n", name);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment