dovecot/dovecot-a744ae38a9e1.patch

4504 lines
150 KiB
Diff
Raw Normal View History

diff --git a/NEWS b/NEWS
index 2251856..954db2b 100644
--- a/NEWS
+++ b/NEWS
@@ -15,7 +15,7 @@ v1.1.UNSTABLE
It's similar to REFERENCES, but it doesn't do base subject merging
and it sorts the threads by their newest message.
+ When saving messages, update cache file immediately with the data
- that we expect client to fetch later. Maildir-only currently.
+ that we expect client to fetch later.
+ NFS attribute and data caches are are flushed whenever needed.
See mail_nfs_storage and mail_nfs_index settings.
+ Mailbox list indexes. Mailbox metadata is stored there, so STATUS
diff --git a/configure.in b/configure.in
index 156e9d7..3f93ca6 100644
--- a/configure.in
+++ b/configure.in
@@ -82,7 +82,7 @@ AC_ARG_WITH(passwd,
fi,
want_passwd=yes)
-AC_ARG_WITH(passwd,
+AC_ARG_WITH(nss,
[ --with-nss Build with NSS module support (default)],
if test x$withval = xno; then
want_nss=no
diff --git a/doc/dovecot-ldap-example.conf b/doc/dovecot-ldap-example.conf
index 334ae8f..62ec866 100644
--- a/doc/dovecot-ldap-example.conf
+++ b/doc/dovecot-ldap-example.conf
@@ -106,7 +106,8 @@ base = uid=someone, dc=foo, dc=bar, dc=org
# userdb prefetch instead of userdb ldap in dovecot.conf. In that case you'll
# also have to include user_attrs in pass_attrs field prefixed with "userdb_"
# string. For example:
-#pass_attrs = uid=user,userPassword=password,homeDirectory=userdb_home,uidNumber=userdb_uid,gidNumber=userdb_gid
+#pass_attrs = uid=user,userPassword=password,\
+# homeDirectory=userdb_home,uidNumber=userdb_uid,gidNumber=userdb_gid
# Filter for password lookups
#pass_filter = (&(objectClass=posixAccount)(uid=%u))
diff --git a/doc/dovecot-sql-example.conf b/doc/dovecot-sql-example.conf
index bfc4b82..55fc998 100644
--- a/doc/dovecot-sql-example.conf
+++ b/doc/dovecot-sql-example.conf
@@ -88,7 +88,9 @@
# password_query = SELECT concat(userid, '@', domain) AS user, password FROM users WHERE userid = '%n' AND domain = '%d'
# password_query = SELECT pw AS password FROM users WHERE userid = '%u' AND active = 'Y'
#
-#password_query = SELECT userid as user, password FROM users WHERE userid = '%u'
+#password_query = \
+# SELECT userid as user, password \
+# FROM users WHERE userid = '%u'
# Query to retrieve the user information.
#
@@ -114,4 +116,7 @@
# userdb prefetch instead of userdb sql in dovecot.conf. In that case you'll
# also have to return userdb fields in password_query prefixed with "userdb_"
# string. For example:
-#password_query = SELECT userid as user, password, home as userdb_home, uid as userdb_uid, gid as userdb_gid FROM users WHERE userid = '%u'
+#password_query = \
+# SELECT userid as user, password, \
+# home as userdb_home, uid as userdb_uid, gid as userdb_gid \
+# FROM users WHERE userid = '%u'
diff --git a/src/auth/auth-client-connection.c b/src/auth/auth-client-connection.c
index d596854..e913a2e 100644
--- a/src/auth/auth-client-connection.c
+++ b/src/auth/auth-client-connection.c
@@ -275,10 +275,9 @@ auth_client_connection_create(struct auth_master_listener *listener, int fd)
conn->fd = fd;
conn->input =
- i_stream_create_file(fd, default_pool,
- AUTH_CLIENT_MAX_LINE_LENGTH, FALSE);
+ i_stream_create_file(fd, AUTH_CLIENT_MAX_LINE_LENGTH, FALSE);
conn->output =
- o_stream_create_file(fd, default_pool, (size_t)-1, FALSE);
+ o_stream_create_file(fd, (size_t)-1, FALSE);
o_stream_set_flush_callback(conn->output, auth_client_output, conn);
conn->io = io_add(fd, IO_READ, auth_client_input, conn);
diff --git a/src/auth/auth-master-connection.c b/src/auth/auth-master-connection.c
index d18c70e..ddfe029 100644
--- a/src/auth/auth-master-connection.c
+++ b/src/auth/auth-master-connection.c
@@ -242,10 +242,8 @@ auth_master_connection_create(struct auth_master_listener *listener, int fd)
conn->listener = listener;
conn->refcount = 1;
conn->fd = fd;
- conn->input = i_stream_create_file(fd, default_pool,
- MAX_INBUF_SIZE, FALSE);
- conn->output = o_stream_create_file(fd, default_pool,
- (size_t)-1, FALSE);
+ conn->input = i_stream_create_file(fd, MAX_INBUF_SIZE, FALSE);
+ conn->output = o_stream_create_file(fd, (size_t)-1, FALSE);
o_stream_set_flush_callback(conn->output, master_output, conn);
conn->io = io_add(fd, IO_READ, master_input, conn);
diff --git a/src/auth/auth-worker-client.c b/src/auth/auth-worker-client.c
index 2fca127..58ba197 100644
--- a/src/auth/auth-worker-client.c
+++ b/src/auth/auth-worker-client.c
@@ -468,10 +468,9 @@ auth_worker_client_create(struct auth *auth, int fd)
client->auth = auth;
client->fd = fd;
client->input =
- i_stream_create_file(fd, default_pool,
- AUTH_WORKER_MAX_LINE_LENGTH, FALSE);
+ i_stream_create_file(fd, AUTH_WORKER_MAX_LINE_LENGTH, FALSE);
client->output =
- o_stream_create_file(fd, default_pool, (size_t)-1, FALSE);
+ o_stream_create_file(fd, (size_t)-1, FALSE);
o_stream_set_flush_callback(client->output, auth_worker_output, client);
client->io = io_add(fd, IO_READ, auth_worker_input, client);
diff --git a/src/auth/auth-worker-server.c b/src/auth/auth-worker-server.c
index 537650b..b36c61e 100644
--- a/src/auth/auth-worker-server.c
+++ b/src/auth/auth-worker-server.c
@@ -79,10 +79,9 @@ static struct auth_worker_connection *auth_worker_create(void)
conn = i_new(struct auth_worker_connection, 1);
conn->fd = fd;
- conn->input = i_stream_create_file(fd, default_pool,
- AUTH_WORKER_MAX_LINE_LENGTH, FALSE);
- conn->output =
- o_stream_create_file(fd, default_pool, (size_t)-1, FALSE);
+ conn->input = i_stream_create_file(fd, AUTH_WORKER_MAX_LINE_LENGTH,
+ FALSE);
+ conn->output = o_stream_create_file(fd, (size_t)-1, FALSE);
conn->io = io_add(fd, IO_READ, worker_input, conn);
conn->requests = buffer_create_dynamic(default_pool, 128);
diff --git a/src/auth/db-ldap.c b/src/auth/db-ldap.c
index 7b564d1..363461c 100644
--- a/src/auth/db-ldap.c
+++ b/src/auth/db-ldap.c
@@ -52,9 +52,11 @@ struct db_ldap_result_iterate_context {
char *attr, **vals;
const char *name, *value, *template, *val_1_arr[2];
+ const char *const *static_attrs;
BerElement *ber;
string_t *var, *debug;
+ unsigned int value_idx;
};
#define DEF_STR(name) DEF_STRUCT_STR(name, ldap_settings)
@@ -102,9 +104,9 @@ struct ldap_settings default_ldap_settings = {
MEMBER(scope) "subtree",
MEMBER(base) NULL,
MEMBER(ldap_version) 2,
- MEMBER(user_attrs) "uid,homeDirectory,,,uidNumber,gidNumber",
+ MEMBER(user_attrs) "homeDirectory=home,uidNumber=uid,gidNumber=gid",
MEMBER(user_filter) "(&(objectClass=posixAccount)(uid=%u))",
- MEMBER(pass_attrs) "uid,userPassword",
+ MEMBER(pass_attrs) "uid=user,userPassword=password",
MEMBER(pass_filter) "(&(objectClass=posixAccount)(uid=%u))",
MEMBER(default_pass_scheme) "crypt"
};
@@ -618,10 +620,10 @@ static void ldap_conn_close(struct ldap_connection *conn, bool flush_requests)
void db_ldap_set_attrs(struct ldap_connection *conn, const char *attrlist,
char ***attr_names_r, struct hash_table *attr_map,
- const char *const default_attr_map[],
const char *skip_attr)
{
const char *const *attr;
+ string_t *static_data;
char *name, *value, *p;
unsigned int i, j, size;
@@ -630,6 +632,7 @@ void db_ldap_set_attrs(struct ldap_connection *conn, const char *attrlist,
t_push();
attr = t_strsplit(attrlist, ",");
+ static_data = t_str_new(128);
/* @UNSAFE */
for (size = 0; attr[size] != NULL; size++) ;
@@ -637,13 +640,17 @@ void db_ldap_set_attrs(struct ldap_connection *conn, const char *attrlist,
for (i = j = 0; i < size; i++) {
p = strchr(attr[i], '=');
- if (p == NULL) {
- name = p_strdup(conn->pool, attr[i]);
- value = *default_attr_map == NULL ? name :
- p_strdup(conn->pool, *default_attr_map);
- } else {
+ if (p == NULL)
+ name = value = p_strdup(conn->pool, attr[i]);
+ else if (p != attr[i]) {
name = p_strdup_until(conn->pool, attr[i], p);
value = p_strdup(conn->pool, p + 1);
+ } else {
+ /* =<static key>=<static value> */
+ if (str_len(static_data) > 0)
+ str_append_c(static_data, ',');
+ str_append(static_data, p + 1);
+ continue;
}
if (*name != '\0' &&
@@ -651,9 +658,10 @@ void db_ldap_set_attrs(struct ldap_connection *conn, const char *attrlist,
hash_insert(attr_map, name, value);
(*attr_names_r)[j++] = name;
}
-
- if (*default_attr_map != NULL)
- default_attr_map++;
+ }
+ if (str_len(static_data) > 0) {
+ hash_insert(attr_map, "",
+ p_strdup(conn->pool, str_c(static_data)));
}
t_pop();
}
@@ -709,6 +717,7 @@ db_ldap_result_iterate_init(struct ldap_connection *conn, LDAPMessage *entry,
struct hash_table *attr_map)
{
struct db_ldap_result_iterate_context *ctx;
+ const char *static_data;
ctx = t_new(struct db_ldap_result_iterate_context, 1);
ctx->conn = conn;
@@ -716,6 +725,10 @@ db_ldap_result_iterate_init(struct ldap_connection *conn, LDAPMessage *entry,
ctx->auth_request = auth_request;
ctx->attr_map = attr_map;
+ static_data = hash_lookup(attr_map, "");
+ if (static_data != NULL)
+ ctx->static_attrs = t_strsplit(static_data, ",");
+
if (auth_request->auth->verbose_debug)
ctx->debug = t_str_new(256);
@@ -764,12 +777,13 @@ db_ldap_result_change_attr(struct db_ldap_result_iterate_context *ctx)
ctx->vals = ldap_get_values(ctx->conn->ld, ctx->entry,
ctx->attr);
ctx->value = ctx->vals[0];
+ ctx->value_idx = 0;
}
static void
db_ldap_result_return_value(struct db_ldap_result_iterate_context *ctx)
{
- bool first = ctx->value == ctx->vals[0];
+ bool first = ctx->value_idx == 0;
if (ctx->template != NULL) {
ctx->var_table[0].value = ctx->value;
@@ -791,6 +805,8 @@ db_ldap_result_return_value(struct db_ldap_result_iterate_context *ctx)
static bool db_ldap_result_int_next(struct db_ldap_result_iterate_context *ctx)
{
+ const char *p;
+
while (ctx->attr != NULL) {
if (ctx->vals == NULL) {
/* a new attribute */
@@ -798,7 +814,7 @@ static bool db_ldap_result_int_next(struct db_ldap_result_iterate_context *ctx)
} else {
/* continuing existing attribute */
if (ctx->value != NULL)
- ctx->value++;
+ ctx->value = ctx->vals[++ctx->value_idx];
}
if (ctx->value != NULL) {
@@ -812,6 +828,19 @@ static bool db_ldap_result_int_next(struct db_ldap_result_iterate_context *ctx)
ctx->ber);
}
+ if (ctx->static_attrs != NULL && *ctx->static_attrs != NULL) {
+ p = strchr(*ctx->static_attrs, '=');
+ if (p == NULL) {
+ ctx->name = *ctx->static_attrs;
+ ctx->value = "";
+ } else {
+ ctx->name = t_strdup_until(*ctx->static_attrs, p);
+ ctx->value = p + 1;
+ }
+ ctx->static_attrs++;
+ return TRUE;
+ }
+
db_ldap_result_iterate_finish(ctx);
return FALSE;
}
diff --git a/src/auth/db-ldap.h b/src/auth/db-ldap.h
index 621e8b9..36823c2 100644
--- a/src/auth/db-ldap.h
+++ b/src/auth/db-ldap.h
@@ -99,7 +99,6 @@ void db_ldap_search(struct ldap_connection *conn, struct ldap_request *request,
void db_ldap_set_attrs(struct ldap_connection *conn, const char *attrlist,
char ***attr_names_r, struct hash_table *attr_map,
- const char *const default_attr_map[],
const char *skip_attr);
struct ldap_connection *db_ldap_init(const char *config_path);
diff --git a/src/auth/db-passwd-file.c b/src/auth/db-passwd-file.c
index 74c1f3d..27ac55e 100644
--- a/src/auth/db-passwd-file.c
+++ b/src/auth/db-passwd-file.c
@@ -183,7 +183,7 @@ static bool passwd_file_open(struct passwd_file *pw)
pw->users = hash_create(default_pool, pw->pool, 100,
str_hash, (hash_cmp_callback_t *)strcmp);
- input = i_stream_create_file(pw->fd, default_pool, 4096, FALSE);
+ input = i_stream_create_file(pw->fd, 4096, FALSE);
while ((line = i_stream_read_next_line(input)) != NULL) {
if (*line == '\0' || *line == ':' || *line == '#')
continue; /* no username or comment */
diff --git a/src/auth/passdb-ldap.c b/src/auth/passdb-ldap.c
index 45db512..1d43851 100644
--- a/src/auth/passdb-ldap.c
+++ b/src/auth/passdb-ldap.c
@@ -15,10 +15,6 @@
#include <ldap.h>
#include <stdlib.h>
-static const char *default_attr_map[] = {
- "user", "password", NULL
-};
-
struct ldap_passdb_module {
struct passdb_module module;
@@ -434,7 +430,7 @@ passdb_ldap_preinit(struct auth_passdb *auth_passdb, const char *args)
if (conn->set.auth_bind_userdn != NULL)
conn->set.auth_bind = TRUE;
db_ldap_set_attrs(conn, conn->set.pass_attrs, &conn->pass_attr_names,
- conn->pass_attr_map, default_attr_map,
+ conn->pass_attr_map,
conn->set.auth_bind ? "password" : NULL);
module->module.cache_key =
auth_cache_parse_key(auth_passdb->auth->pool,
diff --git a/src/auth/userdb-ldap.c b/src/auth/userdb-ldap.c
index 3a8d2f5..170add0 100644
--- a/src/auth/userdb-ldap.c
+++ b/src/auth/userdb-ldap.c
@@ -26,10 +26,6 @@ struct userdb_ldap_request {
userdb_callback_t *userdb_callback;
};
-static const char *default_attr_map[] = {
- "", "home", "mail", "system_user", "uid", "gid", NULL
-};
-
static void
ldap_query_get_result(struct ldap_connection *conn, LDAPMessage *entry,
struct auth_request *auth_request)
@@ -37,6 +33,8 @@ ldap_query_get_result(struct ldap_connection *conn, LDAPMessage *entry,
struct db_ldap_result_iterate_context *ldap_iter;
const char *name, *const *values;
+ auth_request_init_userdb_reply(auth_request);
+
ldap_iter = db_ldap_result_iterate_init(conn, entry, auth_request,
conn->user_attr_map);
while (db_ldap_result_iterate_next_all(ldap_iter, &name, &values)) {
@@ -139,7 +137,7 @@ userdb_ldap_preinit(struct auth_userdb *auth_userdb, const char *args)
(hash_cmp_callback_t *)strcmp);
db_ldap_set_attrs(conn, conn->set.user_attrs, &conn->user_attr_names,
- conn->user_attr_map, default_attr_map, NULL);
+ conn->user_attr_map, NULL);
module->module.cache_key =
auth_cache_parse_key(auth_userdb->auth->pool,
conn->set.user_filter);
diff --git a/src/deliver/auth-client.c b/src/deliver/auth-client.c
index 083a70e..8a865bd 100644
--- a/src/deliver/auth-client.c
+++ b/src/deliver/auth-client.c
@@ -179,10 +179,8 @@ static struct auth_connection *auth_connection_new(const char *auth_socket)
conn = i_new(struct auth_connection, 1);
conn->fd = fd;
- conn->input =
- i_stream_create_file(fd, default_pool, MAX_INBUF_SIZE, FALSE);
- conn->output =
- o_stream_create_file(fd, default_pool, MAX_OUTBUF_SIZE, FALSE);
+ conn->input = i_stream_create_file(fd, MAX_INBUF_SIZE, FALSE);
+ conn->output = o_stream_create_file(fd, MAX_OUTBUF_SIZE, FALSE);
conn->io = io_add(fd, IO_READ, auth_input, conn);
return conn;
}
diff --git a/src/deliver/deliver.c b/src/deliver/deliver.c
index 42bb812..e38a621 100644
--- a/src/deliver/deliver.c
+++ b/src/deliver/deliver.c
@@ -229,7 +229,7 @@ static void config_file_init(const char *path)
i_fatal_status(EX_CONFIG, "open(%s) failed: %m", path);
t_push();
- input = i_stream_create_file(fd, default_pool, 1024, TRUE);
+ input = i_stream_create_file(fd, 1024, TRUE);
while ((line = i_stream_read_next_line(input)) != NULL) {
/* @UNSAFE: line is modified */
@@ -417,7 +417,7 @@ static struct istream *create_mbox_stream(int fd, const char *envelope_sender)
envelope_sender = address_sanitize(envelope_sender);
mbox_hdr = mbox_from_create(envelope_sender, ioloop_time);
- input = i_stream_create_file(fd, default_pool, 4096, FALSE);
+ input = i_stream_create_file(fd, 4096, FALSE);
input_filter =
i_stream_create_header_filter(input,
HEADER_FILTER_EXCLUDE |
@@ -428,14 +428,12 @@ static struct istream *create_mbox_stream(int fd, const char *envelope_sender)
NULL);
i_stream_unref(&input);
- input_list[0] = i_stream_create_from_data(default_pool, mbox_hdr,
- strlen(mbox_hdr));
+ input_list[0] = i_stream_create_from_data(mbox_hdr, strlen(mbox_hdr));
input_list[1] = input_filter;
- input_list[2] = i_stream_create_from_data(default_pool, "\n", 1);
+ input_list[2] = i_stream_create_from_data("\n", 1);
input_list[3] = NULL;
- input = i_stream_create_seekable(input_list, default_pool,
- MAIL_MAX_MEMORY_BUFFER,
+ input = i_stream_create_seekable(input_list, MAIL_MAX_MEMORY_BUFFER,
"/tmp/dovecot.deliver.");
i_stream_unref(&input_list[0]);
i_stream_unref(&input_list[1]);
@@ -754,28 +752,46 @@ int main(int argc, char *argv[])
if (mail_set_seq(mail, 1) < 0)
i_fatal("mail_set_seq() failed");
+ storage = NULL;
default_mailbox_name = mailbox;
- if (deliver_mail != NULL)
- (void)deliver_mail(ns, &storage, mail, destination, mailbox);
+ if (deliver_mail == NULL)
+ ret = -1;
+ else {
+ if (deliver_mail(ns, &storage, mail,
+ destination, mailbox) <= 0) {
+ /* if message was saved, don't bounce it even though
+ the script failed later. */
+ ret = saved_mail ? 0 : -1;
+ } else {
+ /* success. message may or may not have been saved. */
+ ret = 0;
+ }
+ }
- if (!saved_mail && !tried_default_save) {
+ if (ret < 0 && !tried_default_save) {
/* plugins didn't handle this. save into the default mailbox. */
i_stream_seek(input, 0);
- (void)deliver_save(ns, &storage, mailbox, mail, 0, NULL);
+ ret = deliver_save(ns, &storage, mailbox, mail, 0, NULL);
}
- if (!saved_mail && strcasecmp(mailbox, "INBOX") != 0) {
+ if (ret < 0 && strcasecmp(mailbox, "INBOX") != 0) {
/* still didn't work. try once more to save it
to INBOX. */
i_stream_seek(input, 0);
- (void)deliver_save(ns, &storage, "INBOX", mail, 0, NULL);
+ ret = deliver_save(ns, &storage, "INBOX", mail, 0, NULL);
}
- if (!saved_mail) {
+ if (ret < 0 ) {
const char *error_string, *msgid;
enum mail_error error;
int ret;
- error_string = mail_storage_get_last_error(ns->storage, &error);
+ if (storage == NULL) {
+ /* This shouldn't happen */
+ i_error("BUG: Saving failed for unknown storage");
+ return EX_TEMPFAIL;
+ }
+
+ error_string = mail_storage_get_last_error(storage, &error);
if (error != MAIL_ERROR_NOSPACE ||
getenv("QUOTA_FULL_TEMPFAIL") != NULL) {
/* Saving to INBOX should always work unless
diff --git a/src/deliver/duplicate.c b/src/deliver/duplicate.c
index bb6f5d5..a724cc9 100644
--- a/src/deliver/duplicate.c
+++ b/src/deliver/duplicate.c
@@ -94,7 +94,7 @@ static int duplicate_read(struct duplicate_file *file)
}
/* <timestamp> <id_size> <user_size> <id> <user> */
- input = i_stream_create_file(fd, default_pool, 4096, FALSE);
+ input = i_stream_create_file(fd, 4096, FALSE);
change_count = 0;
while (i_stream_read_data(input, &data, &size, sizeof(stamp) +
@@ -218,7 +218,7 @@ void duplicate_flush(void)
if (duplicate_file == NULL || !file->changed || file->new_fd == -1)
return;
- output = o_stream_create_file(file->new_fd, default_pool, 4096, FALSE);
+ output = o_stream_create_file(file->new_fd, 4096, FALSE);
iter = hash_iterate_init(file->hash);
while (hash_iterate(iter, &key, &value)) {
struct duplicate *d = value;
diff --git a/src/dict/dict-server.c b/src/dict/dict-server.c
index 1fa9609..6acf34b 100644
--- a/src/dict/dict-server.c
+++ b/src/dict/dict-server.c
@@ -446,9 +446,9 @@ dict_client_connection_init(struct dict_server *server, int fd)
conn = i_new(struct dict_client_connection, 1);
conn->server = server;
conn->fd = fd;
- conn->input = i_stream_create_file(fd, default_pool,
- DICT_CLIENT_MAX_LINE_LENGTH, FALSE);
- conn->output = o_stream_create_file(fd, default_pool, 128*1024, FALSE);
+ conn->input = i_stream_create_file(fd, DICT_CLIENT_MAX_LINE_LENGTH,
+ FALSE);
+ conn->output = o_stream_create_file(fd, 128*1024, FALSE);
conn->io = io_add(fd, IO_READ, dict_client_connection_input, conn);
return conn;
}
diff --git a/src/imap-login/client.c b/src/imap-login/client.c
index 6fa0e76..e984ce9 100644
--- a/src/imap-login/client.c
+++ b/src/imap-login/client.c
@@ -68,10 +68,8 @@ static void client_set_title(struct imap_client *client)
static void client_open_streams(struct imap_client *client, int fd)
{
- client->input = i_stream_create_file(fd, default_pool,
- MAX_INBUF_SIZE, FALSE);
- client->output = o_stream_create_file(fd, default_pool, MAX_OUTBUF_SIZE,
- FALSE);
+ client->input = i_stream_create_file(fd, MAX_INBUF_SIZE, FALSE);
+ client->output = o_stream_create_file(fd, MAX_OUTBUF_SIZE, FALSE);
client->parser = imap_parser_create(client->input, client->output,
MAX_IMAP_LINE);
}
diff --git a/src/imap/client.c b/src/imap/client.c
index 1036e9e..99df764 100644
--- a/src/imap/client.c
+++ b/src/imap/client.c
@@ -30,10 +30,9 @@ struct client *client_create(int fd_in, int fd_out,
client = i_new(struct client, 1);
client->fd_in = fd_in;
client->fd_out = fd_out;
- client->input = i_stream_create_file(fd_in, default_pool,
- imap_max_line_length, FALSE);
- client->output = o_stream_create_file(fd_out, default_pool,
- (size_t)-1, FALSE);
+ client->input =
+ i_stream_create_file(fd_in, imap_max_line_length, FALSE);
+ client->output = o_stream_create_file(fd_out, (size_t)-1, FALSE);
o_stream_set_flush_callback(client->output, _client_output, client);
diff --git a/src/imap/cmd-append.c b/src/imap/cmd-append.c
index ccbce56..2f1ec5f 100644
--- a/src/imap/cmd-append.c
+++ b/src/imap/cmd-append.c
@@ -186,7 +186,7 @@ static bool cmd_append_cancel(struct cmd_append_context *ctx, bool nonsync)
/* we have to read the nonsynced literal so we don't treat the message
data as commands. */
- ctx->input = i_stream_create_limit(default_pool, ctx->client->input,
+ ctx->input = i_stream_create_limit(ctx->client->input,
ctx->client->input->v_offset,
ctx->msg_size);
@@ -317,7 +317,7 @@ static bool cmd_append_continue_parsing(struct client_command_context *cmd)
}
/* save the mail */
- ctx->input = i_stream_create_limit(default_pool, client->input,
+ ctx->input = i_stream_create_limit(client->input,
client->input->v_offset,
ctx->msg_size);
ret = mailbox_save_init(ctx->t, flags, keywords,
diff --git a/src/imap/imap-fetch-body.c b/src/imap/imap-fetch-body.c
index 392b4a3..23712d0 100644
--- a/src/imap/imap-fetch-body.c
+++ b/src/imap/imap-fetch-body.c
@@ -280,7 +280,7 @@ static int fetch_stream(struct imap_fetch_context *ctx,
if (size->physical_size == size->virtual_size &&
ctx->cur_mail->has_no_nuls) {
/* no need to kludge with CRs, we can use sendfile() */
- input = i_stream_create_limit(default_pool, ctx->cur_input,
+ input = i_stream_create_limit(ctx->cur_input,
ctx->cur_input->v_offset,
ctx->cur_size);
i_stream_unref(&ctx->cur_input);
diff --git a/src/lib-auth/auth-server-connection.c b/src/lib-auth/auth-server-connection.c
index 3f5d5c0..3ba6111 100644
--- a/src/lib-auth/auth-server-connection.c
+++ b/src/lib-auth/auth-server-connection.c
@@ -226,10 +226,9 @@ auth_server_connection_new(struct auth_client *client, const char *path)
conn->path = p_strdup(pool, path);
conn->fd = fd;
conn->io = io_add(fd, IO_READ, auth_client_input, conn);
- conn->input = i_stream_create_file(fd, default_pool,
- AUTH_CLIENT_MAX_LINE_LENGTH, FALSE);
- conn->output = o_stream_create_file(fd, default_pool, (size_t)-1,
- FALSE);
+ conn->input = i_stream_create_file(fd, AUTH_CLIENT_MAX_LINE_LENGTH,
+ FALSE);
+ conn->output = o_stream_create_file(fd, (size_t)-1, FALSE);
conn->requests = hash_create(default_pool, pool, 100, NULL, NULL);
conn->auth_mechs_buf = buffer_create_dynamic(default_pool, 256);
diff --git a/src/lib-charset/charset-iconv.c b/src/lib-charset/charset-iconv.c
index 955cf5d..da6aa48 100644
--- a/src/lib-charset/charset-iconv.c
+++ b/src/lib-charset/charset-iconv.c
@@ -1,7 +1,8 @@
-/* Copyright (C) 2002 Timo Sirainen */
+/* Copyright (C) 2002-2007 Timo Sirainen */
#include "lib.h"
#include "buffer.h"
+#include "unichar.h"
#include "charset-utf8.h"
#ifdef HAVE_ICONV
@@ -11,31 +12,28 @@
struct charset_translation {
iconv_t cd;
+ enum charset_flags flags;
};
-struct charset_translation *charset_to_utf8_begin(const char *charset,
- bool *unknown_charset)
+int charset_to_utf8_begin(const char *charset, enum charset_flags flags,
+ struct charset_translation **t_r)
{
struct charset_translation *t;
iconv_t cd;
- if (unknown_charset != NULL)
- *unknown_charset = FALSE;
-
if (charset_is_utf8(charset))
cd = (iconv_t)-1;
else {
cd = iconv_open("UTF-8", charset);
- if (cd == (iconv_t)-1) {
- if (unknown_charset != NULL)
- *unknown_charset = TRUE;
- return NULL;
- }
+ if (cd == (iconv_t)-1)
+ return -1;
}
t = i_new(struct charset_translation, 1);
t->cd = cd;
- return t;
+ t->flags = flags;
+ *t_r = t;
+ return 0;
}
void charset_to_utf8_end(struct charset_translation **_t)
@@ -55,70 +53,90 @@ void charset_to_utf8_reset(struct charset_translation *t)
(void)iconv(t->cd, NULL, NULL, NULL, NULL);
}
-enum charset_result
-charset_to_ucase_utf8(struct charset_translation *t,
- const unsigned char *src, size_t *src_size,
- buffer_t *dest)
+static bool
+charset_to_utf8_try(struct charset_translation *t,
+ const unsigned char *src, size_t *src_size, buffer_t *dest,
+ enum charset_result *result)
{
ICONV_CONST char *ic_srcbuf;
- char *ic_destbuf;
- size_t srcleft, destpos, destleft, size;
- enum charset_result ret;
-
- destpos = buffer_get_used_size(dest);
- destleft = buffer_get_size(dest) - destpos;
+ char tmpbuf[8192], *ic_destbuf;
+ size_t srcleft, destleft;
+ bool dtcase = (t->flags & CHARSET_FLAG_DECOMP_TITLECASE) != 0;
+ bool ret = TRUE;
if (t->cd == (iconv_t)-1) {
/* no translation needed - just copy it to outbuf uppercased */
- if (*src_size > destleft)
- *src_size = destleft;
- _charset_utf8_ucase(src, *src_size, dest, destpos);
- return CHARSET_RET_OK;
+ *result = CHARSET_RET_OK;
+ if (!dtcase) {
+ buffer_append(dest, src, *src_size);
+ return TRUE;
+ }
+
+ if (uni_utf8_to_decomposed_titlecase(src, *src_size, dest) < 0)
+ *result = CHARSET_RET_INVALID_INPUT;
+ return TRUE;
+ }
+ if (!dtcase) {
+ destleft = buffer_get_size(dest) - dest->used;
+ if (destleft < *src_size) {
+ /* The buffer is most likely too small to hold the
+ output, so increase it at least to the input size. */
+ destleft = *src_size;
+ }
+ ic_destbuf = buffer_append_space_unsafe(dest, destleft);
+ } else {
+ destleft = sizeof(tmpbuf);
+ ic_destbuf = tmpbuf;
}
- size = destleft;
srcleft = *src_size;
ic_srcbuf = (ICONV_CONST char *) src;
- ic_destbuf = buffer_append_space_unsafe(dest, destleft);
if (iconv(t->cd, &ic_srcbuf, &srcleft,
&ic_destbuf, &destleft) != (size_t)-1)
- ret = CHARSET_RET_OK;
- else if (errno == E2BIG)
- ret = CHARSET_RET_OUTPUT_FULL;
- else if (errno == EINVAL)
- ret = CHARSET_RET_INCOMPLETE_INPUT;
+ *result = CHARSET_RET_OK;
+ else if (errno == E2BIG) {
+ /* set result just to avoid compiler warning */
+ *result = CHARSET_RET_INCOMPLETE_INPUT;
+ ret = FALSE;
+ } else if (errno == EINVAL)
+ *result = CHARSET_RET_INCOMPLETE_INPUT;
else {
/* should be EILSEQ */
- return CHARSET_RET_INVALID_INPUT;
+ *result = CHARSET_RET_INVALID_INPUT;
+ return TRUE;
}
- size -= destleft;
+ *src_size -= srcleft;
- /* give back the memory we didn't use */
- buffer_set_used_size(dest, buffer_get_used_size(dest) - destleft);
+ if (!dtcase) {
+ /* give back the memory we didn't use */
+ buffer_set_used_size(dest, dest->used - destleft);
+ } else {
+ size_t tmpsize = sizeof(tmpbuf) - destleft;
- *src_size -= srcleft;
- _charset_utf8_ucase((unsigned char *) ic_destbuf - size, size,
- dest, destpos);
+ /* we just converted data to UTF-8, it can't be invalid */
+ if (uni_utf8_to_decomposed_titlecase(tmpbuf, tmpsize, dest) < 0)
+ i_unreached();
+ }
return ret;
}
enum charset_result
-charset_to_ucase_utf8_full(struct charset_translation *t,
- const unsigned char *src, size_t *src_size,
- buffer_t *dest)
+charset_to_utf8(struct charset_translation *t,
+ const unsigned char *src, size_t *src_size, buffer_t *dest)
{
- enum charset_result ret;
+ enum charset_result result;
size_t pos, used, size;
+ bool ret;
for (pos = 0;;) {
size = *src_size - pos;
- ret = charset_to_ucase_utf8(t, src + pos, &size, dest);
+ ret = charset_to_utf8_try(t, src + pos, &size, dest, &result);
pos += size;
- if (ret != CHARSET_RET_OUTPUT_FULL) {
+ if (ret) {
*src_size = pos;
- return ret;
+ return result;
}
/* force buffer to grow */
@@ -129,89 +147,4 @@ charset_to_ucase_utf8_full(struct charset_translation *t,
}
}
-static const char *
-charset_to_utf8_string_int(const char *charset, bool *unknown_charset,
- const unsigned char *data, size_t size,
- size_t *utf8_size_r, bool ucase)
-{
- iconv_t cd;
- ICONV_CONST char *inbuf;
- char *outbuf, *outpos;
- size_t inleft, outleft, outsize, pos;
-
- if (charset == NULL || charset_is_utf8(charset)) {
- if (unknown_charset != NULL)
- *unknown_charset = FALSE;
-
- if (!ucase) {
- if (utf8_size_r != NULL)
- *utf8_size_r = size;
- return t_strndup(data, size);
- }
-
- return _charset_utf8_ucase_strdup(data, size, utf8_size_r);
- }
-
- cd = iconv_open("UTF-8", charset);
- if (cd == (iconv_t)-1) {
- if (unknown_charset != NULL)
- *unknown_charset = TRUE;
- return NULL;
- }
-
- if (unknown_charset != NULL)
- *unknown_charset = FALSE;
-
- inbuf = (ICONV_CONST char *) data;
- inleft = size;
-
- outsize = outleft = inleft * 2;
- outbuf = outpos = t_buffer_get(outsize + 1);
-
- while (iconv(cd, &inbuf, &inleft, &outpos, &outleft) == (size_t)-1) {
- if (errno != E2BIG) {
- /* invalid data */
- iconv_close(cd);
- return NULL;
- }
-
- /* output buffer too small, grow it */
- pos = outsize - outleft;
- outsize *= 2;
- outleft = outsize - pos;
-
- outbuf = t_buffer_reget(outbuf, outsize + 1);
- outpos = outbuf + pos;
- }
-
- if (utf8_size_r != NULL)
- *utf8_size_r = (size_t) (outpos - outbuf);
- *outpos++ = '\0';
- t_buffer_alloc((size_t) (outpos - outbuf));
-
- if (ucase)
- str_ucase(outbuf); /* FIXME: utf8 */
-
- iconv_close(cd);
- return outbuf;
-}
-
-const char *
-charset_to_utf8_string(const char *charset, bool *unknown_charset,
- const unsigned char *data, size_t size,
- size_t *utf8_size_r)
-{
- return charset_to_utf8_string_int(charset, unknown_charset,
- data, size, utf8_size_r, FALSE);
-}
-
-const char *
-charset_to_ucase_utf8_string(const char *charset, bool *unknown_charset,
- const unsigned char *data, size_t size,
- size_t *utf8_size_r)
-{
- return charset_to_utf8_string_int(charset, unknown_charset,
- data, size, utf8_size_r, TRUE);
-}
-
#endif
diff --git a/src/lib-charset/charset-utf8.c b/src/lib-charset/charset-utf8.c
index 3917ebb..71c6963 100644
--- a/src/lib-charset/charset-utf8.c
+++ b/src/lib-charset/charset-utf8.c
@@ -2,6 +2,7 @@
#include "lib.h"
#include "buffer.h"
+#include "unichar.h"
#include "charset-utf8.h"
#include <ctype.h>
@@ -14,59 +15,30 @@ bool charset_is_utf8(const char *charset)
strcasecmp(charset, "UTF8") == 0;
}
-void _charset_utf8_ucase(const unsigned char *src, size_t src_size,
- buffer_t *dest, size_t destpos)
-{
- char *destbuf;
- size_t i;
-
- destbuf = buffer_get_space_unsafe(dest, destpos, src_size);
- for (i = 0; i < src_size; i++)
- destbuf[i] = i_toupper(src[i]); /* FIXME: utf8 */
-}
-
-const char *_charset_utf8_ucase_strdup(const unsigned char *data, size_t size,
- size_t *utf8_size_r)
-{
- buffer_t *dest;
-
- dest = buffer_create_dynamic(pool_datastack_create(), size);
- _charset_utf8_ucase(data, size, dest, 0);
- if (utf8_size_r != NULL)
- *utf8_size_r = buffer_get_used_size(dest);
- buffer_append_c(dest, '\0');
- return buffer_free_without_data(dest);
-}
-
-
#ifndef HAVE_ICONV
-#include <ctype.h>
-
struct charset_translation {
- int dummy;
+ enum charset_flags flags;
};
-static struct charset_translation ascii_translation, utf8_translation;
+static struct charset_translation raw_translation = { 0 };
+static struct charset_translation tc_translation = {
+ CHARSET_FLAG_DECOMP_TITLECASE
+};
-struct charset_translation *charset_to_utf8_begin(const char *charset,
- bool *unknown_charset)
+int charset_to_utf8_begin(const char *charset, enum charset_flags flags,
+ struct charset_translation **t_r)
{
- if (unknown_charset != NULL)
- *unknown_charset = FALSE;
-
- if (strcasecmp(charset, "us-ascii") == 0 ||
- strcasecmp(charset, "ascii") == 0)
- return &ascii_translation;
-
- if (strcasecmp(charset, "UTF-8") == 0 ||
- strcasecmp(charset, "UTF8") == 0)
- return &utf8_translation;
+ if (charset_is_utf8(charset)) {
+ if ((flags & CHARSET_FLAG_DECOMP_TITLECASE) != 0)
+ *t_r = &tc_translation;
+ else
+ *t_r = &raw_translation;
+ return 0;
+ }
/* no support for charsets that need translation */
- if (unknown_charset != NULL)
- *unknown_charset = TRUE;
- return NULL;
+ return -1;
}
void charset_to_utf8_end(struct charset_translation **t __attr_unused__)
@@ -78,57 +50,16 @@ void charset_to_utf8_reset(struct charset_translation *t __attr_unused__)
}
enum charset_result
-charset_to_ucase_utf8(struct charset_translation *t __attr_unused__,
- const unsigned char *src, size_t *src_size,
- buffer_t *dest)
+charset_to_utf8(struct charset_translation *t,
+ const unsigned char *src, size_t *src_size, buffer_t *dest)
{
- size_t destpos, destleft;
-
- destpos = buffer_get_used_size(dest);
- destleft = buffer_get_size(dest) - destpos;
-
- /* no translation needed - just copy it to outbuf uppercased */
- if (*src_size > destleft)
- *src_size = destleft;
- _charset_utf8_ucase(src, *src_size, dest, destpos);
- return CHARSET_RET_OK;
-}
-
-const char *
-charset_to_utf8_string(const char *charset, bool *unknown_charset,
- const unsigned char *data, size_t size,
- size_t *utf8_size_r)
-{
- if (charset == NULL || strcasecmp(charset, "us-ascii") == 0 ||
- strcasecmp(charset, "ascii") == 0 ||
- strcasecmp(charset, "UTF-8") == 0 ||
- strcasecmp(charset, "UTF8") == 0) {
- if (unknown_charset != NULL)
- *unknown_charset = FALSE;
- if (utf8_size_r != NULL)
- *utf8_size_r = size;
- return t_strndup(data, size);
- } else {
- if (unknown_charset != NULL)
- *unknown_charset = TRUE;
- return NULL;
- }
-}
-
-const char *
-charset_to_ucase_utf8_string(const char *charset, bool *unknown_charset,
- const unsigned char *data, size_t size,
- size_t *utf8_size_r)
-{
- if (charset == NULL || charset_is_utf8(charset)) {
- if (unknown_charset != NULL)
- *unknown_charset = FALSE;
- return _charset_utf8_ucase_strdup(data, size, utf8_size_r);
- } else {
- if (unknown_charset != NULL)
- *unknown_charset = TRUE;
- return NULL;
+ if ((t->flags & CHARSET_FLAG_DECOMP_TITLECASE) == 0)
+ buffer_append(dest, src, *src_size);
+ else {
+ if (uni_utf8_to_decomposed_titlecase(src, *src_size, dest) < 0)
+ return CHARSET_RET_INVALID_INPUT;
}
+ return CHARSET_RET_OK;
}
#endif
diff --git a/src/lib-charset/charset-utf8.h b/src/lib-charset/charset-utf8.h
index 0476df9..c603edb 100644
--- a/src/lib-charset/charset-utf8.h
+++ b/src/lib-charset/charset-utf8.h
@@ -1,51 +1,32 @@
#ifndef __CHARSET_UTF8_H
#define __CHARSET_UTF8_H
+struct charset_translation;
+
+enum charset_flags {
+ /* Translate the output to decomposed titlecase */
+ CHARSET_FLAG_DECOMP_TITLECASE = 0x01
+};
+
enum charset_result {
CHARSET_RET_OK = 1,
- CHARSET_RET_OUTPUT_FULL = 0,
CHARSET_RET_INCOMPLETE_INPUT = -1,
CHARSET_RET_INVALID_INPUT = -2
};
-/* Begin translation to UTF-8. */
-struct charset_translation *charset_to_utf8_begin(const char *charset,
- bool *unknown_charset);
-
+/* Begin translation to UTF-8. Returns -1 if charset is unknown. */
+int charset_to_utf8_begin(const char *charset, enum charset_flags flags,
+ struct charset_translation **t_r);
void charset_to_utf8_end(struct charset_translation **t);
-
void charset_to_utf8_reset(struct charset_translation *t);
/* Returns TRUE if charset is UTF-8 or ASCII */
bool charset_is_utf8(const char *charset);
/* Translate src to UTF-8. src_size is updated to contain the number of
- characters actually translated from src. Note that dest buffer is used
- only up to its current size, for growing it automatically use
- charset_to_ucase_utf8_full(). */
-enum charset_result
-charset_to_ucase_utf8(struct charset_translation *t,
- const unsigned char *src, size_t *src_size,
- buffer_t *dest);
+ characters actually translated from src. */
enum charset_result
-charset_to_ucase_utf8_full(struct charset_translation *t,
- const unsigned char *src, size_t *src_size,
- buffer_t *dest);
-
-/* Simple wrappers for above functions. If utf8_size is non-NULL, it's set
- to same as strlen(returned data). */
-const char *
-charset_to_utf8_string(const char *charset, bool *unknown_charset,
- const unsigned char *data, size_t size,
- size_t *utf8_size_r);
-const char *
-charset_to_ucase_utf8_string(const char *charset, bool *unknown_charset,
- const unsigned char *data, size_t size,
- size_t *utf8_size_r);
-
-void _charset_utf8_ucase(const unsigned char *src, size_t src_size,
- buffer_t *dest, size_t destpos);
-const char *_charset_utf8_ucase_strdup(const unsigned char *data, size_t size,
- size_t *utf8_size_r);
+charset_to_utf8(struct charset_translation *t,
+ const unsigned char *src, size_t *src_size, buffer_t *dest);
#endif
diff --git a/src/lib-dict/dict-client.c b/src/lib-dict/dict-client.c
index 5064b6f..0fb8446 100644
--- a/src/lib-dict/dict-client.c
+++ b/src/lib-dict/dict-client.c
@@ -244,10 +244,8 @@ static int client_dict_connect(struct client_dict *dict)
/* Dictionary lookups are blocking */
net_set_nonblock(dict->fd, FALSE);
- dict->input = i_stream_create_file(dict->fd, default_pool,
- (size_t)-1, FALSE);
- dict->output = o_stream_create_file(dict->fd, default_pool,
- 4096, FALSE);
+ dict->input = i_stream_create_file(dict->fd, (size_t)-1, FALSE);
+ dict->output = o_stream_create_file(dict->fd, 4096, FALSE);
dict->transaction_id_counter = 0;
t_push();
diff --git a/src/lib-dict/dict-sql.c b/src/lib-dict/dict-sql.c
index 2386c13..c440026 100644
--- a/src/lib-dict/dict-sql.c
+++ b/src/lib-dict/dict-sql.c
@@ -48,7 +48,7 @@ static int sql_dict_read_config(struct sql_dict *dict, const char *path)
return -1;
}
- input = i_stream_create_file(fd, default_pool, (size_t)-1, FALSE);
+ input = i_stream_create_file(fd, (size_t)-1, FALSE);
while ((line = i_stream_read_next_line(input)) != NULL) {
while (*line == ' ') line++;
value = strchr(line, '=');
diff --git a/src/lib-imap/imap-base-subject.c b/src/lib-imap/imap-base-subject.c
index 1e41413..41806b2 100644
--- a/src/lib-imap/imap-base-subject.c
+++ b/src/lib-imap/imap-base-subject.c
@@ -9,39 +9,6 @@
#include "message-header-decode.h"
#include "imap-base-subject.h"
-static bool header_decode(const unsigned char *data, size_t size,
- const char *charset, void *context)
-{
- buffer_t *buf = context;
- struct charset_translation *t;
- unsigned char *buf_data;
- size_t pos, used_size;
-
- pos = buffer_get_used_size(buf);
- if (charset == NULL) {
- /* It's ASCII. */
- buffer_append(buf, data, size);
- } else {
- t = charset_to_utf8_begin(charset, NULL);
- if (t != NULL) {
- (void)charset_to_ucase_utf8(t, data, &size, buf);
- charset_to_utf8_end(&t);
- }
- }
-
- if (size > 0) {
- /* @UNSAFE: uppercase it. Current draft specifies that we
- should touch only ASCII. */
- buf_data = buffer_get_modifiable_data(buf, &used_size);
- for (; pos < used_size; pos++) {
- if (buf_data[pos] >= 'a' && buf_data[pos] <= 'z')
- buf_data[pos] = buf_data[pos] - 'a' + 'A';
- }
- }
-
- return TRUE;
-}
-
static void pack_whitespace(buffer_t *buf)
{
char *data, *dest;
@@ -246,8 +213,8 @@ const char *imap_get_base_subject_cased(pool_t pool, const char *subject,
/* (1) Convert any RFC 2047 encoded-words in the subject to
UTF-8. Convert all tabs and continuations to space.
Convert all multiple spaces to a single space. */
- message_header_decode((const unsigned char *)subject, subject_len,
- header_decode, buf);
+ message_header_decode_utf8((const unsigned char *)subject, subject_len,
+ buf, TRUE);
buffer_append_c(buf, '\0');
pack_whitespace(buf);
diff --git a/src/lib-imap/imap-bodystructure.c b/src/lib-imap/imap-bodystructure.c
index b346722..3eaed30 100644
--- a/src/lib-imap/imap-bodystructure.c
+++ b/src/lib-imap/imap-bodystructure.c
@@ -5,7 +5,6 @@
#include "istream.h"
#include "str.h"
#include "message-parser.h"
-#include "message-content-parser.h"
#include "rfc822-parser.h"
#include "imap-parser.h"
#include "imap-quote.h"
@@ -20,7 +19,6 @@
struct message_part_body_data {
pool_t pool;
- string_t *str; /* temporary */
const char *content_type, *content_subtype;
const char *content_type_params;
const char *content_transfer_encoding;
@@ -32,65 +30,111 @@ struct message_part_body_data {
const char *content_language;
struct message_part_envelope_data *envelope;
-
- unsigned int charset_found:1;
};
-static void parse_content_type(const unsigned char *value, size_t value_len,
- void *context)
+static void parse_content_type(struct message_part_body_data *data,
+ struct message_header_line *hdr)
{
- struct message_part_body_data *data = context;
- size_t i;
+ struct rfc822_parser_context parser;
+ const char *key, *value;
+ string_t *str;
+ unsigned int i;
+ bool charset_found = FALSE;
+
+ rfc822_parser_init(&parser, hdr->full_value, hdr->full_value_len, NULL);
+ (void)rfc822_skip_lwsp(&parser);
- for (i = 0; i < value_len; i++) {
- if (value[i] == '/')
+ str = t_str_new(256);
+ if (rfc822_parse_content_type(&parser, str) < 0)
+ return;
+
+ /* Save content type and subtype */
+ value = str_c(str);
+ for (i = 0; value[i] != '\0'; i++) {
+ if (value[i] == '/') {
+ data->content_subtype =
+ imap_quote(data->pool, str_data(str) + i + 1,
+ str_len(str) - (i + 1));
break;
+ }
}
+ data->content_type =
+ imap_quote(data->pool, str_data(str), i);
- if (i == value_len)
- data->content_type = imap_quote(data->pool, value, value_len);
- else {
- data->content_type = imap_quote(data->pool, value, i);
+ /* parse parameters and save them */
+ str_truncate(str, 0);
+ while (rfc822_parse_content_param(&parser, &key, &value) > 0) {
+ if (strcasecmp(key, "charset") == 0)
+ charset_found = TRUE;
- i++;
- data->content_subtype =
- imap_quote(data->pool, value+i, value_len-i);
+ str_append_c(str, ' ');
+ imap_quote_append_string(str, key, TRUE);
+ str_append_c(str, ' ');
+ imap_quote_append_string(str, value, TRUE);
+ }
+
+ if (!charset_found &&
+ strcasecmp(data->content_type, "\"text\"") == 0) {
+ /* set a default charset */
+ str_append_c(str, ' ');
+ str_append(str, DEFAULT_CHARSET);
+ }
+ if (str_len(str) > 0) {
+ data->content_type_params =
+ p_strdup(data->pool, str_c(str) + 1);
}
}
-static void parse_save_params_list(const unsigned char *name, size_t name_len,
- const unsigned char *value, size_t value_len,
- bool value_quoted __attr_unused__,
- void *context)
+static void parse_content_transfer_encoding(struct message_part_body_data *data,
+ struct message_header_line *hdr)
{
- struct message_part_body_data *data = context;
-
- if (str_len(data->str) != 0)
- str_append_c(data->str, ' ');
+ struct rfc822_parser_context parser;
+ string_t *str;
- if (name_len == 7 && memcasecmp(name, "charset", 7) == 0)
- data->charset_found = TRUE;
+ rfc822_parser_init(&parser, hdr->full_value, hdr->full_value_len, NULL);
+ (void)rfc822_skip_lwsp(&parser);
- imap_quote_append(data->str, name, name_len, TRUE);
- str_append_c(data->str, ' ');
- imap_quote_append(data->str, value, value_len, TRUE);
+ t_push();
+ str = t_str_new(256);
+ if (rfc822_parse_mime_token(&parser, str) >= 0) {
+ data->content_transfer_encoding =
+ imap_quote(data->pool, str_data(str), str_len(str));
+ }
+ t_pop();
}
-static void parse_content_transfer_encoding(const unsigned char *value,
- size_t value_len, void *context)
+static void parse_content_disposition(struct message_part_body_data *data,
+ struct message_header_line *hdr)
{
- struct message_part_body_data *data = context;
+ struct rfc822_parser_context parser;
+ const char *key, *value;
+ string_t *str;
- data->content_transfer_encoding =
- imap_quote(data->pool, value, value_len);
-}
+ rfc822_parser_init(&parser, hdr->full_value, hdr->full_value_len, NULL);
+ (void)rfc822_skip_lwsp(&parser);
-static void parse_content_disposition(const unsigned char *value,
- size_t value_len, void *context)
-{
- struct message_part_body_data *data = context;
+ t_push();
+ str = t_str_new(256);
+ if (rfc822_parse_mime_token(&parser, str) < 0) {
+ t_pop();
+ return;
+ }
+ data->content_disposition =
+ imap_quote(data->pool, str_data(str), str_len(str));
- data->content_disposition = imap_quote(data->pool, value, value_len);
+ /* parse parameters and save them */
+ str_truncate(str, 0);
+ while (rfc822_parse_content_param(&parser, &key, &value) > 0) {
+ str_append_c(str, ' ');
+ imap_quote_append_string(str, key, TRUE);
+ str_append_c(str, ' ');
+ imap_quote_append_string(str, value, TRUE);
+ }
+ if (str_len(str) > 0) {
+ data->content_disposition_params =
+ p_strdup(data->pool, str_c(str) + 1);
+ }
+ t_pop();
}
static void parse_content_language(const unsigned char *value, size_t value_len,
@@ -164,27 +208,13 @@ static void parse_content_header(struct message_part_body_data *d,
case 't':
case 'T':
if (strcasecmp(name, "Type") == 0 && d->content_type == NULL) {
- d->str = str_new(default_pool, 256);
- message_content_parse_header(value, value_len,
- parse_content_type,
- parse_save_params_list, d);
- if (!d->charset_found &&
- strncasecmp(d->content_type, "\"text\"", 6) == 0) {
- /* set a default charset */
- if (str_len(d->str) != 0)
- str_append_c(d->str, ' ');
- str_append(d->str, DEFAULT_CHARSET);
- }
- d->content_type_params =
- p_strdup_empty(pool, str_c(d->str));
- str_free(&d->str);
+ t_push();
+ parse_content_type(d, hdr);
+ t_pop();
}
if (strcasecmp(name, "Transfer-Encoding") == 0 &&
- d->content_transfer_encoding == NULL) {
- message_content_parse_header(value, value_len,
- parse_content_transfer_encoding,
- null_parse_content_param_callback, d);
- }
+ d->content_transfer_encoding == NULL)
+ parse_content_transfer_encoding(d, hdr);
break;
case 'l':
@@ -202,15 +232,8 @@ static void parse_content_header(struct message_part_body_data *d,
imap_quote(pool, value, value_len);
}
if (strcasecmp(name, "Disposition") == 0 &&
- d->content_disposition_params == NULL) {
- d->str = str_new(default_pool, 256);
- message_content_parse_header(value, value_len,
- parse_content_disposition,
- parse_save_params_list, d);
- d->content_disposition_params =
- p_strdup_empty(pool, str_c(d->str));
- str_free(&d->str);
- }
+ d->content_disposition_params == NULL)
+ parse_content_disposition(d, hdr);
break;
}
}
@@ -672,8 +695,7 @@ bool imap_body_parse_from_bodystructure(const char *bodystructure,
const struct imap_arg *args;
int ret;
- input = i_stream_create_from_data(pool_datastack_create(),
- bodystructure, strlen(bodystructure));
+ input = i_stream_create_from_data(bodystructure, strlen(bodystructure));
(void)i_stream_read(input);
parser = imap_parser_create(input, NULL, (size_t)-1);
diff --git a/src/lib-imap/imap-envelope.c b/src/lib-imap/imap-envelope.c
index f2c670b..892368d 100644
--- a/src/lib-imap/imap-envelope.c
+++ b/src/lib-imap/imap-envelope.c
@@ -380,8 +380,7 @@ bool imap_envelope_parse(const char *envelope, enum imap_envelope_field field,
i_assert(field < IMAP_ENVELOPE_FIELDS);
- input = i_stream_create_from_data(pool_datastack_create(), envelope,
- strlen(envelope));
+ input = i_stream_create_from_data(envelope, strlen(envelope));
parser = imap_parser_create(input, NULL, (size_t)-1);
(void)i_stream_read(input);
diff --git a/src/lib-index/mail-cache-compress.c b/src/lib-index/mail-cache-compress.c
index 7916142..fe15217 100644
--- a/src/lib-index/mail-cache-compress.c
+++ b/src/lib-index/mail-cache-compress.c
@@ -138,7 +138,7 @@ mail_cache_copy(struct mail_cache *cache, struct mail_index_transaction *trans,
}
cache_view = mail_cache_view_open(cache, view);
- output = o_stream_create_file(fd, default_pool, 0, FALSE);
+ output = o_stream_create_file(fd, 0, FALSE);
memset(&hdr, 0, sizeof(hdr));
hdr.version = MAIL_CACHE_VERSION;
diff --git a/src/lib-index/mailbox-list-index-sync.c b/src/lib-index/mailbox-list-index-sync.c
index e8a89f2..d586a42 100644
--- a/src/lib-index/mailbox-list-index-sync.c
+++ b/src/lib-index/mailbox-list-index-sync.c
@@ -850,8 +850,7 @@ mailbox_list_index_sync_write(struct mailbox_list_index_sync_ctx *ctx)
int ret = 0;
if (ctx->index->mmap_base == NULL) {
- ctx->output = o_stream_create_file(ctx->index->fd, default_pool,
- 0, FALSE);
+ ctx->output = o_stream_create_file(ctx->index->fd, 0, FALSE);
ctx->output_buf = buffer_create_dynamic(default_pool, 4096);
o_stream_seek(ctx->output, ctx->hdr.used_space);
}
diff --git a/src/lib-mail/Makefile.am b/src/lib-mail/Makefile.am
index 668dd5a..6cd4c87 100644
--- a/src/lib-mail/Makefile.am
+++ b/src/lib-mail/Makefile.am
@@ -7,7 +7,6 @@ AM_CPPFLAGS = \
libmail_a_SOURCES = \
istream-header-filter.c \
message-address.c \
- message-content-parser.c \
message-date.c \
message-decoder.c \
message-header-decode.c \
@@ -25,7 +24,6 @@ headers = \
istream-header-filter.h \
mail-types.h \
message-address.h \
- message-content-parser.h \
message-date.h \
message-decoder.h \
message-header-decode.h \
diff --git a/src/lib-mail/istream-header-filter.c b/src/lib-mail/istream-header-filter.c
index 402942b..9b14784 100644
--- a/src/lib-mail/istream-header-filter.c
+++ b/src/lib-mail/istream-header-filter.c
@@ -329,24 +329,22 @@ i_stream_create_header_filter(struct istream *input,
header_filter_callback *callback, void *context)
{
struct header_filter_istream *mstream;
- pool_t pool;
unsigned int i;
i_assert((flags & (HEADER_FILTER_INCLUDE|HEADER_FILTER_EXCLUDE)) != 0);
- pool = pool_alloconly_create("header filter stream", 4096);
- mstream = p_new(pool, struct header_filter_istream, 1);
- mstream->pool = pool;
+ mstream = i_new(struct header_filter_istream, 1);
+ mstream->pool = pool_alloconly_create("header filter stream", 4096);
mstream->input = input;
i_stream_ref(mstream->input);
mstream->headers = headers_count == 0 ? NULL :
- p_new(pool, const char *, headers_count);
+ p_new(mstream->pool, const char *, headers_count);
for (i = 0; i < headers_count; i++)
- mstream->headers[i] = p_strdup(pool, headers[i]);
+ mstream->headers[i] = p_strdup(mstream->pool, headers[i]);
mstream->headers_count = headers_count;
- mstream->hdr_buf = buffer_create_dynamic(pool, 1024);
+ mstream->hdr_buf = buffer_create_dynamic(mstream->pool, 1024);
mstream->callback = callback;
mstream->context = context;
@@ -366,5 +364,5 @@ i_stream_create_header_filter(struct istream *input,
mstream->istream.istream.blocking = input->blocking;
mstream->istream.istream.seekable = input->seekable;
- return _i_stream_create(&mstream->istream, pool, -1, 0);
+ return _i_stream_create(&mstream->istream, -1, 0);
}
diff --git a/src/lib-mail/message-content-parser.c b/src/lib-mail/message-content-parser.c
deleted file mode 100644
index 38cb876..0000000
--- a/src/lib-mail/message-content-parser.c
+++ /dev/null
@@ -1,78 +0,0 @@
-/* Copyright (C) 2002-2005 Timo Sirainen */
-
-#include "lib.h"
-#include "str.h"
-#include "rfc822-parser.h"
-#include "message-content-parser.h"
-
-parse_content_callback_t *null_parse_content_callback = NULL;
-parse_content_param_callback_t *null_parse_content_param_callback = NULL;
-
-void message_content_parse_header(const unsigned char *data, size_t size,
- parse_content_callback_t *callback,
- parse_content_param_callback_t *param_cb,
- void *context)
-{
- struct rfc822_parser_context parser;
- string_t *str;
- size_t key_len;
- bool quoted_string;
-
- rfc822_parser_init(&parser, data, size, NULL);
-
- t_push();
- str = t_str_new(256);
-
- /* get content type */
- (void)rfc822_skip_lwsp(&parser);
- if (rfc822_parse_mime_token(&parser, str) > 0) {
- if (*parser.data == '/') {
- parser.data++;
- str_append_c(str, '/');
- (void)rfc822_parse_mime_token(&parser, str);
- }
- }
-
- if (callback != NULL)
- callback(str_data(str), str_len(str), context);
-
- if (param_cb == NULL) {
- /* we don't care about parameters */
- t_pop();
- return;
- }
-
- while (parser.data != parser.end && *parser.data == ';') {
- parser.data++;
- (void)rfc822_skip_lwsp(&parser);
-
- str_truncate(str, 0);
- if (rfc822_parse_mime_token(&parser, str) <= 0)
- break;
-
- /* <token> "=" <token> | <quoted-string> */
- if (str_len(str) == 0 || *parser.data != '=')
- break;
- parser.data++;
- if (rfc822_skip_lwsp(&parser) <= 0)
- break;
-
- quoted_string = parser.data != parser.end &&
- *parser.data == '"';
- key_len = str_len(str);
- if (quoted_string) {
- if (rfc822_parse_quoted_string(&parser, str) < 0)
- break;
- } else {
- if (rfc822_parse_mime_token(&parser, str) < 0)
- break;
- }
-
- param_cb(str_data(str), key_len,
- str_data(str) + key_len, str_len(str) - key_len,
- quoted_string, context);
-
- str_truncate(str, 0);
- }
- t_pop();
-}
diff --git a/src/lib-mail/message-content-parser.h b/src/lib-mail/message-content-parser.h
deleted file mode 100644
index 018bf1e..0000000
--- a/src/lib-mail/message-content-parser.h
+++ /dev/null
@@ -1,21 +0,0 @@
-#ifndef __MESSAGE_CONTENT_PARSER_H
-#define __MESSAGE_CONTENT_PARSER_H
-
-/* NOTE: name and value aren't \0-terminated. */
-typedef void parse_content_callback_t(const unsigned char *value,
- size_t value_len, void *context);
-typedef void parse_content_param_callback_t(const unsigned char *name,
- size_t name_len,
- const unsigned char *value,
- size_t value_len,
- bool value_quoted, void *context);
-
-extern parse_content_callback_t *null_parse_content_callback;
-extern parse_content_param_callback_t *null_parse_content_param_callback;
-
-void message_content_parse_header(const unsigned char *data, size_t size,
- parse_content_callback_t *callback,
- parse_content_param_callback_t *param_cb,
- void *context);
-
-#endif
diff --git a/src/lib-mail/message-decoder.c b/src/lib-mail/message-decoder.c
index dbfae9c..5b23bb5 100644
--- a/src/lib-mail/message-decoder.c
+++ b/src/lib-mail/message-decoder.c
@@ -2,12 +2,13 @@
#include "lib.h"
#include "buffer.h"
-#include "strescape.h"
#include "base64.h"
+#include "str.h"
+#include "unichar.h"
#include "charset-utf8.h"
#include "quoted-printable.h"
+#include "rfc822-parser.h"
#include "message-parser.h"
-#include "message-content-parser.h"
#include "message-header-decode.h"
#include "message-decoder.h"
@@ -41,14 +42,16 @@ struct message_decoder_context {
char *content_charset;
enum content_type content_type;
+ unsigned int dtcase:1;
unsigned int charset_utf8:1;
};
-struct message_decoder_context *message_decoder_init_ucase(void)
+struct message_decoder_context *message_decoder_init(bool dtcase)
{
struct message_decoder_context *ctx;
ctx = i_new(struct message_decoder_context, 1);
+ ctx->dtcase = dtcase;
ctx->buf = buffer_create_dynamic(default_pool, 8192);
ctx->buf2 = buffer_create_dynamic(default_pool, 8192);
return ctx;
@@ -69,72 +72,68 @@ void message_decoder_deinit(struct message_decoder_context **_ctx)
i_free(ctx);
}
-static bool
-message_decode_header_callback(const unsigned char *data, size_t size,
- const char *charset, void *context)
+static void
+parse_content_transfer_encoding(struct message_decoder_context *ctx,
+ struct message_header_line *hdr)
{
- struct message_decoder_context *ctx = context;
- struct charset_translation *t;
- bool unknown_charset;
-
- if (charset == NULL || charset_is_utf8(charset)) {
- /* ASCII / UTF-8 */
- _charset_utf8_ucase(data, size, ctx->buf, ctx->buf->used);
- return TRUE;
- }
-
- t = charset_to_utf8_begin(charset, &unknown_charset);
- if (unknown_charset) {
- /* let's just ignore this part */
- return TRUE;
- }
-
- /* ignore any errors */
- (void)charset_to_ucase_utf8_full(t, data, &size, ctx->buf);
- charset_to_utf8_end(&t);
- return TRUE;
-}
+ struct rfc822_parser_context parser;
+ string_t *value;
-static void parse_content_encoding(const unsigned char *value, size_t value_len,
- void *context)
-{
- struct message_decoder_context *ctx = context;
+ t_push();
+ value = t_str_new(64);
+ rfc822_parser_init(&parser, hdr->full_value, hdr->full_value_len, NULL);
- ctx->content_type = CONTENT_TYPE_UNKNOWN;
+ (void)rfc822_skip_lwsp(&parser);
+ (void)rfc822_parse_mime_token(&parser, value);
- switch (value_len) {
+ switch (str_len(value)) {
case 4:
- if (memcasecmp(value, "7bit", 4) == 0 ||
- memcasecmp(value, "8bit", 4) == 0)
+ if (memcasecmp(str_data(value), "7bit", 4) == 0 ||
+ memcasecmp(str_data(value), "8bit", 4) == 0)
ctx->content_type = CONTENT_TYPE_BINARY;
break;
case 6:
- if (memcasecmp(value, "base64", 6) == 0)
+ if (memcasecmp(str_data(value), "base64", 6) == 0)
ctx->content_type = CONTENT_TYPE_BASE64;
- else if (memcasecmp(value, "binary", 6) == 0)
+ else if (memcasecmp(str_data(value), "binary", 6) == 0)
ctx->content_type = CONTENT_TYPE_BINARY;
break;
case 16:
- if (memcasecmp(value, "quoted-printable", 16) == 0)
+ if (memcasecmp(str_data(value), "quoted-printable", 16) == 0)
ctx->content_type = CONTENT_TYPE_QP;
break;
}
+ t_pop();
}
static void
-parse_content_type_param(const unsigned char *name, size_t name_len,
- const unsigned char *value, size_t value_len,
- bool value_quoted, void *context)
+parse_content_type(struct message_decoder_context *ctx,
+ struct message_header_line *hdr)
{
- struct message_decoder_context *ctx = context;
-
- if (name_len == 7 && memcasecmp(name, "charset", 7) == 0 &&
- ctx->content_charset == NULL) {
- ctx->content_charset = i_strndup(value, value_len);
- if (value_quoted) str_unescape(ctx->content_charset);
+ struct rfc822_parser_context parser;
+ const char *key, *value;
+ string_t *str;
+
+ if (ctx->content_charset != NULL)
+ return;
+
+ rfc822_parser_init(&parser, hdr->full_value, hdr->full_value_len, NULL);
+ (void)rfc822_skip_lwsp(&parser);
+ t_push();
+ str = t_str_new(64);
+ if (rfc822_parse_content_type(&parser, str) <= 0) {
+ t_pop();
+ return;
+ }
- ctx->charset_utf8 = charset_is_utf8(ctx->content_charset);
+ while (rfc822_parse_content_param(&parser, &key, &value) > 0) {
+ if (strcasecmp(key, "charset") == 0) {
+ ctx->content_charset = i_strdup(value);
+ ctx->charset_utf8 = charset_is_utf8(value);
+ break;
+ }
}
+ t_pop();
}
static bool message_decode_header(struct message_decoder_context *ctx,
@@ -149,37 +148,32 @@ static bool message_decode_header(struct message_decoder_context *ctx,
}
if (hdr->name_len == 12 &&
- strcasecmp(hdr->name, "Content-Type") == 0) {
- message_content_parse_header(hdr->full_value,
- hdr->full_value_len,
- null_parse_content_callback,
- parse_content_type_param, ctx);
- }
+ strcasecmp(hdr->name, "Content-Type") == 0)
+ parse_content_type(ctx, hdr);
if (hdr->name_len == 25 &&
- strcasecmp(hdr->name, "Content-Transfer-Encoding") == 0) {
- message_content_parse_header(hdr->full_value,
- hdr->full_value_len,
- parse_content_encoding,
- null_parse_content_param_callback,
- ctx);
- }
+ strcasecmp(hdr->name, "Content-Transfer-Encoding") == 0)
+ parse_content_transfer_encoding(ctx, hdr);
buffer_set_used_size(ctx->buf, 0);
- message_header_decode(hdr->full_value, hdr->full_value_len,
- message_decode_header_callback, ctx);
+ message_header_decode_utf8(hdr->full_value, hdr->full_value_len,
+ ctx->buf, ctx->dtcase);
value_len = ctx->buf->used;
- _charset_utf8_ucase((const unsigned char *)hdr->name, hdr->name_len,
- ctx->buf, ctx->buf->used);
- buffer_append_c(ctx->buf, '\0');
+ if (ctx->dtcase) {
+ (void)uni_utf8_to_decomposed_titlecase(hdr->name, hdr->name_len,
+ ctx->buf);
+ buffer_append_c(ctx->buf, '\0');
+ }
ctx->hdr = *hdr;
ctx->hdr.full_value = ctx->buf->data;
ctx->hdr.full_value_len = value_len;
ctx->hdr.value_len = 0;
- ctx->hdr.name = CONST_PTR_OFFSET(ctx->buf->data,
- ctx->hdr.full_value_len);
- ctx->hdr.name_len = ctx->buf->used - 1 - value_len;
+ if (ctx->dtcase) {
+ ctx->hdr.name = CONST_PTR_OFFSET(ctx->buf->data,
+ ctx->hdr.full_value_len);
+ ctx->hdr.name_len = ctx->buf->used - 1 - value_len;
+ }
output->hdr = &ctx->hdr;
return TRUE;
@@ -199,8 +193,7 @@ static void translation_buf_decode(struct message_decoder_context *ctx,
memcpy(trans_buf + ctx->translation_size, data, skip);
pos = *size;
- (void)charset_to_ucase_utf8_full(ctx->charset_trans,
- *data, &pos, ctx->buf2);
+ (void)charset_to_utf8(ctx->charset_trans, *data, &pos, ctx->buf2);
i_assert(pos > ctx->translation_size);
skip = (ctx->translation_size + skip) - pos;
@@ -219,14 +212,13 @@ static bool message_decode_body(struct message_decoder_context *ctx,
unsigned char new_buf[MAX_ENCODING_BUF_SIZE+1];
const unsigned char *data = NULL;
size_t pos, size = 0, skip = 0;
- bool unknown_charset;
int ret;
if (ctx->charset_trans == NULL && !ctx->charset_utf8) {
- ctx->charset_trans =
- charset_to_utf8_begin(ctx->content_charset != NULL ?
- ctx->content_charset : "UTF-8",
- &unknown_charset);
+ if (charset_to_utf8_begin(ctx->content_charset != NULL ?
+ ctx->content_charset : "UTF-8",
+ ctx->dtcase, &ctx->charset_trans) < 0)
+ ctx->charset_trans = NULL;
}
if (ctx->encoding_size != 0) {
@@ -304,10 +296,16 @@ static bool message_decode_body(struct message_decoder_context *ctx,
}
if (ctx->charset_utf8) {
- buffer_set_used_size(ctx->buf2, 0);
- _charset_utf8_ucase(data, size, ctx->buf2, ctx->buf2->used);
- output->data = ctx->buf2->data;
- output->size = ctx->buf2->used;
+ if (ctx->dtcase) {
+ buffer_set_used_size(ctx->buf2, 0);
+ (void)uni_utf8_to_decomposed_titlecase(data, size,
+ ctx->buf);
+ output->data = ctx->buf2->data;
+ output->size = ctx->buf2->used;
+ } else {
+ output->data = data;
+ output->size = size;
+ }
} else if (ctx->charset_trans == NULL) {
output->data = data;
output->size = size;
@@ -317,8 +315,8 @@ static bool message_decode_body(struct message_decoder_context *ctx,
translation_buf_decode(ctx, &data, &size);
pos = size;
- (void)charset_to_ucase_utf8_full(ctx->charset_trans,
- data, &pos, ctx->buf2);
+ (void)charset_to_utf8(ctx->charset_trans,
+ data, &pos, ctx->buf2);
if (pos != size) {
ctx->translation_size = size - pos;
i_assert(ctx->translation_size <=
diff --git a/src/lib-mail/message-decoder.h b/src/lib-mail/message-decoder.h
index 7fb6371..043cf3e 100644
--- a/src/lib-mail/message-decoder.h
+++ b/src/lib-mail/message-decoder.h
@@ -5,8 +5,9 @@ struct message_block;
/* Decode message's contents as UTF-8, both the headers and the MIME bodies.
The bodies are decoded from quoted-printable and base64 formats if needed.
- The data is returned uppercased. */
-struct message_decoder_context *message_decoder_init_ucase(void);
+ If dtcase=TRUE, the data is returned through
+ uni_utf8_to_decomposed_titlecase(). */
+struct message_decoder_context *message_decoder_init(bool dtcase);
void message_decoder_deinit(struct message_decoder_context **ctx);
/* Decode input and return decoded output. Headers are returned only in their
diff --git a/src/lib-mail/message-header-decode.c b/src/lib-mail/message-header-decode.c
index 19f5e2e..b8fb50a 100644
--- a/src/lib-mail/message-header-decode.c
+++ b/src/lib-mail/message-header-decode.c
@@ -3,6 +3,8 @@
#include "lib.h"
#include "base64.h"
#include "buffer.h"
+#include "unichar.h"
+#include "charset-utf8.h"
#include "quoted-printable.h"
#include "message-header-decode.h"
@@ -113,3 +115,58 @@ void message_header_decode(const unsigned char *data, size_t size,
}
t_pop();
}
+
+struct decode_utf8_context {
+ buffer_t *dest;
+ unsigned int changed:1;
+ unsigned int called:1;
+ unsigned int dtcase:1;
+};
+
+static bool
+decode_utf8_callback(const unsigned char *data, size_t size,
+ const char *charset, void *context)
+{
+ struct decode_utf8_context *ctx = context;
+ struct charset_translation *t;
+
+ /* one call with charset=NULL means nothing changed */
+ if (!ctx->called && charset == NULL)
+ ctx->called = TRUE;
+ else
+ ctx->changed = TRUE;
+
+ if (charset == NULL || charset_is_utf8(charset)) {
+ /* ASCII / UTF-8 */
+ if (ctx->dtcase) {
+ (void)uni_utf8_to_decomposed_titlecase(data, size,
+ ctx->dest);
+ } else {
+ buffer_append(ctx->dest, data, size);
+ }
+ return TRUE;
+ }
+
+ if (charset_to_utf8_begin(charset, ctx->dtcase, &t) < 0) {
+ /* let's just ignore this part */
+ return TRUE;
+ }
+
+ /* ignore any errors */
+ (void)charset_to_utf8(t, data, &size, ctx->dest);
+ charset_to_utf8_end(&t);
+ return TRUE;
+}
+
+bool message_header_decode_utf8(const unsigned char *data, size_t size,
+ buffer_t *dest, bool dtcase)
+{
+ struct decode_utf8_context ctx;
+ size_t used = dest->used;
+
+ memset(&ctx, 0, sizeof(ctx));
+ ctx.dest = dest;
+ ctx.dtcase = dtcase;
+ message_header_decode(data, size, decode_utf8_callback, &ctx);
+ return ctx.changed || (dest->used - used != size);
+}
diff --git a/src/lib-mail/message-header-decode.h b/src/lib-mail/message-header-decode.h
index 6a1e9f2..b765523 100644
--- a/src/lib-mail/message-header-decode.h
+++ b/src/lib-mail/message-header-decode.h
@@ -13,4 +13,10 @@ void message_header_decode(const unsigned char *data, size_t size,
message_header_decode_callback_t *callback,
void *context);
+/* Append decoded RFC2047 header as UTF-8 to given buffer. If dtcase=TRUE,
+ the header is appended through uni_utf8_to_decomposed_titlecase().
+ Returns TRUE if output changed in any way from input. */
+bool message_header_decode_utf8(const unsigned char *data, size_t size,
+ buffer_t *dest, bool dtcase);
+
#endif
diff --git a/src/lib-mail/message-parser.c b/src/lib-mail/message-parser.c
index 4443773..670be66 100644
--- a/src/lib-mail/message-parser.c
+++ b/src/lib-mail/message-parser.c
@@ -1,9 +1,9 @@
/* Copyright (C) 2002-2006 Timo Sirainen */
#include "lib.h"
+#include "str.h"
#include "istream.h"
-#include "strescape.h"
-#include "message-content-parser.h"
+#include "rfc822-parser.h"
#include "message-parser.h"
/* RFC-2046 requires boundaries are max. 70 chars + "--" prefix + "--" suffix.
@@ -388,49 +388,46 @@ static int parse_next_body_to_eof(struct message_parser_ctx *ctx,
return 1;
}
-static void
-parse_content_type(const unsigned char *value, size_t value_len, void *context)
+static void parse_content_type(struct message_parser_ctx *ctx,
+ struct message_header_line *hdr)
{
- struct message_parser_ctx *ctx = context;
- const char *str;
+ struct rfc822_parser_context parser;
+ const char *key, *value;
+ string_t *content_type;
- if (ctx->part_seen_content_type || value_len == 0)
+ if (ctx->part_seen_content_type)
return;
ctx->part_seen_content_type = TRUE;
- t_push();
- str = t_strndup(value, value_len);
- if (strcasecmp(str, "message/rfc822") == 0)
+ rfc822_parser_init(&parser, hdr->full_value, hdr->full_value_len, NULL);
+ (void)rfc822_skip_lwsp(&parser);
+
+ content_type = t_str_new(64);
+ if (rfc822_parse_content_type(&parser, content_type) < 0)
+ return;
+
+ if (strcasecmp(str_c(content_type), "message/rfc822") == 0)
ctx->part->flags |= MESSAGE_PART_FLAG_MESSAGE_RFC822;
- else if (strncasecmp(str, "text", 4) == 0 &&
- (str[4] == '/' || str[4] == '\0'))
+ else if (strncasecmp(str_c(content_type), "text", 4) == 0 &&
+ (str_len(content_type) == 4 ||
+ str_data(content_type)[4] == '/'))
ctx->part->flags |= MESSAGE_PART_FLAG_TEXT;
- else if (strncasecmp(str, "multipart/", 10) == 0) {
+ else if (strncasecmp(str_c(content_type), "multipart/", 10) == 0) {
ctx->part->flags |= MESSAGE_PART_FLAG_MULTIPART;
- if (strcasecmp(str+10, "digest") == 0)
+ if (strcasecmp(str_c(content_type)+10, "digest") == 0)
ctx->part->flags |= MESSAGE_PART_FLAG_MULTIPART_DIGEST;
}
- t_pop();
-}
-
-static void
-parse_content_type_param(const unsigned char *name, size_t name_len,
- const unsigned char *value, size_t value_len,
- bool value_quoted, void *context)
-{
- struct message_parser_ctx *ctx = context;
- char *boundary;
if ((ctx->part->flags & MESSAGE_PART_FLAG_MULTIPART) == 0 ||
- name_len != 8 || memcasecmp(name, "boundary", 8) != 0)
+ ctx->last_boundary != NULL)
return;
- if (ctx->last_boundary == NULL) {
- boundary = p_strndup(ctx->parser_pool, value, value_len);
- if (value_quoted)
- str_unescape(boundary);
- ctx->last_boundary = boundary;
+ while (rfc822_parse_content_param(&parser, &key, &value) > 0) {
+ if (strcasecmp(key, "boundary") == 0) {
+ ctx->last_boundary = p_strdup(ctx->parser_pool, value);
+ break;
+ }
}
}
@@ -463,10 +460,9 @@ static int parse_next_header(struct message_parser_ctx *ctx,
if (hdr->continues)
hdr->use_full_value = TRUE;
else {
- message_content_parse_header(hdr->full_value,
- hdr->full_value_len,
- parse_content_type,
- parse_content_type_param, ctx);
+ t_push();
+ parse_content_type(ctx, hdr);
+ t_pop();
}
}
diff --git a/src/lib-mail/message-search.c b/src/lib-mail/message-search.c
index db3b4da..c8be4f2 100644
--- a/src/lib-mail/message-search.c
+++ b/src/lib-mail/message-search.c
@@ -3,11 +3,12 @@
#include "lib.h"
#include "buffer.h"
#include "istream.h"
+#include "str.h"
#include "str-find.h"
#include "charset-utf8.h"
+#include "rfc822-parser.h"
#include "message-decoder.h"
#include "message-parser.h"
-#include "message-content-parser.h"
#include "message-search.h"
struct message_search_context {
@@ -25,45 +26,34 @@ struct message_search_context {
unsigned int content_type_text:1; /* text/any or message/any */
};
-static void parse_content_type(const unsigned char *value, size_t value_len,
- void *context)
-{
- struct message_search_context *ctx = context;
- const char *str;
-
- t_push();
- str = t_strndup(value, value_len);
- ctx->content_type_text =
- strncasecmp(str, "text/", 5) == 0 ||
- strncasecmp(str, "message/", 8) == 0;
- t_pop();
-}
-
int message_search_init(pool_t pool, const char *key, const char *charset,
enum message_search_flags flags,
struct message_search_context **ctx_r)
{
struct message_search_context *ctx;
- bool unknown_charset;
+ struct charset_translation *t;
+ string_t *key_utf8;
size_t key_len;
- /* get the key uppercased */
+ if (charset_to_utf8_begin(charset, TRUE, &t) < 0)
+ return 0;
+
t_push();
- key = charset_to_ucase_utf8_string(charset, &unknown_charset,
- (const unsigned char *)key,
- strlen(key), &key_len);
- if (key == NULL) {
+ key_utf8 = t_str_new(I_MAX(128, key_len*2));
+ key_len = strlen(key);
+ if (charset_to_utf8(t, (const unsigned char *)key, &key_len,
+ key_utf8) != CHARSET_RET_OK) {
t_pop();
- return unknown_charset ? 0 : -1;
+ return -1;
}
ctx = *ctx_r = p_new(pool, struct message_search_context, 1);
ctx->pool = pool;
- ctx->key = p_strdup(pool, key);
- ctx->key_len = key_len;
+ ctx->key = p_strdup(pool, str_c(key_utf8));
+ ctx->key_len = str_len(key_utf8);
ctx->key_charset = p_strdup(pool, charset);
ctx->flags = flags;
- ctx->decoder = message_decoder_init_ucase();
+ ctx->decoder = message_decoder_init(TRUE);
ctx->str_find_ctx = str_find_init(pool, ctx->key);
t_pop();
return 1;
@@ -81,6 +71,25 @@ void message_search_deinit(struct message_search_context **_ctx)
p_free(ctx->pool, ctx);
}
+static void parse_content_type(struct message_search_context *ctx,
+ struct message_header_line *hdr)
+{
+ struct rfc822_parser_context parser;
+ string_t *content_type;
+
+ t_push();
+ rfc822_parser_init(&parser, hdr->full_value, hdr->full_value_len, NULL);
+ (void)rfc822_skip_lwsp(&parser);
+
+ content_type = t_str_new(64);
+ if (rfc822_parse_content_type(&parser, content_type) >= 0) {
+ ctx->content_type_text =
+ strncasecmp(str_c(content_type), "text/", 5) == 0 ||
+ strncasecmp(str_c(content_type), "message/", 8) == 0;
+ }
+ t_pop();
+}
+
static void handle_header(struct message_search_context *ctx,
struct message_header_line *hdr)
{
@@ -90,9 +99,7 @@ static void handle_header(struct message_search_context *ctx,
hdr->use_full_value = TRUE;
return;
}
- message_content_parse_header(hdr->full_value,
- hdr->full_value_len,
- parse_content_type, NULL, ctx);
+ parse_content_type(ctx, hdr);
}
}
diff --git a/src/lib-mail/rfc822-parser.c b/src/lib-mail/rfc822-parser.c
index 2d2d4d6..94f075f 100644
--- a/src/lib-mail/rfc822-parser.c
+++ b/src/lib-mail/rfc822-parser.c
@@ -2,6 +2,7 @@
#include "lib.h"
#include "str.h"
+#include "strescape.h"
#include "rfc822-parser.h"
/*
@@ -308,7 +309,7 @@ rfc822_parse_domain_literal(struct rfc822_parser_context *ctx, string_t *str)
} else if (*ctx->data == ']') {
ctx->data++;
str_append_n(str, start, ctx->data - start);
- return ctx->data != ctx->end;
+ return rfc822_skip_lwsp(ctx);
}
}
@@ -329,13 +330,77 @@ int rfc822_parse_domain(struct rfc822_parser_context *ctx, string_t *str)
if (rfc822_skip_lwsp(ctx) <= 0)
return -1;
- if (*ctx->data == '[') {
- if (rfc822_parse_domain_literal(ctx, str) < 0)
- return -1;
+ if (*ctx->data == '[')
+ return rfc822_parse_domain_literal(ctx, str);
+ else
+ return rfc822_parse_dot_atom(ctx, str);
+}
+
+int rfc822_parse_content_type(struct rfc822_parser_context *ctx, string_t *str)
+{
+ if (rfc822_skip_lwsp(ctx) <= 0)
+ return -1;
+
+ /* get main type */
+ if (rfc822_parse_mime_token(ctx, str) <= 0)
+ return -1;
+
+ /* skip over "/" */
+ if (*ctx->data != '/')
+ return -1;
+ ctx->data++;
+ if (rfc822_skip_lwsp(ctx) <= 0)
+ return -1;
+ str_append_c(str, '/');
+
+ /* get subtype */
+ return rfc822_parse_mime_token(ctx, str);
+}
+
+int rfc822_parse_content_param(struct rfc822_parser_context *ctx,
+ const char **key_r, const char **value_r)
+{
+ string_t *tmp;
+ size_t value_pos;
+ int ret;
+
+ /* .. := *(";" parameter)
+ parameter := attribute "=" value
+ attribute := token
+ value := token / quoted-string
+ */
+ *key_r = NULL;
+ *value_r = NULL;
+
+ if (ctx->data == ctx->end)
+ return 0;
+ if (*ctx->data != ';')
+ return -1;
+ ctx->data++;
+
+ if (rfc822_skip_lwsp(ctx) <= 0)
+ return -1;
+
+ tmp = t_str_new(64);
+ if (rfc822_parse_mime_token(ctx, tmp) <= 0)
+ return -1;
+ str_append_c(tmp, '\0');
+ value_pos = str_len(tmp);
+
+ if (*ctx->data != '=')
+ return -1;
+ ctx->data++;
+
+ if ((ret = rfc822_skip_lwsp(ctx)) <= 0) {
+ /* broken / no value */
+ } else if (*ctx->data == '"') {
+ ret = rfc822_parse_quoted_string(ctx, tmp);
+ str_unescape(str_c_modifiable(tmp) + value_pos);
} else {
- if (rfc822_parse_dot_atom(ctx, str) < 0)
- return -1;
+ ret = rfc822_parse_mime_token(ctx, tmp);
}
- return ctx->data != ctx->end;
+ *key_r = str_c(tmp);
+ *value_r = *key_r + value_pos;
+ return ret < 0 ? -1 : 1;
}
diff --git a/src/lib-mail/rfc822-parser.h b/src/lib-mail/rfc822-parser.h
index ba5b148..0ec54f4 100644
--- a/src/lib-mail/rfc822-parser.h
+++ b/src/lib-mail/rfc822-parser.h
@@ -6,10 +6,17 @@ struct rfc822_parser_context {
string_t *last_comment;
};
+/* Parse given data using RFC 822 token parser. */
void rfc822_parser_init(struct rfc822_parser_context *ctx,
const unsigned char *data, size_t size,
string_t *last_comment);
+/* The functions below return 1 = more data available, 0 = no more data
+ available (but a value might have been returned now), -1 = invalid input.
+
+ LWSP is automatically skipped after value, but not before it. So typically
+ you begin with skipping LWSP and then start using the parse functions. */
+
/* Parse comment. Assumes parser's data points to '(' */
int rfc822_skip_comment(struct rfc822_parser_context *ctx);
/* Skip LWSP if there is any */
@@ -29,4 +36,12 @@ int rfc822_parse_phrase(struct rfc822_parser_context *ctx, string_t *str);
/* dot-atom / domain-literal */
int rfc822_parse_domain(struct rfc822_parser_context *ctx, string_t *str);
+/* Parse Content-Type header's type/subtype. */
+int rfc822_parse_content_type(struct rfc822_parser_context *ctx, string_t *str);
+/* For Content-Type style parameter parsing. Expect ";" key "=" value.
+ value is unescaped if needed. The returned strings are allocated from data
+ stack. Returns 1 = key/value set, 0 = no more data, -1 = invalid input. */
+int rfc822_parse_content_param(struct rfc822_parser_context *ctx,
+ const char **key_r, const char **value_r);
+
#endif
diff --git a/src/lib-settings/settings.c b/src/lib-settings/settings.c
index 9608c47..c9d45cb 100644
--- a/src/lib-settings/settings.c
+++ b/src/lib-settings/settings.c
@@ -1,6 +1,7 @@
-/* Copyright (C) 2002 Timo Sirainen */
+/* Copyright (C) 2002-2007 Timo Sirainen */
#include "lib.h"
+#include "str.h"
#include "istream.h"
#include "strescape.h"
#include "settings.h"
@@ -71,6 +72,7 @@ bool settings_read(const char *path, const char *section,
struct istream *input;
const char *errormsg, *next_section, *name;
char *line, *key, *p, quote;
+ string_t *full_line;
size_t len;
int fd, linenum, last_section_line = 0, skip, sections, root_section;
@@ -90,8 +92,9 @@ bool settings_read(const char *path, const char *section,
next_section = t_strcut(section, '/');
}
+ full_line = t_str_new(512);
linenum = 0; sections = 0; root_section = 0; errormsg = NULL;
- input = i_stream_create_file(fd, default_pool, 2048, TRUE);
+ input = i_stream_create_file(fd, 2048, TRUE);
for (;;) {
line = i_stream_read_next_line(input);
if (line == NULL) {
@@ -140,6 +143,17 @@ bool settings_read(const char *path, const char *section,
len--;
line[len] = '\0';
+ if (len > 0 && line[len-1] == '\\') {
+ /* continues in next line */
+ line[len-1] = '\0';
+ str_append(full_line, line);
+ continue;
+ }
+ if (str_len(full_line) > 0) {
+ str_append(full_line, line);
+ line = str_c_modifiable(full_line);
+ }
+
/* a) key = value
b) section_type [section_name] {
c) } */
@@ -246,6 +260,7 @@ bool settings_read(const char *path, const char *section,
path, linenum, errormsg);
break;
}
+ str_truncate(full_line, 0);
}
i_stream_destroy(&input);
diff --git a/src/lib-storage/index/cydir/cydir-mail.c b/src/lib-storage/index/cydir/cydir-mail.c
index db2eee7..d995615 100644
--- a/src/lib-storage/index/cydir/cydir-mail.c
+++ b/src/lib-storage/index/cydir/cydir-mail.c
@@ -84,7 +84,7 @@ static uoff_t cydir_mail_get_physical_size(struct mail *_mail)
return data->physical_size;
if (cydir_mail_stat(_mail, &st) < 0)
- return (time_t)-1;
+ return (uoff_t)-1;
data->physical_size = data->virtual_size = st.st_size;
index_mail_cache_add(mail, MAIL_CACHE_PHYSICAL_FULL_SIZE,
@@ -113,8 +113,7 @@ cydir_mail_get_stream(struct mail *_mail, struct message_size *hdr_size,
return NULL;
}
mail->data.stream =
- i_stream_create_file(fd, default_pool,
- MAIL_READ_BLOCK_SIZE, TRUE);
+ i_stream_create_file(fd, MAIL_READ_BLOCK_SIZE, TRUE);
}
return index_mail_init_stream(mail, hdr_size, body_size);
diff --git a/src/lib-storage/index/cydir/cydir-save.c b/src/lib-storage/index/cydir/cydir-save.c
index 53f9f7c..07b4321 100644
--- a/src/lib-storage/index/cydir/cydir-save.c
+++ b/src/lib-storage/index/cydir/cydir-save.c
@@ -85,8 +85,8 @@ int cydir_save_init(struct mailbox_transaction_context *_t,
path = cydir_get_save_path(ctx, ctx->mail_count);
ctx->fd = open(path, O_WRONLY | O_CREAT | O_EXCL, 0660);
if (ctx->fd != -1) {
- output = o_stream_create_file(ctx->fd, default_pool, 0, FALSE);
- ctx->output = o_stream_create_crlf(default_pool, output);
+ output = o_stream_create_file(ctx->fd, 0, FALSE);
+ ctx->output = o_stream_create_crlf(output);
o_stream_unref(&output);
if (received_date != (time_t)-1) {
diff --git a/src/lib-storage/index/cydir/cydir-storage.c b/src/lib-storage/index/cydir/cydir-storage.c
index 994dcc6..f5975b6 100644
--- a/src/lib-storage/index/cydir/cydir-storage.c
+++ b/src/lib-storage/index/cydir/cydir-storage.c
@@ -347,7 +347,8 @@ static void cydir_notify_changes(struct mailbox *box)
index_mailbox_check_add(&mbox->ibox, mbox->path);
}
-static int cydir_list_iter_is_mailbox(struct mailbox_list_iterate_context *ctx,
+static int cydir_list_iter_is_mailbox(struct mailbox_list_iterate_context *ctx
+ __attr_unused__,
const char *dir, const char *fname,
enum mailbox_list_file_type type,
enum mailbox_info_flags *flags)
@@ -359,9 +360,9 @@ static int cydir_list_iter_is_mailbox(struct mailbox_list_iterate_context *ctx,
/* try to avoid stat() with these checks */
if (type != MAILBOX_LIST_FILE_TYPE_DIR &&
type != MAILBOX_LIST_FILE_TYPE_SYMLINK &&
- type != MAILBOX_LIST_FILE_TYPE_UNKNOWN &&
- (ctx->flags & MAILBOX_LIST_ITER_RETURN_NO_FLAGS) != 0) {
+ type != MAILBOX_LIST_FILE_TYPE_UNKNOWN) {
/* it's a file */
+ *flags |= MAILBOX_NOSELECT | MAILBOX_NOINFERIORS;
return 0;
}
@@ -376,7 +377,8 @@ static int cydir_list_iter_is_mailbox(struct mailbox_list_iterate_context *ctx,
ret = 0;
}
} else {
- /* non-selectable, but may contain subdirs */
+ /* non-selectable. probably either access denied, or symlink
+ destination not found. don't bother logging errors. */
*flags |= MAILBOX_NOSELECT;
}
t_pop();
diff --git a/src/lib-storage/index/dbox/dbox-file.c b/src/lib-storage/index/dbox/dbox-file.c
index 7eba324..6d4ca35 100644
--- a/src/lib-storage/index/dbox/dbox-file.c
+++ b/src/lib-storage/index/dbox/dbox-file.c
@@ -132,8 +132,7 @@ int dbox_file_seek(struct dbox_mailbox *mbox, uint32_t file_seq, uoff_t offset,
}
mbox->file->input =
- i_stream_create_file(mbox->file->fd, default_pool,
- 65536, FALSE);
+ i_stream_create_file(mbox->file->fd, 65536, FALSE);
if (dbox_file_read_header(mbox, mbox->file) < 0)
return -1;
diff --git a/src/lib-storage/index/dbox/dbox-mail.c b/src/lib-storage/index/dbox/dbox-mail.c
index d59dd27..659b001 100644
--- a/src/lib-storage/index/dbox/dbox-mail.c
+++ b/src/lib-storage/index/dbox/dbox-mail.c
@@ -229,8 +229,7 @@ dbox_mail_get_stream(struct mail *_mail,
offset += mbox->file->mail_header_size;
mail->data.stream =
- i_stream_create_limit(default_pool, mbox->file->input,
- offset,
+ i_stream_create_limit(mbox->file->input, offset,
mbox->file->seeked_mail_size);
}
diff --git a/src/lib-storage/index/dbox/dbox-sync-expunge.c b/src/lib-storage/index/dbox/dbox-sync-expunge.c
index 2cf3ed5..b7eb66c 100644
--- a/src/lib-storage/index/dbox/dbox-sync-expunge.c
+++ b/src/lib-storage/index/dbox/dbox-sync-expunge.c
@@ -145,7 +145,7 @@ static int dbox_sync_expunge_copy(struct dbox_sync_context *ctx,
/* try again with another file name */
}
- output = o_stream_create_file(fd, default_pool, 0, FALSE);
+ output = o_stream_create_file(fd, 0, FALSE);
lock_path = file_dotlock_get_lock_path(dotlock);
memset(&dest_entry, 0, sizeof(dest_entry));
@@ -190,7 +190,7 @@ static int dbox_sync_expunge_copy(struct dbox_sync_context *ctx,
/* copy the mail */
full_size = mbox->file->mail_header_size +
mbox->file->seeked_mail_size;
- input = i_stream_create_limit(default_pool, mbox->file->input,
+ input = i_stream_create_limit(mbox->file->input,
mbox->file->seeked_offset,
full_size);
bytes = o_stream_send_istream(output, input);
diff --git a/src/lib-storage/index/dbox/dbox-uidlist.c b/src/lib-storage/index/dbox/dbox-uidlist.c
index 3d7e1ad..c9fd1d4 100644
--- a/src/lib-storage/index/dbox/dbox-uidlist.c
+++ b/src/lib-storage/index/dbox/dbox-uidlist.c
@@ -353,7 +353,7 @@ static int dbox_uidlist_read(struct dbox_uidlist *uidlist)
uidlist->ino = st.st_ino;
uidlist->mtime = st.st_mtime;
- input = i_stream_create_file(uidlist->fd, default_pool, 65536, FALSE);
+ input = i_stream_create_file(uidlist->fd, 65536, FALSE);
/* read header: <version> <uidvalidity> <next-uid>.
Note that <next-uid> may be updated by UID lines, so it can't be
@@ -559,7 +559,7 @@ static int dbox_uidlist_full_rewrite(struct dbox_uidlist *uidlist)
return 0;
}
- output = o_stream_create_file(uidlist->lock_fd, default_pool, 0, FALSE);
+ output = o_stream_create_file(uidlist->lock_fd, 0, FALSE);
t_push();
str = t_str_new(256);
@@ -739,7 +739,7 @@ static int dbox_uidlist_append_changes(struct dbox_uidlist_append_ctx *ctx)
"lseek(%s) failed: %m", ctx->uidlist->path);
return -1;
}
- output = o_stream_create_file(ctx->uidlist->fd, default_pool, 0, FALSE);
+ output = o_stream_create_file(ctx->uidlist->fd, 0, FALSE);
uid_start = ctx->uidlist->last_uid + 1;
@@ -967,9 +967,8 @@ dbox_file_append(struct dbox_uidlist_append_ctx *ctx,
file->path = i_strdup(path);
file->fd = fd;
- file->input = i_stream_create_file(file->fd, default_pool,
- 65536, FALSE);
- file->output = o_stream_create_file(file->fd, default_pool, 0, FALSE);
+ file->input = i_stream_create_file(file->fd, 65536, FALSE);
+ file->output = o_stream_create_file(file->fd, 0, FALSE);
if ((uoff_t)st->st_size < sizeof(struct dbox_file_header)) {
if (dbox_file_write_header(mbox, file) < 0) {
dbox_file_close(file);
@@ -1152,7 +1151,7 @@ int dbox_uidlist_append_locked(struct dbox_uidlist_append_ctx *ctx,
/* we'll always use CRLF linefeeds for mails (but not the header,
so don't do this before dbox_file_write_header()) */
- output = o_stream_create_crlf(default_pool, file->output);
+ output = o_stream_create_crlf(file->output);
o_stream_unref(&file->output);
file->output = output;
diff --git a/src/lib-storage/index/index-mail-headers.c b/src/lib-storage/index/index-mail-headers.c
index 81879c1..bcf69f1 100644
--- a/src/lib-storage/index/index-mail-headers.c
+++ b/src/lib-storage/index/index-mail-headers.c
@@ -7,6 +7,7 @@
#include "str.h"
#include "message-date.h"
#include "message-parser.h"
+#include "message-header-decode.h"
#include "istream-tee.h"
#include "istream-header-filter.h"
#include "imap-envelope.h"
@@ -349,9 +350,9 @@ struct istream *index_mail_cache_parse_init(struct mail *_mail,
i_assert(mail->data.parser_ctx == NULL);
- tee = tee_i_stream_create(input, default_pool);
- input = tee_i_stream_create_child(tee, default_pool);
- input2 = tee_i_stream_create_child(tee, default_pool);
+ tee = tee_i_stream_create(input);
+ input = tee_i_stream_create_child(tee);
+ input2 = tee_i_stream_create_child(tee);
index_mail_parse_header_init(mail, NULL);
mail->data.parser_ctx =
@@ -547,9 +548,9 @@ index_mail_get_parsed_header(struct index_mail *mail, unsigned int field_idx)
return array_idx(&header_values, 0);
}
-const char *const *index_mail_get_headers(struct mail *_mail, const char *field)
+static const char *const *
+index_mail_get_raw_headers(struct index_mail *mail, const char *field)
{
- struct index_mail *mail = (struct index_mail *)_mail;
const char *headers[2], *value;
struct mailbox_header_lookup_ctx *headers_ctx;
unsigned char *data;
@@ -620,11 +621,62 @@ const char *const *index_mail_get_headers(struct mail *_mail, const char *field)
return array_idx(&header_values, 0);
}
-const char *index_mail_get_first_header(struct mail *mail, const char *field)
+static const char *const *
+index_mail_headers_decode(struct index_mail *mail, const char *const *list,
+ unsigned int max_count)
+{
+ const char **decoded_list;
+ unsigned int i, count;
+ buffer_t *buf;
+
+ count = strarray_length(list);
+ if (count > max_count)
+ count = max_count;
+ decoded_list = p_new(mail->data_pool, const char *, count + 1);
+
+ t_push();
+ buf = buffer_create_dynamic(pool_datastack_create(), 512);
+
+ for (i = 0; i < count; i++) {
+ buffer_set_used_size(buf, 0);
+ if (!message_header_decode_utf8((const unsigned char *)list[i],
+ strlen(list[i]), buf, FALSE))
+ decoded_list[i] = list[i];
+ else {
+ decoded_list[i] = p_strndup(mail->data_pool,
+ buf->data, buf->used);
+ }
+ }
+ t_pop();
+ return decoded_list;
+}
+
+const char *const *index_mail_get_headers(struct mail *_mail, const char *field,
+ bool decode_to_utf8)
{
- const char *const *list = index_mail_get_headers(mail, field);
+ struct index_mail *mail = (struct index_mail *)_mail;
+ const char *const *list;
+
+ list = index_mail_get_raw_headers(mail, field);
+ if (!decode_to_utf8 || list == NULL || *list == NULL)
+ return list;
+
+ return index_mail_headers_decode(mail, list, (unsigned int)-1);
+}
+
+const char *index_mail_get_first_header(struct mail *_mail, const char *field,
+ bool decode_to_utf8)
+{
+ struct index_mail *mail = (struct index_mail *)_mail;
+ const char *const *list;
+
+ list = index_mail_get_raw_headers(mail, field);
+ if (list == NULL || *list == NULL)
+ return NULL;
- return list == NULL ? NULL : list[0];
+ if (decode_to_utf8)
+ list = index_mail_headers_decode(mail, list, 1);
+ return list[0];
}
static void header_cache_callback(struct message_header_line *hdr,
@@ -658,8 +710,7 @@ index_mail_get_header_stream(struct mail *_mail,
if (mail->data.filter_stream != NULL)
i_stream_destroy(&mail->data.filter_stream);
mail->data.filter_stream =
- i_stream_create_from_data(default_pool,
- str_data(dest),
+ i_stream_create_from_data(str_data(dest),
str_len(dest));
return mail->data.filter_stream;
}
diff --git a/src/lib-storage/index/index-mail.h b/src/lib-storage/index/index-mail.h
index acfdbfc..18eed3a 100644
--- a/src/lib-storage/index/index-mail.h
+++ b/src/lib-storage/index/index-mail.h
@@ -146,9 +146,11 @@ int index_mail_parse_headers(struct index_mail *mail,
struct mailbox_header_lookup_ctx *headers);
void index_mail_headers_get_envelope(struct index_mail *mail);
-const char *index_mail_get_first_header(struct mail *_mail, const char *field);
+const char *index_mail_get_first_header(struct mail *_mail, const char *field,
+ bool decode_to_utf8);
const char *const *
-index_mail_get_headers(struct mail *_mail, const char *field);
+index_mail_get_headers(struct mail *_mail, const char *field,
+ bool decode_to_utf8);
struct istream *
index_mail_get_header_stream(struct mail *_mail,
struct mailbox_header_lookup_ctx *headers);
diff --git a/src/lib-storage/index/index-sort.c b/src/lib-storage/index/index-sort.c
index 36eb2b3..19b286c 100644
--- a/src/lib-storage/index/index-sort.c
+++ b/src/lib-storage/index/index-sort.c
@@ -31,6 +31,7 @@
#include "array.h"
#include "bsearch-insert-pos.h"
#include "str.h"
+#include "unichar.h"
#include "message-address.h"
#include "imap-base-subject.h"
#include "index-storage.h"
@@ -157,7 +158,7 @@ static const char *get_first_mailbox(struct mail *mail, const char *header)
struct message_address *addr;
const char *str;
- str = mail_get_first_header(mail, header);
+ str = mail_get_first_header_utf8(mail, header);
if (str == NULL)
return "";
@@ -171,6 +172,7 @@ static const char *
sort_header_get(enum mail_sort_type sort_type, struct mail *mail, uint32_t seq)
{
const char *str;
+ string_t *buf;
mail_set_seq(mail, seq);
switch (sort_type & MAIL_SORT_MASK) {
@@ -180,14 +182,21 @@ sort_header_get(enum mail_sort_type sort_type, struct mail *mail, uint32_t seq)
imap_get_base_subject_cased(pool_datastack_create(),
str, NULL);
case MAIL_SORT_CC:
- return get_first_mailbox(mail, "Cc");
+ str = get_first_mailbox(mail, "Cc");
+ break;
case MAIL_SORT_FROM:
- return get_first_mailbox(mail, "From");
+ str = get_first_mailbox(mail, "From");
+ break;
case MAIL_SORT_TO:
- return get_first_mailbox(mail, "To");
+ str = get_first_mailbox(mail, "To");
+ break;
default:
i_unreached();
}
+
+ buf = t_str_new(128);
+ (void)uni_utf8_to_decomposed_titlecase(str, (size_t)-1, buf);
+ return str_c(buf);
}
static int sort_node_cmp_type(struct sort_cmp_context *ctx,
@@ -213,7 +222,7 @@ static int sort_node_cmp_type(struct sort_cmp_context *ctx,
sort_header_get(sort_type, ctx->mail, n1->seq);
str2 = sort_header_get(sort_type, ctx->mail, n2->seq);
- ret = strcasecmp(str1, str2);
+ ret = strcmp(str1, str2);
t_pop();
break;
case MAIL_SORT_ARRIVAL:
@@ -358,9 +367,9 @@ index_sort_add_ids_range(struct mail_search_sort_program *program,
str = sort_header_get(program->sort_program[0], mail,
nodes[i].seq);
- if (i == idx2 && strcasecmp(str, last_str) == 0)
+ if (i == idx2 && strcmp(str, last_str) == 0)
nodes[i].sort_id = last_id;
- else if (strcasecmp(str, str_c(prev_str)) == 0 && prev_id != 0)
+ else if (strcmp(str, str_c(prev_str)) == 0 && prev_id != 0)
nodes[i].sort_id = prev_id;
else {
/* divide the available space so that each message gets
diff --git a/src/lib-storage/index/maildir/maildir-keywords.c b/src/lib-storage/index/maildir/maildir-keywords.c
index 8349741..e7194e1 100644
--- a/src/lib-storage/index/maildir/maildir-keywords.c
+++ b/src/lib-storage/index/maildir/maildir-keywords.c
@@ -133,7 +133,7 @@ static int maildir_keywords_sync(struct maildir_keywords *mk)
}
maildir_keywords_clear(mk);
- input = i_stream_create_file(fd, default_pool, 1024, FALSE);
+ input = i_stream_create_file(fd, 1024, FALSE);
while ((line = i_stream_read_next_line(input)) != NULL) {
p = strchr(line, ' ');
if (p == NULL) {
diff --git a/src/lib-storage/index/maildir/maildir-mail.c b/src/lib-storage/index/maildir/maildir-mail.c
index fd89bbd..2e397cf 100644
--- a/src/lib-storage/index/maildir/maildir-mail.c
+++ b/src/lib-storage/index/maildir/maildir-mail.c
@@ -62,8 +62,7 @@ maildir_open_mail(struct maildir_mailbox *mbox, struct mail *mail,
return NULL;
}
- return i_stream_create_file(fd, default_pool,
- MAIL_READ_BLOCK_SIZE, TRUE);
+ return i_stream_create_file(fd, MAIL_READ_BLOCK_SIZE, TRUE);
}
static int maildir_mail_stat(struct mail *mail, struct stat *st)
diff --git a/src/lib-storage/index/maildir/maildir-save.c b/src/lib-storage/index/maildir/maildir-save.c
index 3a36f82..4d03143 100644
--- a/src/lib-storage/index/maildir/maildir-save.c
+++ b/src/lib-storage/index/maildir/maildir-save.c
@@ -377,11 +377,10 @@ int maildir_save_init(struct mailbox_transaction_context *_t,
ctx->received_date = received_date;
ctx->input = input;
- output = o_stream_create_file(ctx->fd, system_pool, 0, FALSE);
+ output = o_stream_create_file(ctx->fd, 0, FALSE);
ctx->output = (ctx->mbox->storage->storage.flags &
MAIL_STORAGE_FLAG_SAVE_CRLF) != 0 ?
- o_stream_create_crlf(default_pool, output) :
- o_stream_create_lf(default_pool, output);
+ o_stream_create_crlf(output) : o_stream_create_lf(output);
o_stream_unref(&output);
flags &= ~MAIL_RECENT;
diff --git a/src/lib-storage/index/maildir/maildir-uidlist.c b/src/lib-storage/index/maildir/maildir-uidlist.c
index db87689..9bd8bf5 100644
--- a/src/lib-storage/index/maildir/maildir-uidlist.c
+++ b/src/lib-storage/index/maildir/maildir-uidlist.c
@@ -523,7 +523,7 @@ maildir_uidlist_update_read(struct maildir_uidlist *uidlist,
st.st_size/8));
}
- input = i_stream_create_file(fd, default_pool, 4096, FALSE);
+ input = i_stream_create_file(fd, 4096, FALSE);
i_stream_seek(input, uidlist->last_read_offset);
orig_next_uid = uidlist->next_uid;
@@ -804,7 +804,7 @@ static int maildir_uidlist_write_fd(struct maildir_uidlist *uidlist, int fd,
i_assert(fd != -1);
- output = o_stream_create_file(fd, default_pool, 0, FALSE);
+ output = o_stream_create_file(fd, 0, FALSE);
str = t_str_new(512);
if (output->offset == 0) {
diff --git a/src/lib-storage/index/mbox/istream-raw-mbox.c b/src/lib-storage/index/mbox/istream-raw-mbox.c
index 6cd8afd..5b57839 100644
--- a/src/lib-storage/index/mbox/istream-raw-mbox.c
+++ b/src/lib-storage/index/mbox/istream-raw-mbox.c
@@ -343,14 +343,14 @@ static const struct stat *_stat(struct _istream *stream, bool exact)
return &stream->statbuf;
}
-struct istream *i_stream_create_raw_mbox(pool_t pool, struct istream *input,
+struct istream *i_stream_create_raw_mbox(struct istream *input,
bool kludge_one_mail_only)
{
struct raw_mbox_istream *rstream;
i_stream_ref(input);
- rstream = p_new(pool, struct raw_mbox_istream, 1);
+ rstream = i_new(struct raw_mbox_istream, 1);
rstream->one_mail_only = kludge_one_mail_only;
rstream->input = input;
@@ -370,7 +370,7 @@ struct istream *i_stream_create_raw_mbox(pool_t pool, struct istream *input,
rstream->istream.istream.blocking = input->blocking;
rstream->istream.istream.seekable = input->seekable;
- return _i_stream_create(&rstream->istream, pool, -1,
+ return _i_stream_create(&rstream->istream, -1,
input->real_stream->abs_start_offset);
}
diff --git a/src/lib-storage/index/mbox/istream-raw-mbox.h b/src/lib-storage/index/mbox/istream-raw-mbox.h
index b6a74fa..582f812 100644
--- a/src/lib-storage/index/mbox/istream-raw-mbox.h
+++ b/src/lib-storage/index/mbox/istream-raw-mbox.h
@@ -3,7 +3,7 @@
/* Create a mbox stream for parsing mbox. Reading stops before From-line,
you'll have to call istream_raw_mbox_next() to get to next message. */
-struct istream *i_stream_create_raw_mbox(pool_t pool, struct istream *input,
+struct istream *i_stream_create_raw_mbox(struct istream *input,
bool kludge_one_mail_only);
/* Return offset to beginning of the "\nFrom"-line. */
diff --git a/src/lib-storage/index/mbox/mbox-file.c b/src/lib-storage/index/mbox/mbox-file.c
index cfbe626..f7166a9 100644
--- a/src/lib-storage/index/mbox/mbox-file.c
+++ b/src/lib-storage/index/mbox/mbox-file.c
@@ -73,8 +73,7 @@ int mbox_file_open_stream(struct mbox_mailbox *mbox)
i_assert(mbox->mbox_fd == -1 && mbox->mbox_readonly);
mbox->mbox_stream =
- i_stream_create_raw_mbox(default_pool,
- mbox->mbox_file_stream,
+ i_stream_create_raw_mbox(mbox->mbox_file_stream,
one_mail_only);
return 0;
}
@@ -84,18 +83,16 @@ int mbox_file_open_stream(struct mbox_mailbox *mbox)
return -1;
}
- if (mbox->mbox_writeonly) {
+ if (mbox->mbox_writeonly)
+ mbox->mbox_file_stream = i_stream_create_from_data(NULL, 0);
+ else {
mbox->mbox_file_stream =
- i_stream_create_from_data(default_pool, NULL, 0);
- } else {
- mbox->mbox_file_stream =
- i_stream_create_file(mbox->mbox_fd, default_pool,
+ i_stream_create_file(mbox->mbox_fd,
MAIL_READ_BLOCK_SIZE, FALSE);
}
mbox->mbox_stream =
- i_stream_create_raw_mbox(default_pool, mbox->mbox_file_stream,
- one_mail_only);
+ i_stream_create_raw_mbox(mbox->mbox_file_stream, one_mail_only);
return 0;
}
diff --git a/src/lib-storage/index/mbox/mbox-mail.c b/src/lib-storage/index/mbox/mbox-mail.c
index 79190d9..5fef2e9 100644
--- a/src/lib-storage/index/mbox/mbox-mail.c
+++ b/src/lib-storage/index/mbox/mbox-mail.c
@@ -209,7 +209,7 @@ static struct istream *mbox_mail_get_stream(struct mail *_mail,
raw_stream = mbox->mbox_stream;
offset = istream_raw_mbox_get_header_offset(raw_stream);
- raw_stream = i_stream_create_limit(default_pool, raw_stream,
+ raw_stream = i_stream_create_limit(raw_stream,
offset, (uoff_t)-1);
data->stream =
i_stream_create_header_filter(raw_stream,
diff --git a/src/lib-storage/index/mbox/mbox-save.c b/src/lib-storage/index/mbox/mbox-save.c
index 2ffd3b0..fa69025 100644
--- a/src/lib-storage/index/mbox/mbox-save.c
+++ b/src/lib-storage/index/mbox/mbox-save.c
@@ -321,8 +321,7 @@ mbox_save_init_file(struct mbox_save_context *ctx,
if (mbox_seek_to_end(ctx, &ctx->append_offset) < 0)
return -1;
- ctx->output = o_stream_create_file(mbox->mbox_fd, default_pool,
- 0, FALSE);
+ ctx->output = o_stream_create_file(mbox->mbox_fd, 0, FALSE);
}
return 0;
}
@@ -477,8 +476,8 @@ int mbox_save_init(struct mailbox_transaction_context *_t,
ctx->body_output =
(mbox->storage->storage.flags &
MAIL_STORAGE_FLAG_SAVE_CRLF) != 0 ?
- o_stream_create_crlf(default_pool, ctx->output) :
- o_stream_create_lf(default_pool, ctx->output);
+ o_stream_create_crlf(ctx->output) :
+ o_stream_create_lf(ctx->output);
if (ctx->mail != NULL) {
input = index_mail_cache_parse_init(ctx->mail,
ctx->input);
diff --git a/src/lib-storage/index/mbox/mbox-sync-rewrite.c b/src/lib-storage/index/mbox/mbox-sync-rewrite.c
index fa62858..3338bb7 100644
--- a/src/lib-storage/index/mbox/mbox-sync-rewrite.c
+++ b/src/lib-storage/index/mbox/mbox-sync-rewrite.c
@@ -26,8 +26,7 @@ int mbox_move(struct mbox_sync_context *sync_ctx,
i_stream_sync(sync_ctx->input);
- output = o_stream_create_file(sync_ctx->write_fd, default_pool,
- 4096, FALSE);
+ output = o_stream_create_file(sync_ctx->write_fd, 4096, FALSE);
i_stream_seek(sync_ctx->file_input, source);
if (o_stream_seek(output, dest) < 0) {
mbox_set_syscall_error(sync_ctx->mbox,
@@ -36,8 +35,7 @@ int mbox_move(struct mbox_sync_context *sync_ctx,
return -1;
}
- input = i_stream_create_limit(default_pool, sync_ctx->file_input,
- source, size);
+ input = i_stream_create_limit(sync_ctx->file_input, source, size);
ret = o_stream_send_istream(output, input);
i_stream_unref(&input);
diff --git a/src/lib-storage/list/subscription-file.c b/src/lib-storage/list/subscription-file.c
index 5aa1ddf..cc24560 100644
--- a/src/lib-storage/list/subscription-file.c
+++ b/src/lib-storage/list/subscription-file.c
@@ -16,11 +16,9 @@
#define SUBSCRIPTION_FILE_CHANGE_TIMEOUT 30
struct subsfile_list_context {
- pool_t pool;
-
struct mailbox_list *list;
struct istream *input;
- const char *path;
+ char *path;
bool failed;
};
@@ -115,10 +113,10 @@ int subsfile_set_subscribed(struct mailbox_list *list, const char *path,
}
input = fd_in == -1 ? NULL :
- i_stream_create_file(fd_in, default_pool,
- list->mailbox_name_max_length+1, TRUE);
- output = o_stream_create_file(fd_out, default_pool,
- list->mailbox_name_max_length+1, FALSE);
+ i_stream_create_file(fd_in, list->mailbox_name_max_length+1,
+ TRUE);
+ output = o_stream_create_file(fd_out, list->mailbox_name_max_length+1,
+ FALSE);
found = FALSE;
while ((line = next_line(list, path, input,
&failed, FALSE)) != NULL) {
@@ -176,14 +174,9 @@ struct subsfile_list_context *
subsfile_list_init(struct mailbox_list *list, const char *path)
{
struct subsfile_list_context *ctx;
- pool_t pool;
int fd;
- pool = pool_alloconly_create("subsfile_list",
- list->mailbox_name_max_length + 1024);
-
- ctx = p_new(pool, struct subsfile_list_context, 1);
- ctx->pool = pool;
+ ctx = i_new(struct subsfile_list_context, 1);
ctx->list = list;
fd = nfs_safe_open(path, O_RDONLY);
@@ -193,12 +186,10 @@ subsfile_list_init(struct mailbox_list *list, const char *path)
ctx->failed = TRUE;
}
} else {
- ctx->input =
- i_stream_create_file(fd, pool,
- list->mailbox_name_max_length+1,
- TRUE);
+ ctx->input = i_stream_create_file(fd,
+ list->mailbox_name_max_length+1, TRUE);
}
- ctx->path = p_strdup(pool, path);
+ ctx->path = i_strdup(path);
return ctx;
}
@@ -208,7 +199,8 @@ int subsfile_list_deinit(struct subsfile_list_context *ctx)
if (ctx->input != NULL)
i_stream_destroy(&ctx->input);
- pool_unref(ctx->pool);
+ i_free(ctx->path);
+ i_free(ctx);
return ret;
}
@@ -247,7 +239,7 @@ const char *subsfile_list_next(struct subsfile_list_context *ctx)
return NULL;
}
- ctx->input = i_stream_create_file(fd, ctx->pool,
+ ctx->input = i_stream_create_file(fd,
ctx->list->mailbox_name_max_length+1,
TRUE);
}
diff --git a/src/lib-storage/mail-storage-private.h b/src/lib-storage/mail-storage-private.h
index 9e81307..ec8ae3e 100644
--- a/src/lib-storage/mail-storage-private.h
+++ b/src/lib-storage/mail-storage-private.h
@@ -208,8 +208,10 @@ struct mail_vfuncs {
uoff_t (*get_virtual_size)(struct mail *mail);
uoff_t (*get_physical_size)(struct mail *mail);
- const char *(*get_first_header)(struct mail *mail, const char *field);
- const char *const *(*get_headers)(struct mail *mail, const char *field);
+ const char *(*get_first_header)(struct mail *mail, const char *field,
+ bool decode_to_utf8);
+ const char *const *(*get_headers)(struct mail *mail, const char *field,
+ bool decode_to_utf8);
struct istream *
(*get_header_stream)(struct mail *mail,
struct mailbox_header_lookup_ctx *headers);
diff --git a/src/lib-storage/mail-storage.h b/src/lib-storage/mail-storage.h
index 27918ad..2192203 100644
--- a/src/lib-storage/mail-storage.h
+++ b/src/lib-storage/mail-storage.h
@@ -424,8 +424,12 @@ uoff_t mail_get_physical_size(struct mail *mail);
/* Get value for single header field */
const char *mail_get_first_header(struct mail *mail, const char *field);
+/* Like mail_get_first_header(), but decode MIME encoded words to UTF-8 */
+const char *mail_get_first_header_utf8(struct mail *mail, const char *field);
/* Return a NULL-terminated list of values for each found field. */
const char *const *mail_get_headers(struct mail *mail, const char *field);
+/* Like mail_get_headers(), but decode MIME encoded words to UTF-8 */
+const char *const *mail_get_headers_utf8(struct mail *mail, const char *field);
/* Returns stream containing specified headers. */
struct istream *
mail_get_header_stream(struct mail *mail,
diff --git a/src/lib-storage/mail.c b/src/lib-storage/mail.c
index 61846a0..37566d7 100644
--- a/src/lib-storage/mail.c
+++ b/src/lib-storage/mail.c
@@ -93,14 +93,28 @@ const char *mail_get_first_header(struct mail *mail, const char *field)
{
struct mail_private *p = (struct mail_private *)mail;
- return p->v.get_first_header(mail, field);
+ return p->v.get_first_header(mail, field, FALSE);
+}
+
+const char *mail_get_first_header_utf8(struct mail *mail, const char *field)
+{
+ struct mail_private *p = (struct mail_private *)mail;
+
+ return p->v.get_first_header(mail, field, TRUE);
}
const char *const *mail_get_headers(struct mail *mail, const char *field)
{
struct mail_private *p = (struct mail_private *)mail;
- return p->v.get_headers(mail, field);
+ return p->v.get_headers(mail, field, FALSE);
+}
+
+const char *const *mail_get_headers_utf8(struct mail *mail, const char *field)
+{
+ struct mail_private *p = (struct mail_private *)mail;
+
+ return p->v.get_headers(mail, field, TRUE);
}
struct istream *
diff --git a/src/lib/Makefile.am b/src/lib/Makefile.am
index 059fa0a..385d065 100644
--- a/src/lib/Makefile.am
+++ b/src/lib/Makefile.am
@@ -1,5 +1,13 @@
noinst_LIBRARIES = liblib.a
+BUILT_SOURCES = unicodemap.c
+
+EXTRA_DIST = unicodemap.c unicodemap.pl
+
+unicodemap.c:
+ test -f UnicodeData.txt || wget http://www.unicode.org/Public/UNIDATA/UnicodeData.txt
+ perl unicodemap.pl < UnicodeData.txt > $@
+
liblib_a_SOURCES = \
backtrace-string.c \
base64.c \
diff --git a/src/lib/bsearch-insert-pos.h b/src/lib/bsearch-insert-pos.h
index 4558ba9..daaa101 100644
--- a/src/lib/bsearch-insert-pos.h
+++ b/src/lib/bsearch-insert-pos.h
@@ -1,6 +1,25 @@
#ifndef __BSEARCH_INSERT_POS
#define __BSEARCH_INSERT_POS
+/* Binary search template */
+#define BINARY_NUMBER_SEARCH(data, count, value, idx_r) \
+ unsigned int idx, left_idx, right_idx; \
+ \
+ idx = 0; left_idx = 0; right_idx = (count); \
+ while (left_idx < right_idx) { \
+ idx = (left_idx + right_idx) / 2; \
+ \
+ if ((data)[idx] < (value)) \
+ left_idx = idx+1; \
+ else if ((data)[idx] > (value)) \
+ right_idx = idx; \
+ else { \
+ *(idx_r) = idx; \
+ return TRUE; \
+ } \
+ } \
+ return FALSE
+
/* If key is found, returns TRUE and sets idx_r to the position where the key
was found. If key isn't found, returns FALSE and sets idx_r to the position
where the key should be inserted. */
diff --git a/src/lib/file-copy.c b/src/lib/file-copy.c
index 1c0ba00..b7b0256 100644
--- a/src/lib/file-copy.c
+++ b/src/lib/file-copy.c
@@ -68,8 +68,8 @@ static int file_copy_to_tmp(const char *srcpath, const char *tmppath,
/* try to change the group, don't really care if it fails */
(void)fchown(fd_out, (uid_t)-1, st.st_gid);
- input = i_stream_create_file(fd_in, default_pool, 0, FALSE);
- output = o_stream_create_file(fd_out, default_pool, 0, FALSE);
+ input = i_stream_create_file(fd_in, 0, FALSE);
+ output = o_stream_create_file(fd_out, 0, FALSE);
while ((ret = o_stream_send_istream(output, input)) > 0) ;
diff --git a/src/lib/iostream-internal.h b/src/lib/iostream-internal.h
index 16572f1..d8f7743 100644
--- a/src/lib/iostream-internal.h
+++ b/src/lib/iostream-internal.h
@@ -4,7 +4,6 @@
/* This file is private to input stream and output stream implementations */
struct _iostream {
- pool_t pool;
int refcount;
void (*close)(struct _iostream *stream);
@@ -12,7 +11,7 @@ struct _iostream {
void (*set_max_buffer_size)(struct _iostream *stream, size_t max_size);
};
-void _io_stream_init(pool_t pool, struct _iostream *stream);
+void _io_stream_init(struct _iostream *stream);
void _io_stream_ref(struct _iostream *stream);
void _io_stream_unref(struct _iostream *stream);
void _io_stream_close(struct _iostream *stream);
diff --git a/src/lib/iostream.c b/src/lib/iostream.c
index aafbed5..78aab85 100644
--- a/src/lib/iostream.c
+++ b/src/lib/iostream.c
@@ -3,10 +3,8 @@
#include "lib.h"
#include "iostream-internal.h"
-void _io_stream_init(pool_t pool, struct _iostream *stream)
+void _io_stream_init(struct _iostream *stream)
{
- pool_ref(pool);
- stream->pool = pool;
stream->refcount = 1;
}
@@ -17,8 +15,6 @@ void _io_stream_ref(struct _iostream *stream)
void _io_stream_unref(struct _iostream *stream)
{
- pool_t pool;
-
i_assert(stream->refcount > 0);
if (--stream->refcount != 0)
return;
@@ -26,9 +22,7 @@ void _io_stream_unref(struct _iostream *stream)
stream->close(stream);
stream->destroy(stream);
- pool = stream->pool;
- p_free(pool, stream);
- pool_unref(pool);
+ i_free(stream);
}
void _io_stream_close(struct _iostream *stream)
diff --git a/src/lib/istream-data.c b/src/lib/istream-data.c
index becbee3..7e98bb9 100644
--- a/src/lib/istream-data.c
+++ b/src/lib/istream-data.c
@@ -24,12 +24,11 @@ static void _seek(struct _istream *stream, uoff_t v_offset,
stream->istream.v_offset = v_offset;
}
-struct istream *i_stream_create_from_data(pool_t pool, const void *data,
- size_t size)
+struct istream *i_stream_create_from_data(const void *data, size_t size)
{
struct _istream *stream;
- stream = p_new(pool, struct _istream, 1);
+ stream = i_new(struct _istream, 1);
stream->buffer = data;
stream->pos = size;
@@ -41,7 +40,7 @@ struct istream *i_stream_create_from_data(pool_t pool, const void *data,
stream->istream.blocking = TRUE;
stream->istream.seekable = TRUE;
- (void)_i_stream_create(stream, pool, -1, 0);
+ (void)_i_stream_create(stream, -1, 0);
stream->statbuf.st_size = size;
return &stream->istream;
}
diff --git a/src/lib/istream-file.c b/src/lib/istream-file.c
index 49b5eaf..2d270d6 100644
--- a/src/lib/istream-file.c
+++ b/src/lib/istream-file.c
@@ -36,7 +36,7 @@ static void _destroy(struct _iostream *stream)
{
struct _istream *_stream = (struct _istream *) stream;
- p_free(_stream->iostream.pool, _stream->w_buffer);
+ i_free(_stream->w_buffer);
}
static ssize_t _read(struct _istream *stream)
@@ -159,13 +159,13 @@ _stat(struct _istream *stream, bool exact __attr_unused__)
return &stream->statbuf;
}
-struct istream *i_stream_create_file(int fd, pool_t pool,
- size_t max_buffer_size, bool autoclose_fd)
+struct istream *i_stream_create_file(int fd, size_t max_buffer_size,
+ bool autoclose_fd)
{
struct file_istream *fstream;
struct stat st;
- fstream = p_new(pool, struct file_istream, 1);
+ fstream = i_new(struct file_istream, 1);
fstream->autoclose_fd = autoclose_fd;
fstream->istream.iostream.close = _close;
@@ -184,5 +184,5 @@ struct istream *i_stream_create_file(int fd, pool_t pool,
fstream->istream.istream.seekable = TRUE;
}
- return _i_stream_create(&fstream->istream, pool, fd, 0);
+ return _i_stream_create(&fstream->istream, fd, 0);
}
diff --git a/src/lib/istream-internal.h b/src/lib/istream-internal.h
index 23ee63f..780cd5d 100644
--- a/src/lib/istream-internal.h
+++ b/src/lib/istream-internal.h
@@ -32,8 +32,8 @@ struct _istream {
string_t *line_str; /* for i_stream_next_line() if w_buffer == NULL */
};
-struct istream *_i_stream_create(struct _istream *_buf, pool_t pool, int fd,
- uoff_t abs_start_offset);
+struct istream *
+_i_stream_create(struct _istream *_buf, int fd, uoff_t abs_start_offset);
void _i_stream_compress(struct _istream *stream);
void _i_stream_grow_buffer(struct _istream *stream, size_t bytes);
diff --git a/src/lib/istream-limit.c b/src/lib/istream-limit.c
index 7e77751..e1fec24 100644
--- a/src/lib/istream-limit.c
+++ b/src/lib/istream-limit.c
@@ -108,14 +108,14 @@ static const struct stat *_stat(struct _istream *stream, bool exact)
return &stream->statbuf;
}
-struct istream *i_stream_create_limit(pool_t pool, struct istream *input,
+struct istream *i_stream_create_limit(struct istream *input,
uoff_t v_start_offset, uoff_t v_size)
{
struct limit_istream *lstream;
i_stream_ref(input);
- lstream = p_new(pool, struct limit_istream, 1);
+ lstream = i_new(struct limit_istream, 1);
lstream->input = input;
lstream->v_start_offset = v_start_offset;
lstream->v_size = v_size;
@@ -135,7 +135,7 @@ struct istream *i_stream_create_limit(pool_t pool, struct istream *input,
lstream->istream.istream.blocking = input->blocking;
lstream->istream.istream.seekable = input->seekable;
- return _i_stream_create(&lstream->istream, pool, i_stream_get_fd(input),
+ return _i_stream_create(&lstream->istream, i_stream_get_fd(input),
input->real_stream->abs_start_offset +
v_start_offset);
}
diff --git a/src/lib/istream-mmap.c b/src/lib/istream-mmap.c
index 53d47b4..55a378f 100644
--- a/src/lib/istream-mmap.c
+++ b/src/lib/istream-mmap.c
@@ -185,7 +185,7 @@ _stat(struct _istream *stream, bool exact __attr_unused__)
return &stream->statbuf;
}
-struct istream *i_stream_create_mmap(int fd, pool_t pool, size_t block_size,
+struct istream *i_stream_create_mmap(int fd, size_t block_size,
uoff_t start_offset, uoff_t v_size,
bool autoclose_fd)
{
@@ -207,7 +207,7 @@ struct istream *i_stream_create_mmap(int fd, pool_t pool, size_t block_size,
}
}
- mstream = p_new(pool, struct mmap_istream, 1);
+ mstream = i_new(struct mmap_istream, 1);
mstream->autoclose_fd = autoclose_fd;
mstream->v_size = v_size;
@@ -220,7 +220,7 @@ struct istream *i_stream_create_mmap(int fd, pool_t pool, size_t block_size,
mstream->istream.sync = _sync;
mstream->istream.stat = _stat;
- istream = _i_stream_create(&mstream->istream, pool, fd, start_offset);
+ istream = _i_stream_create(&mstream->istream, fd, start_offset);
istream->mmaped = TRUE;
istream->blocking = TRUE;
istream->seekable = TRUE;
diff --git a/src/lib/istream-seekable.c b/src/lib/istream-seekable.c
index b0a95a3..8c60bd1 100644
--- a/src/lib/istream-seekable.c
+++ b/src/lib/istream-seekable.c
@@ -17,7 +17,6 @@
struct seekable_istream {
struct _istream istream;
- pool_t pool;
size_t max_buffer_size;
char *temp_prefix;
@@ -54,8 +53,7 @@ static void _destroy(struct _iostream *stream)
for (i = 0; sstream->input[i] != NULL; i++)
i_stream_unref(&sstream->input[i]);
- p_free(sstream->pool, sstream->temp_prefix);
- pool_unref(sstream->pool);
+ i_free(sstream->temp_prefix);
}
static void _set_max_buffer_size(struct _iostream *stream, size_t max_size)
@@ -124,8 +122,7 @@ static int copy_to_temp_file(struct seekable_istream *sstream)
sstream->fd = fd;
sstream->fd_input =
- i_stream_create_file(fd, sstream->pool,
- sstream->max_buffer_size, TRUE);
+ i_stream_create_file(fd, sstream->max_buffer_size, TRUE);
return 0;
}
@@ -296,7 +293,7 @@ static const struct stat *_stat(struct _istream *stream, bool exact)
}
struct istream *
-i_stream_create_seekable(struct istream *input[], pool_t pool,
+i_stream_create_seekable(struct istream *input[],
size_t max_buffer_size, const char *temp_prefix)
{
struct seekable_istream *sstream;
@@ -308,14 +305,12 @@ i_stream_create_seekable(struct istream *input[], pool_t pool,
i_stream_ref(input[count]);
i_assert(count != 0);
- pool_ref(pool);
- sstream = p_new(pool, struct seekable_istream, 1);
- sstream->pool = pool;
- sstream->temp_prefix = p_strdup(pool, temp_prefix);
- sstream->buffer = buffer_create_dynamic(pool, BUF_INITIAL_SIZE);
+ sstream = i_new(struct seekable_istream, 1);
+ sstream->temp_prefix = i_strdup(temp_prefix);
+ sstream->buffer = buffer_create_dynamic(default_pool, BUF_INITIAL_SIZE);
sstream->max_buffer_size = max_buffer_size;
- sstream->input = p_new(pool, struct istream *, count + 1);
+ sstream->input = i_new(struct istream *, count + 1);
memcpy(sstream->input, input, sizeof(*input) * count);
sstream->cur_input = sstream->input[0];
@@ -332,5 +327,5 @@ i_stream_create_seekable(struct istream *input[], pool_t pool,
sstream->istream.seek = _seek;
sstream->istream.stat = _stat;
- return _i_stream_create(&sstream->istream, pool, -1, 0);
+ return _i_stream_create(&sstream->istream, -1, 0);
}
diff --git a/src/lib/istream-seekable.h b/src/lib/istream-seekable.h
index 2775a07..c9f2a52 100644
--- a/src/lib/istream-seekable.h
+++ b/src/lib/istream-seekable.h
@@ -7,7 +7,7 @@
temp_prefix is used as path and filename prefix for creating the file.
It will be appended by PID, timestamp and 128 bits of weak randomness. */
struct istream *
-i_stream_create_seekable(struct istream *input[], pool_t pool,
+i_stream_create_seekable(struct istream *input[],
size_t max_buffer_size, const char *temp_prefix);
#endif
diff --git a/src/lib/istream-tee.c b/src/lib/istream-tee.c
index 02cf133..af2f525 100644
--- a/src/lib/istream-tee.c
+++ b/src/lib/istream-tee.c
@@ -5,7 +5,6 @@
#include "istream-tee.h"
struct tee_istream {
- pool_t pool;
struct istream *input;
struct tee_child_istream *children;
@@ -88,7 +87,7 @@ static void _destroy(struct _iostream *stream)
tee->max_read_offset - tee->input->v_offset);
i_stream_unref(&tee->input);
- p_free(tee->pool, tee);
+ i_free(tee);
} else {
tee_streams_skip(tstream->tee);
}
@@ -170,24 +169,22 @@ static void _sync(struct _istream *stream)
return i_stream_sync(tstream->tee->input);
}
-struct tee_istream *tee_i_stream_create(struct istream *input, pool_t pool)
+struct tee_istream *tee_i_stream_create(struct istream *input)
{
struct tee_istream *tee;
- tee = p_new(pool, struct tee_istream, 1);
- tee->pool = pool;
+ tee = i_new(struct tee_istream, 1);
tee->input = input;
i_stream_ref(input);
return tee;
}
-struct istream *
-tee_i_stream_create_child(struct tee_istream *tee, pool_t pool)
+struct istream *tee_i_stream_create_child(struct tee_istream *tee)
{
struct tee_child_istream *tstream;
- tstream = p_new(pool, struct tee_child_istream, 1);
+ tstream = i_new(struct tee_child_istream, 1);
tstream->tee = tee;
tstream->istream.iostream.close = _close;
@@ -202,6 +199,6 @@ tee_i_stream_create_child(struct tee_istream *tee, pool_t pool)
tstream->next = tee->children;
tee->children = tstream;
- return _i_stream_create(&tstream->istream, pool,
+ return _i_stream_create(&tstream->istream,
i_stream_get_fd(tee->input), 0);
}
diff --git a/src/lib/istream-tee.h b/src/lib/istream-tee.h
index 0e814cf..0ceeb6c 100644
--- a/src/lib/istream-tee.h
+++ b/src/lib/istream-tee.h
@@ -7,9 +7,8 @@
If the stream's buffer gets full because some child isn't consuming the
data, other streams get returned 0 by i_stream_read(). */
-struct tee_istream *tee_i_stream_create(struct istream *input, pool_t pool);
+struct tee_istream *tee_i_stream_create(struct istream *input);
-struct istream *
-tee_i_stream_create_child(struct tee_istream *tee, pool_t pool);
+struct istream *tee_i_stream_create_child(struct tee_istream *tee);
#endif
diff --git a/src/lib/istream.c b/src/lib/istream.c
index 7b194c1..987a5fc 100644
--- a/src/lib/istream.c
+++ b/src/lib/istream.c
@@ -279,19 +279,15 @@ void _i_stream_grow_buffer(struct _istream *stream, size_t bytes)
stream->buffer_size = stream->pos + bytes;
if (stream->buffer_size <= I_STREAM_MIN_SIZE)
stream->buffer_size = I_STREAM_MIN_SIZE;
- else {
- stream->buffer_size =
- pool_get_exp_grown_size(stream->iostream.pool,
- old_size, stream->buffer_size);
- }
+ else
+ stream->buffer_size = nearest_power(stream->buffer_size);
if (stream->max_buffer_size > 0 &&
stream->buffer_size > stream->max_buffer_size)
stream->buffer_size = stream->max_buffer_size;
stream->buffer = stream->w_buffer =
- p_realloc(stream->iostream.pool, stream->w_buffer,
- old_size, stream->buffer_size);
+ i_realloc(stream->w_buffer, old_size, stream->buffer_size);
}
static void _set_max_buffer_size(struct _iostream *stream, size_t max_size)
@@ -307,8 +303,8 @@ _stat(struct _istream *stream, bool exact __attr_unused__)
return &stream->statbuf;
}
-struct istream *_i_stream_create(struct _istream *_stream, pool_t pool, int fd,
- uoff_t abs_start_offset)
+struct istream *
+_i_stream_create(struct _istream *_stream, int fd, uoff_t abs_start_offset)
{
_stream->fd = fd;
_stream->abs_start_offset = abs_start_offset;
@@ -325,7 +321,7 @@ struct istream *_i_stream_create(struct _istream *_stream, pool_t pool, int fd,
_stream->statbuf.st_mtime =
_stream->statbuf.st_ctime = ioloop_time;
- _io_stream_init(pool, &_stream->iostream);
+ _io_stream_init(&_stream->iostream);
return &_stream->istream;
}
@@ -371,7 +367,7 @@ int main(void)
write(fd1, buf, sizeof(buf));
/* test reading */
- input = i_stream_create_file(fd1, default_pool, 512, FALSE);
+ input = i_stream_create_file(fd1, 512, FALSE);
i_assert(i_stream_get_size(input) == sizeof(buf));
i_assert(i_stream_read_data(input, &data, &size, 0) > 0);
@@ -394,8 +390,8 @@ int main(void)
check_buffer(data, size, 900);
/* test moving data */
- output1 = o_stream_create_file(fd1, default_pool, 512, FALSE);
- output2 = o_stream_create_file(fd2, default_pool, 512, FALSE);
+ output1 = o_stream_create_file(fd1, 512, FALSE);
+ output2 = o_stream_create_file(fd2, 512, FALSE);
i_stream_seek(input, 1); size = sizeof(buf)-1;
i_assert(o_stream_send_istream(output2, input) == size);
@@ -410,8 +406,7 @@ int main(void)
i_assert(o_stream_send_istream(output1, input) == sizeof(buf));
/* test moving with limits */
- l_input = i_stream_create_limit(default_pool, input,
- sizeof(buf)/2, 512);
+ l_input = i_stream_create_limit(input, sizeof(buf)/2, 512);
i_stream_seek(l_input, 0);
o_stream_seek(output1, 10);
i_assert(o_stream_send_istream(output1, l_input) == 512);
diff --git a/src/lib/istream.h b/src/lib/istream.h
index a08af7d..bdc1d12 100644
--- a/src/lib/istream.h
+++ b/src/lib/istream.h
@@ -18,14 +18,13 @@ struct istream {
struct _istream *real_stream;
};
-struct istream *i_stream_create_file(int fd, pool_t pool,
- size_t max_buffer_size, bool autoclose_fd);
-struct istream *i_stream_create_mmap(int fd, pool_t pool, size_t block_size,
+struct istream *i_stream_create_file(int fd, size_t max_buffer_size,
+ bool autoclose_fd);
+struct istream *i_stream_create_mmap(int fd, size_t block_size,
uoff_t start_offset, uoff_t v_size,
bool autoclose_fd);
-struct istream *i_stream_create_from_data(pool_t pool, const void *data,
- size_t size);
-struct istream *i_stream_create_limit(pool_t pool, struct istream *input,
+struct istream *i_stream_create_from_data(const void *data, size_t size);
+struct istream *i_stream_create_limit(struct istream *input,
uoff_t v_start_offset, uoff_t v_size);
/* i_stream_close() + i_stream_unref() */
diff --git a/src/lib/macros.h b/src/lib/macros.h
index 1f0705a..42152ec 100644
--- a/src/lib/macros.h
+++ b/src/lib/macros.h
@@ -15,6 +15,9 @@
# define TRUE (!FALSE)
#endif
+#define N_ELEMENTS(arr) \
+ (sizeof(arr) / sizeof((arr)[0]))
+
#define BITS_IN_UINT (CHAR_BIT * sizeof(unsigned int))
#define BITS_IN_SIZE_T (CHAR_BIT * sizeof(size_t))
diff --git a/src/lib/ostream-crlf.c b/src/lib/ostream-crlf.c
index 9a11f94..1150a97 100644
--- a/src/lib/ostream-crlf.c
+++ b/src/lib/ostream-crlf.c
@@ -347,12 +347,11 @@ _send_istream(struct _ostream *outstream, struct istream *instream)
return sent == 0 && instream->stream_errno != 0 ? -1 : (ssize_t)sent;
}
-static struct crlf_ostream *
-o_stream_create_common(pool_t pool, struct ostream *output)
+static struct crlf_ostream *o_stream_create_common(struct ostream *output)
{
struct crlf_ostream *cstream;
- cstream = p_new(pool, struct crlf_ostream, 1);
+ cstream = i_new(struct crlf_ostream, 1);
cstream->output = output;
o_stream_ref(output);
@@ -369,20 +368,20 @@ o_stream_create_common(pool_t pool, struct ostream *output)
return cstream;
}
-struct ostream *o_stream_create_crlf(pool_t pool, struct ostream *output)
+struct ostream *o_stream_create_crlf(struct ostream *output)
{
struct crlf_ostream *cstream;
- cstream = o_stream_create_common(pool, output);
+ cstream = o_stream_create_common(output);
cstream->ostream.sendv = _sendv_crlf;
- return _o_stream_create(&cstream->ostream, pool);
+ return _o_stream_create(&cstream->ostream);
}
-struct ostream *o_stream_create_lf(pool_t pool, struct ostream *output)
+struct ostream *o_stream_create_lf(struct ostream *output)
{
struct crlf_ostream *cstream;
- cstream = o_stream_create_common(pool, output);
+ cstream = o_stream_create_common(output);
cstream->ostream.sendv = _sendv_lf;
- return _o_stream_create(&cstream->ostream, pool);
+ return _o_stream_create(&cstream->ostream);
}
diff --git a/src/lib/ostream-crlf.h b/src/lib/ostream-crlf.h
index b6171a1..7fa38d7 100644
--- a/src/lib/ostream-crlf.h
+++ b/src/lib/ostream-crlf.h
@@ -2,8 +2,8 @@
#define __OSTREAM_CRLF_H
/* Replace all plain LFs with CRLF. */
-struct ostream *o_stream_create_crlf(pool_t pool, struct ostream *output);
+struct ostream *o_stream_create_crlf(struct ostream *output);
/* Replace all CRLF pairs with plain LFs. */
-struct ostream *o_stream_create_lf(pool_t pool, struct ostream *output);
+struct ostream *o_stream_create_lf(struct ostream *output);
#endif
diff --git a/src/lib/ostream-file.c b/src/lib/ostream-file.c
index 47c8887..31a2c59 100644
--- a/src/lib/ostream-file.c
+++ b/src/lib/ostream-file.c
@@ -78,7 +78,7 @@ static void _destroy(struct _iostream *stream)
{
struct file_ostream *fstream = (struct file_ostream *)stream;
- p_free(fstream->ostream.iostream.pool, fstream->buffer);
+ i_free(fstream->buffer);
}
static void _set_max_buffer_size(struct _iostream *stream, size_t max_size)
@@ -322,9 +322,7 @@ static void o_stream_grow_buffer(struct file_ostream *fstream, size_t bytes)
{
size_t size, new_size, end_size;
- size = pool_get_exp_grown_size(fstream->ostream.iostream.pool,
- fstream->buffer_size,
- fstream->buffer_size + bytes);
+ size = nearest_power(fstream->buffer_size + bytes);
if (size > fstream->max_buffer_size) {
/* limit the size */
size = fstream->max_buffer_size;
@@ -339,8 +337,7 @@ static void o_stream_grow_buffer(struct file_ostream *fstream, size_t bytes)
if (size <= fstream->buffer_size)
return;
- fstream->buffer = p_realloc(fstream->ostream.iostream.pool,
- fstream->buffer,
+ fstream->buffer = i_realloc(fstream->buffer,
fstream->buffer_size, size);
if (fstream->tail <= fstream->head && !IS_STREAM_EMPTY(fstream)) {
@@ -737,15 +734,14 @@ static off_t _send_istream(struct _ostream *outstream, struct istream *instream)
}
struct ostream *
-o_stream_create_file(int fd, pool_t pool, size_t max_buffer_size,
- bool autoclose_fd)
+o_stream_create_file(int fd, size_t max_buffer_size, bool autoclose_fd)
{
struct file_ostream *fstream;
struct ostream *ostream;
struct stat st;
off_t offset;
- fstream = p_new(pool, struct file_ostream, 1);
+ fstream = i_new(struct file_ostream, 1);
fstream->fd = fd;
fstream->max_buffer_size = max_buffer_size;
fstream->autoclose_fd = autoclose_fd;
@@ -763,7 +759,7 @@ o_stream_create_file(int fd, pool_t pool, size_t max_buffer_size,
fstream->ostream.sendv = _sendv;
fstream->ostream.send_istream = _send_istream;
- ostream = _o_stream_create(&fstream->ostream, pool);
+ ostream = _o_stream_create(&fstream->ostream);
offset = lseek(fd, 0, SEEK_CUR);
if (offset >= 0) {
diff --git a/src/lib/ostream-internal.h b/src/lib/ostream-internal.h
index 30a6e8b..2c3ee09 100644
--- a/src/lib/ostream-internal.h
+++ b/src/lib/ostream-internal.h
@@ -26,6 +26,6 @@ struct _ostream {
void *context;
};
-struct ostream *_o_stream_create(struct _ostream *_stream, pool_t pool);
+struct ostream *_o_stream_create(struct _ostream *_stream);
#endif
diff --git a/src/lib/ostream.c b/src/lib/ostream.c
index c7a3750..8d686b4 100644
--- a/src/lib/ostream.c
+++ b/src/lib/ostream.c
@@ -157,10 +157,10 @@ off_t o_stream_send_istream(struct ostream *outstream,
return ret;
}
-struct ostream *_o_stream_create(struct _ostream *_stream, pool_t pool)
+struct ostream *_o_stream_create(struct _ostream *_stream)
{
_stream->ostream.real_stream = _stream;
- _io_stream_init(pool, &_stream->iostream);
+ _io_stream_init(&_stream->iostream);
return &_stream->ostream;
}
diff --git a/src/lib/ostream.h b/src/lib/ostream.h
index b81dca9..4917d90 100644
--- a/src/lib/ostream.h
+++ b/src/lib/ostream.h
@@ -24,8 +24,7 @@ typedef int stream_flush_callback_t(void *context);
/* Create new output stream from given file descriptor.
If max_buffer_size is 0, an "optimal" buffer size is used (max 128kB). */
struct ostream *
-o_stream_create_file(int fd, pool_t pool, size_t max_buffer_size,
- bool autoclose_fd);
+o_stream_create_file(int fd, size_t max_buffer_size, bool autoclose_fd);
/* o_stream_close() + o_stream_unref() */
void o_stream_destroy(struct ostream **stream);
diff --git a/src/lib/unichar.c b/src/lib/unichar.c
index fe54970..ea7ce80 100644
--- a/src/lib/unichar.c
+++ b/src/lib/unichar.c
@@ -2,8 +2,14 @@
#include "lib.h"
#include "buffer.h"
+#include "bsearch-insert-pos.h"
#include "unichar.h"
+#include "unicodemap.c"
+
+#define HANGUL_FIRST 0xac00
+#define HANGUL_LAST 0xd7a3
+
static const uint8_t utf8_non1_bytes[256 - 192 - 2] = {
2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,
3,3,3,3,3,3,3,3,3,3,3,3,3,3,3,3,4,4,4,4,4,4,4,4,5,5,5,5,6,6,1,1
@@ -164,3 +170,118 @@ unsigned int uni_utf8_strlen_n(const void *_input, size_t size)
}
return len;
}
+
+static bool uint16_find(const uint16_t *data, unsigned int count,
+ uint16_t value, unsigned int *idx_r)
+{
+ BINARY_NUMBER_SEARCH(data, count, value, idx_r);
+}
+
+static bool uint32_find(const uint32_t *data, unsigned int count,
+ uint32_t value, unsigned int *idx_r)
+{
+ BINARY_NUMBER_SEARCH(data, count, value, idx_r);
+}
+
+unichar_t uni_ucs4_to_titlecase(unichar_t chr)
+{
+ unsigned int idx;
+
+ if (chr <= 0xffff) {
+ if (!uint16_find(titlecase16_keys, N_ELEMENTS(titlecase16_keys),
+ chr, &idx))
+ return chr;
+ else
+ return titlecase16_values[idx];
+ } else {
+ if (!uint32_find(titlecase32_keys, N_ELEMENTS(titlecase32_keys),
+ chr, &idx))
+ return chr;
+ else
+ return titlecase32_values[idx];
+ }
+}
+
+static bool uni_ucs4_decompose_uni(unichar_t *chr)
+{
+ unsigned int idx;
+
+ if (*chr <= 0xffff) {
+ if (!uint16_find(uni16_decomp_keys,
+ N_ELEMENTS(uni16_decomp_keys),
+ *chr, &idx))
+ return FALSE;
+ *chr = uni16_decomp_values[idx];
+ } else {
+ if (!uint32_find(uni32_decomp_keys,
+ N_ELEMENTS(uni32_decomp_keys),
+ *chr, &idx))
+ return FALSE;
+ *chr = uni32_decomp_values[idx];
+ }
+ return TRUE;
+}
+
+static void uni_ucs4_decompose_hangul_utf8(unichar_t chr, buffer_t *output)
+{
+#define SBase HANGUL_FIRST
+#define LBase 0x1100
+#define VBase 0x1161
+#define TBase 0x11A7
+#define LCount 19
+#define VCount 21
+#define TCount 28
+#define NCount (VCount * TCount)
+ unsigned int SIndex = chr - SBase;
+ unichar_t L = LBase + SIndex / NCount;
+ unichar_t V = VBase + (SIndex % NCount) / TCount;
+ unichar_t T = TBase + SIndex % TCount;
+
+ uni_ucs4_to_utf8_c(L, output);
+ uni_ucs4_to_utf8_c(V, output);
+ if (T != TBase) uni_ucs4_to_utf8_c(T, output);
+}
+
+static bool uni_ucs4_decompose_multi_utf8(unichar_t chr, buffer_t *output)
+{
+ const uint16_t *value;
+ unsigned int idx;
+
+ if (chr > 0xffff)
+ return FALSE;
+
+ if (!uint16_find(multidecomp_keys, N_ELEMENTS(multidecomp_keys),
+ chr, &idx))
+ return FALSE;
+
+ value = &multidecomp_values[multidecomp_offsets[idx]];
+ for (; *value != 0; value++)
+ uni_ucs4_to_utf8_c(*value, output);
+ return TRUE;
+}
+
+int uni_utf8_to_decomposed_titlecase(const void *_input, size_t max_len,
+ buffer_t *output)
+{
+ const unsigned char *input = _input;
+ unsigned int bytes;
+ unichar_t chr;
+
+ while (max_len > 0 && *input != '\0') {
+ if (uni_utf8_get_char_n(input, max_len, &chr) <= 0) {
+ /* invalid input */
+ return -1;
+ }
+ bytes = uni_utf8_char_bytes(*input);
+ input += bytes;
+ max_len -= bytes;
+
+ chr = uni_ucs4_to_titlecase(chr);
+ if (chr >= HANGUL_FIRST && chr <= HANGUL_LAST)
+ uni_ucs4_decompose_hangul_utf8(chr, output);
+ else if (uni_ucs4_decompose_uni(&chr) ||
+ !uni_ucs4_decompose_multi_utf8(chr, output))
+ uni_ucs4_to_utf8_c(chr, output);
+ }
+ return 0;
+}
diff --git a/src/lib/unichar.h b/src/lib/unichar.h
index 78b8af5..412f061 100644
--- a/src/lib/unichar.h
+++ b/src/lib/unichar.h
@@ -31,4 +31,13 @@ static inline unsigned int uni_utf8_char_bytes(char chr)
return uni_utf8_non1_bytes[(uint8_t)chr - (192 + 2)];
}
+/* Return given character in titlecase. */
+unichar_t uni_ucs4_to_titlecase(unichar_t chr);
+
+/* Convert UTF-8 input to titlecase and decompose the titlecase characters to
+ output buffer. Returns 0 if ok, -1 if input was invalid. This generates
+ output that's compatible with i;unicode-casemap comparator. */
+int uni_utf8_to_decomposed_titlecase(const void *input, size_t max_len,
+ buffer_t *output);
+
#endif
diff --git a/src/lib/unicodemap.pl b/src/lib/unicodemap.pl
new file mode 100755
index 0000000..93bdfb3
--- /dev/null
+++ b/src/lib/unicodemap.pl
@@ -0,0 +1,134 @@
+#!/usr/bin/env perl
+use strict;
+
+my (@titlecase16_keys, @titlecase16_values);
+my (@titlecase32_keys, @titlecase32_values);
+my (@uni16_decomp_keys, @uni16_decomp_values);
+my (@uni32_decomp_keys, @uni32_decomp_values);
+my (@multidecomp_keys, @multidecomp_offsets, @multidecomp_values);
+while (<>) {
+ chomp $_;
+ my @arr = split(";");
+ my $code = eval("0x".$arr[0]);
+ my $decomp = $arr[5];
+ my $titlecode = $arr[14];
+
+ if ($titlecode ne "") {
+ # titlecase mapping
+ my $value = eval("0x$titlecode");
+ if ($value == $code) {
+ # the same character, ignore
+ } elsif ($code <= 0xffff && $value <= 0xffff) {
+ push @titlecase16_keys, $code;
+ push @titlecase16_values, $value;
+ } else {
+ push @titlecase32_keys, $code;
+ push @titlecase32_values, $value;
+ }
+ } elsif ($decomp =~ /\<[^>]*> (.+)/) {
+ # decompositions
+ my $decomp_codes = $1;
+ if ($decomp_codes =~ /^([0-9A-Z]*)$/i) {
+ # unicharacter decomposition. use separate lists for this
+ my $value = eval("0x$1");
+ if ($value > 0xffff) {
+ print STDERR "We've assumed decomposition codes are max. 16bit\n";
+ exit;
+ }
+ if ($code <= 0xffff) {
+ push @uni16_decomp_keys, $code;
+ push @uni16_decomp_values, $value;
+ } else {
+ push @uni32_decomp_keys, $code;
+ push @uni32_decomp_values, $value;
+ }
+ } else {
+ # multicharacter decomposition.
+ if ($code > 0xffff) {
+ print STDERR "We've assumed multi-decomposition key codes are max. 16bit\n";
+ exit;
+ }
+
+ push @multidecomp_keys, $code;
+ push @multidecomp_offsets, scalar(@multidecomp_values);
+
+ foreach my $dcode (split(" ", $decomp_codes)) {
+ my $value = eval("0x$dcode");
+ if ($value > 0xffff) {
+ print STDERR "We've assumed decomposition codes are max. 16bit\n";
+ exit;
+ }
+ push @multidecomp_values, $value;
+ }
+ push @multidecomp_values, 0;
+ }
+ }
+}
+
+sub print_list {
+ my @list = @{$_[0]};
+
+ my $last = $#list;
+ my $n = 0;
+ foreach my $key (@list) {
+ printf("0x%04x", $key);
+ last if ($n == $last);
+ print ",";
+
+ $n++;
+ if (($n % 8) == 0) {
+ print "\n\t";
+ } else {
+ print " ";
+ }
+ }
+}
+
+print "/* This file is automatically generated by unicodemap.pl from UnicodeData.txt
+
+ NOTE: decompositions for characters having titlecase characters
+ are not included, because we first translate everything to titlecase */\n";
+
+print "static uint16_t titlecase16_keys[] = {\n\t";
+print_list(\@titlecase16_keys);
+print "\n};\n";
+
+print "static uint16_t titlecase16_values[] = {\n\t";
+print_list(\@titlecase16_values);
+print "\n};\n";
+
+print "static uint32_t titlecase32_keys[] = {\n\t";
+print_list(\@titlecase32_keys);
+print "\n};\n";
+
+print "static uint32_t titlecase32_values[] = {\n\t";
+print_list(\@titlecase32_values);
+print "\n};\n";
+
+print "static uint16_t uni16_decomp_keys[] = {\n\t";
+print_list(\@uni16_decomp_keys);
+print "\n};\n";
+
+print "static uint16_t uni16_decomp_values[] = {\n\t";
+print_list(\@uni16_decomp_values);
+print "\n};\n";
+
+print "static uint32_t uni32_decomp_keys[] = {\n\t";
+print_list(\@uni32_decomp_keys);
+print "\n};\n";
+
+print "static uint16_t uni32_decomp_values[] = {\n\t";
+print_list(\@uni32_decomp_values);
+print "\n};\n";
+
+print "static uint16_t multidecomp_keys[] = {\n\t";
+print_list(\@multidecomp_keys);
+print "\n};\n";
+
+print "static uint16_t multidecomp_offsets[] = {\n\t";
+print_list(\@multidecomp_offsets);
+print "\n};\n";
+
+print "static uint16_t multidecomp_values[] = {\n\t";
+print_list(\@multidecomp_values);
+print "\n};\n";
diff --git a/src/login-common/login-proxy.c b/src/login-common/login-proxy.c
index b7e3420..22d3fdb 100644
--- a/src/login-common/login-proxy.c
+++ b/src/login-common/login-proxy.c
@@ -121,11 +121,10 @@ static void proxy_wait_connect(struct login_proxy *proxy)
/* connect successful */
proxy->server_input =
- i_stream_create_file(proxy->server_fd, default_pool,
- MAX_PROXY_INPUT_SIZE, FALSE);
+ i_stream_create_file(proxy->server_fd, MAX_PROXY_INPUT_SIZE,
+ FALSE);
proxy->server_output =
- o_stream_create_file(proxy->server_fd, default_pool,
- (size_t)-1, FALSE);
+ o_stream_create_file(proxy->server_fd, (size_t)-1, FALSE);
io_remove(&proxy->server_io);
proxy->server_io =
diff --git a/src/login-common/master.c b/src/login-common/master.c
index 9a0af95..f2ae8dd 100644
--- a/src/login-common/master.c
+++ b/src/login-common/master.c
@@ -165,7 +165,7 @@ static void master_read_env(int fd)
env_clean();
/* read environment variable lines until empty line comes */
- input = i_stream_create_file(fd, default_pool, 8192, FALSE);
+ input = i_stream_create_file(fd, 8192, FALSE);
do {
switch (i_stream_read(input)) {
case -1:
diff --git a/src/master/auth-process.c b/src/master/auth-process.c
index adeba9a..faa35b5 100644
--- a/src/master/auth-process.c
+++ b/src/master/auth-process.c
@@ -297,10 +297,8 @@ auth_process_new(pid_t pid, int fd, struct auth_process_group *group)
p->pid = pid;
p->fd = fd;
p->io = io_add(fd, IO_READ, auth_process_input, p);
- p->input = i_stream_create_file(fd, default_pool,
- MAX_INBUF_SIZE, FALSE);
- p->output = o_stream_create_file(fd, default_pool, MAX_OUTBUF_SIZE,
- FALSE);
+ p->input = i_stream_create_file(fd, MAX_INBUF_SIZE, FALSE);
+ p->output = o_stream_create_file(fd, MAX_OUTBUF_SIZE, FALSE);
p->requests = hash_create(default_pool, default_pool, 0, NULL, NULL);
group->process_count++;
diff --git a/src/master/log.c b/src/master/log.c
index a43b72c..9ecfb8f 100644
--- a/src/master/log.c
+++ b/src/master/log.c
@@ -179,7 +179,7 @@ int log_create_pipe(struct log_io **log_r, unsigned int max_lines_per_sec)
log_io = i_new(struct log_io, 1);
log_io->refcount = 1;
- log_io->stream = i_stream_create_file(fd[0], default_pool, 1024, TRUE);
+ log_io->stream = i_stream_create_file(fd[0], 1024, TRUE);
log_io->max_lines_per_sec =
max_lines_per_sec != 0 ? max_lines_per_sec : (unsigned int)-1;
diff --git a/src/master/login-process.c b/src/master/login-process.c
index 2b33625..46e0d5f 100644
--- a/src/master/login-process.c
+++ b/src/master/login-process.c
@@ -449,7 +449,7 @@ login_process_new(struct login_group *group, pid_t pid, int fd)
p->pid = pid;
p->fd = fd;
p->io = io_add(fd, IO_READ, login_process_input, p);
- p->output = o_stream_create_file(fd, default_pool,
+ p->output = o_stream_create_file(fd,
sizeof(struct master_login_reply)*10,
FALSE);
child_process_add(pid, &p->process);
diff --git a/src/master/mail-process.c b/src/master/mail-process.c
index e770314..3a695ce 100644
--- a/src/master/mail-process.c
+++ b/src/master/mail-process.c
@@ -124,6 +124,7 @@ static bool validate_uid_gid(struct settings *set, uid_t uid, gid_t gid,
i_error("Logins with login process UID %s (user %s) "
"not permitted (see login_user in config file).",
dec2str(uid), user);
+ return FALSE;
}
if (uid < (uid_t)set->first_valid_uid ||
diff --git a/src/plugins/acl/acl-backend-vfile-acllist.c b/src/plugins/acl/acl-backend-vfile-acllist.c
index 1b7f64b..41b0110 100644
--- a/src/plugins/acl/acl-backend-vfile-acllist.c
+++ b/src/plugins/acl/acl-backend-vfile-acllist.c
@@ -83,7 +83,7 @@ static int acl_backend_vfile_acllist_read(struct acl_backend_vfile *backend)
backend->acllist_mtime = st.st_mtime;
acllist_clear(backend, st.st_size);
- input = i_stream_create_file(fd, default_pool, (size_t)-1, FALSE);
+ input = i_stream_create_file(fd, (size_t)-1, FALSE);
while ((line = i_stream_read_next_line(input)) != NULL) {
acllist.mtime = 0;
for (p = line; *p >= '0' && *p <= '9'; p++)
@@ -208,7 +208,7 @@ int acl_backend_vfile_acllist_rebuild(struct acl_backend_vfile *backend)
fd = safe_mkstemp(path, mode, uid, gid);
if (fd == -1)
return -1;
- output = o_stream_create_file(fd, default_pool, 0, FALSE);
+ output = o_stream_create_file(fd, 0, FALSE);
ret = 0;
acllist_clear(backend, 0);
diff --git a/src/plugins/acl/acl-backend-vfile.c b/src/plugins/acl/acl-backend-vfile.c
index c1be2e3..2c784a7 100644
--- a/src/plugins/acl/acl-backend-vfile.c
+++ b/src/plugins/acl/acl-backend-vfile.c
@@ -345,7 +345,7 @@ acl_backend_vfile_read(struct acl_object_vfile *aclobj, const char *path,
if (aclobj->aclobj.backend->debug)
i_info("acl vfile: reading file %s", path);
- input = i_stream_create_file(fd, default_pool, 4096, FALSE);
+ input = i_stream_create_file(fd, 4096, FALSE);
if (!array_is_created(&aclobj->rights)) {
aclobj->rights_pool =
diff --git a/src/plugins/expire/auth-client.c b/src/plugins/expire/auth-client.c
index 9011564..9511888 100644
--- a/src/plugins/expire/auth-client.c
+++ b/src/plugins/expire/auth-client.c
@@ -46,10 +46,8 @@ static int auth_connection_connect(struct auth_connection *conn)
}
conn->fd = fd;
- conn->input =
- i_stream_create_file(fd, default_pool, MAX_INBUF_SIZE, FALSE);
- conn->output =
- o_stream_create_file(fd, default_pool, MAX_OUTBUF_SIZE, FALSE);
+ conn->input = i_stream_create_file(fd, MAX_INBUF_SIZE, FALSE);
+ conn->output = o_stream_create_file(fd, MAX_OUTBUF_SIZE, FALSE);
conn->io = io_add(fd, IO_READ, auth_input, conn);
o_stream_send_str(conn->output, "VERSION\t1\t0\n");
diff --git a/src/plugins/fts-squat/squat-test.c b/src/plugins/fts-squat/squat-test.c
index 61bed52..f520945 100644
--- a/src/plugins/fts-squat/squat-test.c
+++ b/src/plugins/fts-squat/squat-test.c
@@ -59,7 +59,7 @@ int main(int argc __attr_unused__, char *argv[])
return 1;
build_ctx = squat_trie_build_init(trie, &last_uid);
- input = i_stream_create_file(fd, default_pool, 0, FALSE);
+ input = i_stream_create_file(fd, 0, FALSE);
while ((line = i_stream_read_next_line(input)) != NULL) {
if (last != input->v_offset/(1024*100)) {
fprintf(stderr, "\r%ukB", (unsigned)(input->v_offset/1024));
diff --git a/src/plugins/fts-squat/squat-trie.c b/src/plugins/fts-squat/squat-trie.c
index 7c49497..0a4f466 100644
--- a/src/plugins/fts-squat/squat-trie.c
+++ b/src/plugins/fts-squat/squat-trie.c
@@ -1474,7 +1474,7 @@ trie_nodes_write(struct squat_trie_build_context *ctx, uint32_t *uidvalidity_r)
return -1;
}
- ctx->output = o_stream_create_file(trie->fd, default_pool, 0, FALSE);
+ ctx->output = o_stream_create_file(trie->fd, 0, FALSE);
o_stream_cork(ctx->output);
if (hdr.used_file_size == 0) {
o_stream_send(ctx->output, &hdr, sizeof(hdr));
@@ -1799,7 +1799,7 @@ static int squat_trie_compress_init(struct squat_trie_compress_context *ctx,
}
ctx->trie = trie;
- ctx->output = o_stream_create_file(ctx->fd, default_pool, 0, FALSE);
+ ctx->output = o_stream_create_file(ctx->fd, 0, FALSE);
ctx->node_count = trie->hdr->node_count;
/* write a dummy header first */
diff --git a/src/plugins/fts-squat/squat-uidlist.c b/src/plugins/fts-squat/squat-uidlist.c
index 6d4d56a..50e9fb1 100644
--- a/src/plugins/fts-squat/squat-uidlist.c
+++ b/src/plugins/fts-squat/squat-uidlist.c
@@ -580,8 +580,7 @@ static int squat_uidlist_write_init(struct squat_uidlist *uidlist)
return -1;
}
- uidlist->output = o_stream_create_file(uidlist->fd, default_pool,
- 0, FALSE);
+ uidlist->output = o_stream_create_file(uidlist->fd, 0, FALSE);
o_stream_cork(uidlist->output);
if (uidlist->hdr.used_file_size < sizeof(uidlist->hdr)) {
/* creating a new file, write a dummy header */
@@ -761,7 +760,7 @@ squat_uidlist_compress_begin(struct squat_uidlist *uidlist,
ctx->failed = TRUE;
i_error("open(%s) failed: %m", ctx->tmp_path);
} else {
- ctx->output = o_stream_create_file(fd, default_pool, 0, TRUE);
+ ctx->output = o_stream_create_file(fd, 0, TRUE);
o_stream_send(ctx->output, &ctx->hdr, sizeof(ctx->hdr));
}
diff --git a/src/plugins/fts/fts-storage.c b/src/plugins/fts/fts-storage.c
index 3cd1fab..a0f62e1 100644
--- a/src/plugins/fts/fts-storage.c
+++ b/src/plugins/fts/fts-storage.c
@@ -176,7 +176,7 @@ static int fts_build_mail(struct fts_storage_build_context *ctx)
parser = message_parser_init(pool_datastack_create(), input,
MESSAGE_HEADER_PARSER_FLAG_CLEAN_ONELINE,
0);
- decoder = message_decoder_init_ucase();
+ decoder = message_decoder_init(TRUE);
for (;;) {
ret = message_parser_parse_next_block(parser, &raw_block);
i_assert(ret != 0);
diff --git a/src/plugins/trash/trash-plugin.c b/src/plugins/trash/trash-plugin.c
index a5dcd02..e1c15ba 100644
--- a/src/plugins/trash/trash-plugin.c
+++ b/src/plugins/trash/trash-plugin.c
@@ -262,7 +262,7 @@ static int read_configuration(const char *path)
p_clear(config_pool);
p_array_init(&trash_boxes, config_pool, INIT_TRASH_MAILBOX_COUNT);
- input = i_stream_create_file(fd, default_pool, (size_t)-1, FALSE);
+ input = i_stream_create_file(fd, (size_t)-1, FALSE);
while ((line = i_stream_read_next_line(input)) != NULL) {
/* <priority> <mailbox name> */
name = strchr(line, ' ');
diff --git a/src/plugins/zlib/istream-zlib.c b/src/plugins/zlib/istream-zlib.c
index 06b7b75..c93bca7 100644
--- a/src/plugins/zlib/istream-zlib.c
+++ b/src/plugins/zlib/istream-zlib.c
@@ -35,7 +35,7 @@ static void _destroy(struct _iostream *stream __attr_unused__)
{
struct _istream *_stream = (struct _istream *) stream;
- p_free(_stream->iostream.pool, _stream->w_buffer);
+ i_free(_stream->w_buffer);
}
static ssize_t _read(struct _istream *stream)
@@ -193,12 +193,12 @@ static void _sync(struct _istream *stream)
zstream->cached_size = (uoff_t)-1;
}
-struct istream *i_stream_create_zlib(int fd, pool_t pool)
+struct istream *i_stream_create_zlib(int fd)
{
struct zlib_istream *zstream;
struct stat st;
- zstream = p_new(pool, struct zlib_istream, 1);
+ zstream = i_new(struct zlib_istream, 1);
zstream->fd = fd;
zstream->file = gzdopen(fd, "r");
zstream->cached_size = (uoff_t)-1;
@@ -218,5 +218,5 @@ struct istream *i_stream_create_zlib(int fd, pool_t pool)
zstream->istream.istream.seekable = TRUE;
}
- return _i_stream_create(&zstream->istream, pool, fd, 0);
+ return _i_stream_create(&zstream->istream, fd, 0);
}
diff --git a/src/plugins/zlib/istream-zlib.h b/src/plugins/zlib/istream-zlib.h
index 88b3735..697c2ce 100644
--- a/src/plugins/zlib/istream-zlib.h
+++ b/src/plugins/zlib/istream-zlib.h
@@ -1,6 +1,6 @@
#ifndef __ISTREAM_ZLIB_H
#define __ISTREAM_ZLIB_H
-struct istream *i_stream_create_zlib(int fd, pool_t pool);
+struct istream *i_stream_create_zlib(int fd);
#endif
diff --git a/src/plugins/zlib/zlib-plugin.c b/src/plugins/zlib/zlib-plugin.c
index 97cab54..549dfc4 100644
--- a/src/plugins/zlib/zlib-plugin.c
+++ b/src/plugins/zlib/zlib-plugin.c
@@ -41,10 +41,8 @@ zlib_mailbox_open(struct mail_storage *storage, const char *name,
int fd;
fd = open(path, O_RDONLY);
- if (fd != -1) {
- input = zlib_input =
- i_stream_create_zlib(fd, default_pool);
- }
+ if (fd != -1)
+ input = zlib_input = i_stream_create_zlib(fd);
}
}
diff --git a/src/pop3-login/client.c b/src/pop3-login/client.c
index 040b6b9..3f5dc1c 100644
--- a/src/pop3-login/client.c
+++ b/src/pop3-login/client.c
@@ -65,10 +65,8 @@ static void client_set_title(struct pop3_client *client)
static void client_open_streams(struct pop3_client *client, int fd)
{
- client->input = i_stream_create_file(fd, default_pool,
- MAX_INBUF_SIZE, FALSE);
- client->output = o_stream_create_file(fd, default_pool,
- MAX_OUTBUF_SIZE, FALSE);
+ client->input = i_stream_create_file(fd, MAX_INBUF_SIZE, FALSE);
+ client->output = o_stream_create_file(fd, MAX_OUTBUF_SIZE, FALSE);
}
static void client_start_tls(struct pop3_client *client)
diff --git a/src/pop3/client.c b/src/pop3/client.c
index 10e96cb..3efc9e9 100644
--- a/src/pop3/client.c
+++ b/src/pop3/client.c
@@ -140,10 +140,8 @@ struct client *client_create(int fd_in, int fd_out,
client = i_new(struct client, 1);
client->fd_in = fd_in;
client->fd_out = fd_out;
- client->input = i_stream_create_file(fd_in, default_pool,
- MAX_INBUF_SIZE, FALSE);
- client->output = o_stream_create_file(fd_out, default_pool,
- (size_t)-1, FALSE);
+ client->input = i_stream_create_file(fd_in, MAX_INBUF_SIZE, FALSE);
+ client->output = o_stream_create_file(fd_out, (size_t)-1, FALSE);
o_stream_set_flush_callback(client->output, client_output, client);
client->io = io_add(fd_in, IO_READ, client_input, client);
diff --git a/src/util/rawlog.c b/src/util/rawlog.c
index ae89533..097f3f4 100644
--- a/src/util/rawlog.c
+++ b/src/util/rawlog.c
@@ -240,19 +240,16 @@ rawlog_proxy_create(int client_in_fd, int client_out_fd, int server_fd,
proxy = i_new(struct rawlog_proxy, 1);
proxy->server_fd = server_fd;
proxy->server_input =
- i_stream_create_file(server_fd, default_pool,
- MAX_PROXY_INPUT_SIZE, FALSE);
+ i_stream_create_file(server_fd, MAX_PROXY_INPUT_SIZE, FALSE);
proxy->server_output =
- o_stream_create_file(server_fd, default_pool,
- (size_t)-1, FALSE);
+ o_stream_create_file(server_fd, (size_t)-1, FALSE);
proxy->server_io = io_add(server_fd, IO_READ, server_input, proxy);
o_stream_set_flush_callback(proxy->server_output, server_output, proxy);
proxy->client_in_fd = client_in_fd;
proxy->client_out_fd = client_out_fd;
proxy->client_output =
- o_stream_create_file(client_out_fd, default_pool,
- (size_t)-1, FALSE);
+ o_stream_create_file(client_out_fd, (size_t)-1, FALSE);
proxy->client_io = io_add(proxy->client_in_fd, IO_READ,
client_input, proxy);
o_stream_set_flush_callback(proxy->client_output, client_output, proxy);