| 
     1 /*
    2  * $FreeBSD$
    3  *
    4  * Copyright (c) 2011-2013, 2015, 2019 Juniper Networks, Inc.
    5  * All rights reserved.
    6  *
    7  * Redistribution and use in source and binary forms, with or without
    8  * modification, are permitted provided that the following conditions
    9  * are met:
   10  * 1. Redistributions of source code must retain the above copyright
   11  *    notice, this list of conditions and the following disclaimer.
   12  * 2. Redistributions in binary form must reproduce the above copyright
   13  *    notice, this list of conditions and the following disclaimer in the
   14  *    documentation and/or other materials provided with the distribution.
   15  *
   16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
   17  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
   18  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
   19  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
   20  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
   21  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
   22  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
   23  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
   24  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
   25  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
   26  * SUCH DAMAGE.
   27  */
   28 
   29 #include <sys/cdefs.h>
   30 #include <sys/param.h>
   31 #include <sys/systm.h>
   32 #include <sys/buf.h>
   33 #include <sys/conf.h>
   34 #include <sys/errno.h>
   35 #include <sys/fcntl.h>
   36 #include <sys/file.h>
   37 #include <sys/filedesc.h>
   38 #include <sys/ioccom.h>
   39 #include <sys/jail.h>
   40 #include <sys/kernel.h>
   41 #include <sys/lock.h>
   42 #include <sys/malloc.h>
   43 #include <sys/mdioctl.h>
   44 #include <sys/mount.h>
   45 #include <sys/mutex.h>
   46 #include <sys/namei.h>
   47 #include <sys/priv.h>
   48 #include <sys/proc.h>
   49 #include <sys/queue.h>
   50 #include <sys/vnode.h>
   51 
   52 #include <security/mac_veriexec/mac_veriexec.h>
   53 #include <security/mac_veriexec/mac_veriexec_internal.h>
   54 
   55 #include "veriexec_ioctl.h"
   56 
   57 /*
   58  * We need a mutex while updating lists etc.
   59  */
   60 extern struct mtx ve_mutex;
   61 
   62 /*
   63  * Handle the ioctl for the device
   64  */
   65 static int
   66 verifiedexecioctl(struct cdev *dev __unused, u_long cmd, caddr_t data,
   67     int flags, struct thread *td)
   68 {
   69         struct nameidata nid;
   70         struct vattr vattr;
   71         struct verified_exec_label_params *lparams;
   72         struct verified_exec_params *params;
   73         int error = 0;
   74 
   75         /*
   76          * These commands are considered safe requests for anyone who has
   77          * permission to access to device node.
   78          */
   79         switch (cmd) {
   80         case VERIEXEC_GETSTATE:
   81                 {
   82                         int *ip = (int *)data;
   83 
   84                         if (ip)
   85                                 *ip = mac_veriexec_get_state();
   86                         else
   87                             error = EINVAL;
   88 
   89                         return (error);
   90                 }
   91                 break;
   92         default:
   93                 break;
   94         }
   95 
   96         /*
   97          * Anything beyond this point is considered dangerous, so we need to
   98          * only allow processes that have kmem write privs to do them.
   99          *
  100          * MAC/veriexec will grant kmem write privs to "trusted" processes.
  101          */
  102         error = priv_check(td, PRIV_KMEM_WRITE);
  103         if (error)
  104                 return (error);
  105 
  106         lparams = (struct verified_exec_label_params *)data;
  107         if (cmd == VERIEXEC_LABEL_LOAD)
  108                 params = &lparams->params;
  109         else
  110                 params = (struct verified_exec_params *)data;
  111 
  112         switch (cmd) {
  113         case VERIEXEC_ACTIVE:
  114                 mtx_lock(&ve_mutex);
  115                 if (mac_veriexec_in_state(VERIEXEC_STATE_LOADED))
  116                         mac_veriexec_set_state(VERIEXEC_STATE_ACTIVE);
  117                 else
  118                         error = EINVAL;
  119                 mtx_unlock(&ve_mutex);
  120                 break;
  121         case VERIEXEC_DEBUG_ON:
  122                 mtx_lock(&ve_mutex);
  123                 {
  124                         int *ip = (int *)data;
  125                         
  126                         mac_veriexec_debug++;
  127                         if (ip) {
  128                                 if (*ip > 0)
  129                                         mac_veriexec_debug = *ip;       
  130                                 *ip = mac_veriexec_debug;
  131                         }
  132                 }
  133                 mtx_unlock(&ve_mutex);
  134                 break;
  135         case VERIEXEC_DEBUG_OFF:
  136                 mac_veriexec_debug = 0;
  137                 break;
  138         case VERIEXEC_ENFORCE:
  139                 mtx_lock(&ve_mutex);
  140                 if (mac_veriexec_in_state(VERIEXEC_STATE_LOADED))
  141                         mac_veriexec_set_state(VERIEXEC_STATE_ACTIVE |
  142                             VERIEXEC_STATE_ENFORCE);
  143                 else
  144                         error = EINVAL;
  145                 mtx_unlock(&ve_mutex);
  146                 break;
  147         case VERIEXEC_GETVERSION:
  148                 {
  149                         int *ip = (int *)data;
  150 
  151                         if (ip)
  152                                 *ip = MAC_VERIEXEC_VERSION;
  153                         else
  154                                 error = EINVAL;
  155                 }
  156                 break;
  157         case VERIEXEC_LOCK:
  158                 mtx_lock(&ve_mutex);
  159                 mac_veriexec_set_state(VERIEXEC_STATE_LOCKED);
  160                 mtx_unlock(&ve_mutex);
  161                 break;
  162         case VERIEXEC_LOAD:
  163                 if (prison0.pr_securelevel > 0)
  164                         return (EPERM); /* no updates when secure */
  165 
  166                 /* FALLTHROUGH */
  167         case VERIEXEC_LABEL_LOAD:
  168         case VERIEXEC_SIGNED_LOAD:
  169                 /*
  170                  * If we use a loader that will only use a
  171                  * digitally signed hash list - which it verifies.
  172                  * We can load fingerprints provided veriexec is not locked.
  173                  */
  174                 if (prison0.pr_securelevel > 0 &&
  175                     !mac_veriexec_in_state(VERIEXEC_STATE_LOADED)) {
  176                         /*
  177                          * If securelevel has been raised and we
  178                          * do not have any fingerprints loaded,
  179                          * it would dangerous to do so now.
  180                          */
  181                         return (EPERM);
  182                 }
  183                 if (mac_veriexec_in_state(VERIEXEC_STATE_LOCKED))
  184                         error = EPERM;
  185                 else {
  186                         size_t labellen = 0;
  187                         int flags = FREAD;
  188                         int override = (cmd != VERIEXEC_LOAD);
  189 
  190                         /*
  191                          * Get the attributes for the file name passed
  192                          * stash the file's device id and inode number
  193                          * along with it's fingerprint in a list for
  194                          * exec to use later.
  195                          */
  196                         /*
  197                          * FreeBSD seems to copy the args to kernel space
  198                          */
  199                         NDINIT(&nid, LOOKUP, FOLLOW, UIO_SYSSPACE, params->file);
  200                         if ((error = vn_open(&nid, &flags, 0, NULL)) != 0)
  201                                 return (error);
  202 
  203                         error = VOP_GETATTR(nid.ni_vp, &vattr, td->td_ucred);
  204                         if (error != 0) {
  205                                 mac_veriexec_set_fingerprint_status(nid.ni_vp,
  206                                     FINGERPRINT_INVALID);
  207                                 VOP_UNLOCK(nid.ni_vp);
  208                                 (void) vn_close(nid.ni_vp, FREAD, td->td_ucred,
  209                                     td);
  210                                 return (error);
  211                         }
  212                         if (override) {
  213                                 /*
  214                                  * If the file is on a "verified" filesystem
  215                                  * someone may be playing games.
  216                                  */
  217                                 if ((nid.ni_vp->v_mount->mnt_flag &
  218                                     MNT_VERIFIED) != 0)
  219                                         override = 0;
  220                         }
  221 
  222                         /*
  223                          * invalidate the node fingerprint status
  224                          * which will have been set in the vn_open
  225                          * and would always be FINGERPRINT_NOTFOUND
  226                          */
  227                         mac_veriexec_set_fingerprint_status(nid.ni_vp,
  228                             FINGERPRINT_INVALID);
  229                         VOP_UNLOCK(nid.ni_vp);
  230                         (void) vn_close(nid.ni_vp, FREAD, td->td_ucred, td);
  231                         if (params->flags & VERIEXEC_LABEL)
  232                                 labellen = strnlen(lparams->label,
  233                                     sizeof(lparams->label) - 1) + 1;
  234 
  235                         mtx_lock(&ve_mutex);
  236                         error = mac_veriexec_metadata_add_file(
  237                             ((params->flags & VERIEXEC_FILE) != 0),
  238                             vattr.va_fsid, vattr.va_fileid, vattr.va_gen,
  239                             params->fingerprint,
  240                             (params->flags & VERIEXEC_LABEL) ?
  241                             lparams->label : NULL, labellen,
  242                             params->flags, params->fp_type, override);
  243 
  244                         mac_veriexec_set_state(VERIEXEC_STATE_LOADED);
  245                         mtx_unlock(&ve_mutex);
  246                 }
  247                 break;
  248         default:
  249                 error = ENODEV;
  250         }
  251         return (error);
  252 }
  253 
  254 struct cdevsw veriexec_cdevsw = {
  255         .d_version =    D_VERSION,
  256         .d_ioctl =      verifiedexecioctl,
  257         .d_name =       "veriexec",
  258 };
  259 
  260 static void
  261 veriexec_drvinit(void *unused __unused)
  262 {
  263 
  264         make_dev(&veriexec_cdevsw, 0, UID_ROOT, GID_WHEEL, 0600, "veriexec");
  265 }
  266 
  267 SYSINIT(veriexec, SI_SUB_PSEUDO, SI_ORDER_ANY, veriexec_drvinit, NULL);
  268 MODULE_DEPEND(veriexec, mac_veriexec, MAC_VERIEXEC_VERSION,
  269     MAC_VERIEXEC_VERSION, MAC_VERIEXEC_VERSION);
Cache object: c6f721d692aed60441080c81741176ce 
 
 |