+static void command_post_select(struct sched *s, struct task *t)
+{
+ struct command_task *ct = t->private_data;
+ struct sockaddr_un unix_addr;
+ char buf[sizeof(uint32_t) + sizeof(int)];
+ uint32_t cookie;
+ int query_shmid, fd;
+
+ t->ret = 1;
+ if (!FD_ISSET(ct->fd, &s->rfds))
+ return;
+ t->ret = para_accept(ct->fd, &unix_addr, sizeof(unix_addr));
+ if (t->ret < 0)
+ return;
+ /*
+ * The following errors may be caused by a malicious local user. So do
+ * not return an error in this case as this would terminate para_afs
+ * and para_server.
+ */
+ fd = t->ret;
+ /* FIXME: This is easily dosable (peer doesn't send data) */
+ t->ret = recv_bin_buffer(fd, buf, sizeof(buf));
+ if (t->ret < 0) {
+ PARA_NOTICE_LOG("%s (%d)\n", PARA_STRERROR(-t->ret), t->ret);
+ goto out;
+ }
+ if (t->ret != sizeof(buf)) {
+ PARA_NOTICE_LOG("short read (%d bytes, expected %lu)\n",
+ t->ret, (long unsigned) sizeof(buf));
+ goto out;
+ }
+ cookie = *(uint32_t *)buf;
+ if (cookie != ct->cookie) {
+ PARA_NOTICE_LOG("received invalid cookie(got %u, expected %u)\n",
+ (unsigned)cookie, (unsigned)ct->cookie);
+ goto out;
+ }
+ query_shmid = *(int *)(buf + sizeof(cookie));
+ if (query_shmid < 0) {
+ PARA_WARNING_LOG("received invalid query shmid %d)\n",
+ query_shmid);
+ goto out;
+ }
+ /* Ignore return value: Errors might be ok here. */
+ call_callback(fd, query_shmid);
+out:
+ t->ret = 1;
+ close(fd);
+}
+
+static void register_command_task(uint32_t cookie)
+{
+ struct command_task *ct = &command_task_struct;
+ ct->fd = setup_command_socket_or_die();
+ ct->cookie = cookie;
+
+ ct->task.pre_select = command_pre_select;
+ ct->task.post_select = command_post_select;
+ ct->task.private_data = ct;
+ sprintf(ct->task.status, "command task");
+ register_task(&ct->task);
+}
+
+void register_tasks(uint32_t cookie)
+{
+ register_signal_task();
+ register_command_task(cookie);
+}
+
+static char *database_dir;
+
+static int make_database_dir(void)
+{
+ int ret;
+
+ if (!database_dir) {
+ if (conf.afs_database_dir_given)
+ database_dir = para_strdup(conf.afs_database_dir_arg);
+ else {
+ char *home = para_homedir();
+ database_dir = make_message(
+ "%s/.paraslash/afs_database", home);
+ free(home);
+ }
+ }
+ PARA_INFO_LOG("afs_database dir %s\n", database_dir);
+ ret = para_mkdir(database_dir, 0777);
+ if (ret >= 0 || ret == -E_EXIST)
+ return 1;
+ free(database_dir);
+ database_dir = NULL;
+ return ret;
+}
+
+static int open_afs_tables(void)
+{
+ int ret = make_database_dir();