/* -*- c-file-style: "java"; indent-tabs-mode: nil; fill-column: 78 -*- * * distcc -- A simple distributed compiler system * * Copyright (C) 2003, 2004 by Martin Pool * * 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 #include #include #include #include #include #include #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_aton(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; } }