summaryrefslogtreecommitdiff
path: root/distcc/src/access.c
diff options
context:
space:
mode:
Diffstat (limited to 'distcc/src/access.c')
-rw-r--r--distcc/src/access.c125
1 files changed, 125 insertions, 0 deletions
diff --git a/distcc/src/access.c b/distcc/src/access.c
new file mode 100644
index 0000000..9b15f96
--- /dev/null
+++ b/distcc/src/access.c
@@ -0,0 +1,125 @@
+/* -*- c-file-style: "java"; indent-tabs-mode: nil; fill-column: 78 -*-
+ *
+ * distcc -- A simple distributed compiler system
+ *
+ * Copyright (C) 2003, 2004 by Martin Pool <mbp@samba.org>
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License as
+ * published by the Free Software Foundation; either version 2 of the
+ * License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
+ * USA
+ */
+
+ /*
+ * They that forsake the law praise the wicked: but such
+ * as keep the law contend with them.
+ * -- Proverbs 28:4
+ */
+
+
+#include "config.h"
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include <sys/types.h>
+#include <sys/socket.h>
+#include <netinet/in.h>
+#include <arpa/inet.h>
+
+#include "types.h"
+#include "distcc.h"
+#include "trace.h"
+#include "access.h"
+#include "exitcode.h"
+
+/**
+ * @file
+ *
+ * Simple IP-based access-control system
+ */
+
+static const in_addr_t allones = 0xffffffffUL;
+
+
+/**
+ * Interpret a "HOST/BITS" mask specification. Return @p value and @p mask.
+ **/
+int dcc_parse_mask(const char *spec,
+ in_addr_t *value,
+ in_addr_t *mask)
+{
+ int value_len;
+ struct in_addr ia;
+ int mask_bits;
+ char *value_str;
+ int matched;
+ const char *mask_str;
+
+ value_len = strcspn(spec, "/");
+
+ /* get bit before slash */
+ value_str = strdup(spec);
+ value_str[value_len] = '\0';
+ matched = inet_pton(AF_INET, value_str, &ia);
+
+ /* extract and parse value part */
+ if (!matched) {
+ rs_log_error("can't parse internet address \"%s\"", value_str);
+ free(value_str);
+ return EXIT_BAD_ARGUMENTS;
+ }
+ free(value_str);
+ *value = ia.s_addr;
+
+ mask_str = &spec[value_len + 1];
+ if (spec[value_len] && *mask_str) {
+ /* find mask length as a number of bits */
+ mask_bits = atoi(mask_str);
+ if (mask_bits < 0 || mask_bits > 32) {
+ rs_log_error("invalid mask \"%s\"", mask_str);
+ return EXIT_BAD_ARGUMENTS;
+ }
+
+ /* Make a network-endian mask with the top mask_bits set. */
+ if (mask_bits == 32)
+ *mask = allones;
+ else
+ *mask = htonl(~(allones >> mask_bits));
+ } else {
+ *mask = allones;
+ }
+ return 0;
+}
+
+
+/**
+ * Check whether a client ought to be allowed.
+ *
+ * @returns 0 for allowed, or EXIT_ACCESS_DENIED.
+ **/
+int dcc_check_address(in_addr_t client,
+ in_addr_t value,
+ in_addr_t mask)
+{
+ if ((client & mask) == (value & mask)) {
+ rs_trace("match client %#lx, value %#lx, mask %#lx",
+ (long) client, (long) value, (long) mask);
+ return 0;
+ } else {
+ rs_trace("deny client %#lx, value %#lx, mask %#lx",
+ (long) client, (long) value, (long) mask);
+ return EXIT_ACCESS_DENIED;
+ }
+}