LCOV - code coverage report
Current view: top level - source3/smbd - msdfs.c (source / functions) Hit Total Coverage
Test: coverage report for master 70ed9daf Lines: 342 793 43.1 %
Date: 2024-01-11 09:59:51 Functions: 14 21 66.7 %

          Line data    Source code
       1             : /*
       2             :    Unix SMB/Netbios implementation.
       3             :    Version 3.0
       4             :    MSDFS services for Samba
       5             :    Copyright (C) Shirish Kalele 2000
       6             :    Copyright (C) Jeremy Allison 2007
       7             :    Copyright (C) Robin McCorkell 2015
       8             : 
       9             :    This program is free software; you can redistribute it and/or modify
      10             :    it under the terms of the GNU General Public License as published by
      11             :    the Free Software Foundation; either version 3 of the License, or
      12             :    (at your option) any later version.
      13             : 
      14             :    This program is distributed in the hope that it will be useful,
      15             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      16             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      17             :    GNU General Public License for more details.
      18             : 
      19             :    You should have received a copy of the GNU General Public License
      20             :    along with this program.  If not, see <http://www.gnu.org/licenses/>.
      21             : 
      22             : */
      23             : 
      24             : #define DBGC_CLASS DBGC_MSDFS
      25             : #include "includes.h"
      26             : #include "system/filesys.h"
      27             : #include "smbd/smbd.h"
      28             : #include "smbd/globals.h"
      29             : #include "msdfs.h"
      30             : #include "auth.h"
      31             : #include "../auth/auth_util.h"
      32             : #include "lib/param/loadparm.h"
      33             : #include "libcli/security/security.h"
      34             : #include "librpc/gen_ndr/ndr_dfsblobs.h"
      35             : #include "lib/tsocket/tsocket.h"
      36             : #include "lib/global_contexts.h"
      37             : #include "source3/lib/substitute.h"
      38             : #include "source3/smbd/dir.h"
      39             : 
      40             : /**********************************************************************
      41             :  Parse a DFS pathname of the form(s)
      42             : 
      43             :  \hostname\service                      - self referral
      44             :  \hostname\service\remainingpath        - Windows referral path
      45             : 
      46             :  FIXME! Should we also parse:
      47             :  \hostname\service/remainingpath        - POSIX referral path
      48             :  as currently nothing uses this ?
      49             : 
      50             :  into the dfs_path components. Strict form.
      51             : 
      52             :  Checks DFS path starts with separator.
      53             :  Checks hostname is ours.
      54             :  Ensures servicename (share) is sent, and
      55             :      if so, terminates the name or is followed by
      56             :      \pathname.
      57             : 
      58             :  If returned, remainingpath is untouched. Caller must call
      59             :  check_path_syntax() on it.
      60             : 
      61             :  Called by all non-fileserver processing (DFS RPC, FSCTL_DFS_GET_REFERRALS)
      62             :  etc. Errors out on any inconsistency in the path.
      63             : **********************************************************************/
      64             : 
      65       14944 : static NTSTATUS parse_dfs_path_strict(TALLOC_CTX *ctx,
      66             :                                 const char *pathname,
      67             :                                 char **_hostname,
      68             :                                 char **_servicename,
      69             :                                 char **_remaining_path)
      70             : {
      71       14944 :         char *pathname_local = NULL;
      72       14944 :         char *p = NULL;
      73       14944 :         const char *hostname = NULL;
      74       14944 :         const char *servicename = NULL;
      75       14944 :         const char *reqpath = NULL;
      76       14944 :         bool my_hostname = false;
      77           0 :         NTSTATUS status;
      78             : 
      79       14944 :         DBG_DEBUG("path = |%s|\n", pathname);
      80             : 
      81       14944 :         pathname_local = talloc_strdup(talloc_tos(), pathname);
      82       14944 :         if (pathname_local == NULL) {
      83           0 :                 return NT_STATUS_NO_MEMORY;
      84             :         }
      85             :         /*
      86             :          * parse_dfs_path_strict() is called from
      87             :          * get_referred_path() and create_junction()
      88             :          * which use Windows DFS paths of \server\share.
      89             :          */
      90             : 
      91             :         /*
      92             :          * Strict DFS paths *must* start with the
      93             :          * path separator '\\'.
      94             :          */
      95             : 
      96       14944 :         if (pathname_local[0] != '\\') {
      97           2 :                 DBG_ERR("path %s doesn't start with \\\n",
      98             :                         pathname_local);
      99           2 :                 status = NT_STATUS_NOT_FOUND;
     100           2 :                 goto out;
     101             :         }
     102             : 
     103             :         /* Now tokenize. */
     104             :         /* Parse out hostname. */
     105       14942 :         p = strchr(pathname_local + 1, '\\');
     106       14942 :         if (p == NULL) {
     107           0 :                 DBG_ERR("can't parse hostname from path %s\n",
     108             :                         pathname_local);
     109           0 :                 status = NT_STATUS_NOT_FOUND;
     110           0 :                 goto out;
     111             :         }
     112       14942 :         *p = '\0';
     113       14942 :         hostname = &pathname_local[1];
     114             : 
     115       14942 :         DBG_DEBUG("hostname: %s\n", hostname);
     116             : 
     117             :         /* Is this really our hostname ? */
     118       14942 :         my_hostname = is_myname_or_ipaddr(hostname);
     119       14942 :         if (!my_hostname) {
     120           0 :                 DBG_ERR("Hostname %s is not ours.\n",
     121             :                         hostname);
     122           0 :                 status = NT_STATUS_NOT_FOUND;
     123           0 :                 goto out;
     124             :         }
     125             : 
     126       14942 :         servicename = p + 1;
     127             : 
     128             :         /*
     129             :          * Find the end of servicename by looking for
     130             :          * a directory separator character. The character
     131             :          * should be '\\' for a Windows path.
     132             :          * If there is no separator, then this is a self-referral
     133             :          * of "\server\share".
     134             :          */
     135             : 
     136       14942 :         p = strchr(servicename, '\\');
     137       14942 :         if (p != NULL) {
     138        4450 :                 *p = '\0';
     139             :         }
     140             : 
     141       14942 :         DBG_DEBUG("servicename: %s\n", servicename);
     142             : 
     143       14942 :         if (p == NULL) {
     144             :                 /* Client sent self referral "\server\share". */
     145       10492 :                 reqpath = "";
     146             :         } else {
     147             :                 /* Step past the '\0' we just replaced '\\' with. */
     148        4450 :                 reqpath = p + 1;
     149             :         }
     150             : 
     151       14942 :         DBG_DEBUG("rest of the path: %s\n", reqpath);
     152             : 
     153       14942 :         if (_hostname != NULL) {
     154           0 :                 *_hostname = talloc_strdup(ctx, hostname);
     155           0 :                 if (*_hostname == NULL) {
     156           0 :                         status = NT_STATUS_NO_MEMORY;
     157           0 :                         goto out;
     158             :                 }
     159             :         }
     160       14942 :         if (_servicename != NULL) {
     161       14942 :                 *_servicename = talloc_strdup(ctx, servicename);
     162       14942 :                 if (*_servicename == NULL) {
     163           0 :                         status = NT_STATUS_NO_MEMORY;
     164           0 :                         goto out;
     165             :                 }
     166             :         }
     167       14942 :         if (_remaining_path != NULL) {
     168       14942 :                 *_remaining_path = talloc_strdup(ctx, reqpath);
     169       14942 :                 if (*_remaining_path == NULL) {
     170           0 :                         status = NT_STATUS_NO_MEMORY;
     171           0 :                         goto out;
     172             :                 }
     173             :         }
     174             : 
     175       14942 :         status = NT_STATUS_OK;
     176       14944 : out:
     177       14944 :         TALLOC_FREE(pathname_local);
     178       14944 :         return status;
     179             : }
     180             : 
     181             : /********************************************************
     182             :  Fake up a connection struct for the VFS layer, for use in
     183             :  applications (such as the python bindings), that do not want the
     184             :  global working directory changed under them.
     185             : 
     186             :  SMB_VFS_CONNECT requires root privileges.
     187             : *********************************************************/
     188             : 
     189        8406 : static NTSTATUS create_conn_struct_as_root(TALLOC_CTX *ctx,
     190             :                             struct tevent_context *ev,
     191             :                             struct messaging_context *msg,
     192             :                             connection_struct **pconn,
     193             :                             int snum,
     194             :                             const char *path,
     195             :                             const struct auth_session_info *session_info)
     196             : {
     197         115 :         connection_struct *conn;
     198         115 :         char *connpath;
     199         115 :         const char *vfs_user;
     200         115 :         struct smbd_server_connection *sconn;
     201        8406 :         const char *servicename = lp_const_servicename(snum);
     202         115 :         bool ok;
     203             : 
     204        8406 :         sconn = talloc_zero(ctx, struct smbd_server_connection);
     205        8406 :         if (sconn == NULL) {
     206           0 :                 return NT_STATUS_NO_MEMORY;
     207             :         }
     208             : 
     209        8406 :         sconn->ev_ctx = ev;
     210        8406 :         sconn->msg_ctx = msg;
     211             : 
     212        8406 :         conn = conn_new(sconn);
     213        8406 :         if (conn == NULL) {
     214           0 :                 TALLOC_FREE(sconn);
     215           0 :                 return NT_STATUS_NO_MEMORY;
     216             :         }
     217             : 
     218             :         /* Now we have conn, we need to make sconn a child of conn,
     219             :          * for a proper talloc tree */
     220        8406 :         talloc_steal(conn, sconn);
     221             : 
     222        8406 :         if (snum == -1 && servicename == NULL) {
     223        1324 :                 servicename = "Unknown Service (snum == -1)";
     224             :         }
     225             : 
     226        8406 :         connpath = talloc_strdup(conn, path);
     227        8406 :         if (!connpath) {
     228           0 :                 TALLOC_FREE(conn);
     229           0 :                 return NT_STATUS_NO_MEMORY;
     230             :         }
     231        8406 :         connpath = talloc_string_sub(conn,
     232             :                                      connpath,
     233             :                                      "%S",
     234             :                                      servicename);
     235        8406 :         if (!connpath) {
     236           0 :                 TALLOC_FREE(conn);
     237           0 :                 return NT_STATUS_NO_MEMORY;
     238             :         }
     239             : 
     240             :         /* needed for smbd_vfs_init() */
     241             : 
     242        8406 :         conn->params->service = snum;
     243        8406 :         conn->cnum = TID_FIELD_INVALID;
     244             : 
     245        8406 :         SMB_ASSERT(session_info != NULL);
     246             : 
     247        8406 :         conn->session_info = copy_session_info(conn, session_info);
     248        8406 :         if (conn->session_info == NULL) {
     249           0 :                 DBG_ERR("copy_serverinfo failed\n");
     250           0 :                 TALLOC_FREE(conn);
     251           0 :                 return NT_STATUS_NO_MEMORY;
     252             :         }
     253             : 
     254             :         /* unix_info could be NULL in session_info */
     255        8406 :         if (conn->session_info->unix_info != NULL) {
     256        8406 :                 vfs_user = conn->session_info->unix_info->unix_name;
     257             :         } else {
     258           0 :                 vfs_user = get_current_username();
     259             :         }
     260             : 
     261        8406 :         conn_setup_case_options(conn);
     262             : 
     263        8406 :         set_conn_connectpath(conn, connpath);
     264             : 
     265             :         /*
     266             :          * New code to check if there's a share security descriptor
     267             :          * added from NT server manager. This is done after the
     268             :          * smb.conf checks are done as we need a uid and token. JRA.
     269             :          *
     270             :          */
     271        8406 :         share_access_check(conn->session_info->security_token,
     272             :                            servicename,
     273             :                            MAXIMUM_ALLOWED_ACCESS,
     274        8406 :                            &conn->share_access);
     275             : 
     276        8406 :         if ((conn->share_access & FILE_WRITE_DATA) == 0) {
     277           0 :                 if ((conn->share_access & FILE_READ_DATA) == 0) {
     278             :                         /* No access, read or write. */
     279           0 :                         DBG_WARNING("connection to %s "
     280             :                                     "denied due to security "
     281             :                                     "descriptor.\n",
     282             :                                     servicename);
     283           0 :                         conn_free(conn);
     284           0 :                         return NT_STATUS_ACCESS_DENIED;
     285             :                 }
     286           0 :                 conn->read_only = true;
     287             :         }
     288             : 
     289        8406 :         if (!smbd_vfs_init(conn)) {
     290           0 :                 NTSTATUS status = map_nt_error_from_unix(errno);
     291           0 :                 DEBUG(0,("create_conn_struct: smbd_vfs_init failed.\n"));
     292           0 :                 conn_free(conn);
     293           0 :                 return status;
     294             :         }
     295             : 
     296             :         /* this must be the first filesystem operation that we do */
     297        8406 :         if (SMB_VFS_CONNECT(conn, servicename, vfs_user) < 0) {
     298           0 :                 DEBUG(0,("VFS connect failed!\n"));
     299           0 :                 conn_free(conn);
     300           0 :                 return NT_STATUS_UNSUCCESSFUL;
     301             :         }
     302             : 
     303        8406 :         ok = canonicalize_connect_path(conn);
     304        8406 :         if (!ok) {
     305           0 :                 DBG_ERR("Failed to canonicalize sharepath\n");
     306           0 :                 conn_free(conn);
     307           0 :                 return NT_STATUS_ACCESS_DENIED;
     308             :         }
     309             : 
     310        8406 :         conn->fs_capabilities = SMB_VFS_FS_CAPABILITIES(conn, &conn->ts_res);
     311        8406 :         conn->tcon_done = true;
     312        8406 :         *pconn = talloc_move(ctx, &conn);
     313             : 
     314        8406 :         return NT_STATUS_OK;
     315             : }
     316             : 
     317        8382 : static int conn_struct_tos_destructor(struct conn_struct_tos *c)
     318             : {
     319        8382 :         if (c->oldcwd_fname != NULL) {
     320        4454 :                 vfs_ChDir(c->conn, c->oldcwd_fname);
     321        4454 :                 TALLOC_FREE(c->oldcwd_fname);
     322             :         }
     323        8382 :         SMB_VFS_DISCONNECT(c->conn);
     324        8382 :         conn_free(c->conn);
     325        8382 :         return 0;
     326             : }
     327             : 
     328             : /********************************************************
     329             :  Fake up a connection struct for the VFS layer, for use in
     330             :  applications (such as the python bindings), that do not want the
     331             :  global working directory changed under them.
     332             : 
     333             :  SMB_VFS_CONNECT requires root privileges.
     334             :  This temporary uses become_root() and unbecome_root().
     335             : 
     336             :  But further impersonation has to be cone by the caller.
     337             : *********************************************************/
     338        8394 : NTSTATUS create_conn_struct_tos(struct messaging_context *msg,
     339             :                                 int snum,
     340             :                                 const char *path,
     341             :                                 const struct auth_session_info *session_info,
     342             :                                 struct conn_struct_tos **_c)
     343             : {
     344        8394 :         struct conn_struct_tos *c = NULL;
     345        8394 :         struct tevent_context *ev = NULL;
     346         115 :         NTSTATUS status;
     347             : 
     348        8394 :         *_c = NULL;
     349             : 
     350        8394 :         c = talloc_zero(talloc_tos(), struct conn_struct_tos);
     351        8394 :         if (c == NULL) {
     352           0 :                 return NT_STATUS_NO_MEMORY;
     353             :         }
     354             : 
     355        8394 :         ev = samba_tevent_context_init(c);
     356        8394 :         if (ev == NULL) {
     357           0 :                 TALLOC_FREE(c);
     358           0 :                 return NT_STATUS_NO_MEMORY;
     359             :         }
     360             : 
     361        8394 :         become_root();
     362        8394 :         status = create_conn_struct_as_root(c,
     363             :                                             ev,
     364             :                                             msg,
     365        8394 :                                             &c->conn,
     366             :                                             snum,
     367             :                                             path,
     368             :                                             session_info);
     369        8394 :         unbecome_root();
     370        8394 :         if (!NT_STATUS_IS_OK(status)) {
     371           0 :                 TALLOC_FREE(c);
     372           0 :                 return status;
     373             :         }
     374             : 
     375        8394 :         talloc_set_destructor(c, conn_struct_tos_destructor);
     376             : 
     377        8394 :         *_c = c;
     378        8394 :         return NT_STATUS_OK;
     379             : }
     380             : 
     381             : /********************************************************
     382             :  Fake up a connection struct for the VFS layer.
     383             :  Note: this performs a vfs connect and CHANGES CWD !!!! JRA.
     384             : 
     385             :  See also the comment for create_conn_struct_tos() above!
     386             : 
     387             :  The CWD change is reverted by the destructor of
     388             :  conn_struct_tos when the current talloc_tos() is destroyed.
     389             : *********************************************************/
     390        4466 : NTSTATUS create_conn_struct_tos_cwd(struct messaging_context *msg,
     391             :                                     int snum,
     392             :                                     const char *path,
     393             :                                     const struct auth_session_info *session_info,
     394             :                                     struct conn_struct_tos **_c)
     395             : {
     396        4466 :         struct conn_struct_tos *c = NULL;
     397        4466 :         struct smb_filename smb_fname_connectpath = {0};
     398           0 :         NTSTATUS status;
     399             : 
     400        4466 :         *_c = NULL;
     401             : 
     402        4466 :         status = create_conn_struct_tos(msg,
     403             :                                         snum,
     404             :                                         path,
     405             :                                         session_info,
     406             :                                         &c);
     407        4466 :         if (!NT_STATUS_IS_OK(status)) {
     408           0 :                 return status;
     409             :         }
     410             : 
     411             :         /*
     412             :          * Windows seems to insist on doing trans2getdfsreferral() calls on
     413             :          * the IPC$ share as the anonymous user. If we try to chdir as that
     414             :          * user we will fail.... WTF ? JRA.
     415             :          */
     416             : 
     417        4466 :         c->oldcwd_fname = vfs_GetWd(c, c->conn);
     418        4466 :         if (c->oldcwd_fname == NULL) {
     419           0 :                 status = map_nt_error_from_unix(errno);
     420           0 :                 DEBUG(3, ("vfs_GetWd failed: %s\n", strerror(errno)));
     421           0 :                 TALLOC_FREE(c);
     422           0 :                 return status;
     423             :         }
     424             : 
     425        4466 :         smb_fname_connectpath = (struct smb_filename) {
     426        4466 :                 .base_name = c->conn->connectpath
     427             :         };
     428             : 
     429        4466 :         if (vfs_ChDir(c->conn, &smb_fname_connectpath) != 0) {
     430           0 :                 status = map_nt_error_from_unix(errno);
     431           0 :                 DBG_NOTICE("Can't ChDir to new conn path %s. "
     432             :                            "Error was %s\n",
     433             :                            c->conn->connectpath, strerror(errno));
     434           0 :                 TALLOC_FREE(c->oldcwd_fname);
     435           0 :                 TALLOC_FREE(c);
     436           0 :                 return status;
     437             :         }
     438             : 
     439        4466 :         *_c = c;
     440        4466 :         return NT_STATUS_OK;
     441             : }
     442             : 
     443             : /********************************************************
     444             :  Fake up a connection struct for the VFS layer.
     445             :  This takes an TALLOC_CTX and tevent_context from the
     446             :  caller and the resulting connection_struct is stable
     447             :  across the lifetime of mem_ctx and ev.
     448             : 
     449             :  Note: this performs a vfs connect and changes cwd.
     450             : 
     451             :  See also the comment for create_conn_struct_tos() above!
     452             : *********************************************************/
     453             : 
     454          12 : NTSTATUS create_conn_struct_cwd(TALLOC_CTX *mem_ctx,
     455             :                                 struct tevent_context *ev,
     456             :                                 struct messaging_context *msg,
     457             :                                 const struct auth_session_info *session_info,
     458             :                                 int snum,
     459             :                                 const char *path,
     460             :                                 struct connection_struct **c)
     461             : {
     462           0 :         NTSTATUS status;
     463             : 
     464          12 :         become_root();
     465          12 :         status = create_conn_struct_as_root(mem_ctx,
     466             :                                             ev,
     467             :                                             msg,
     468             :                                             c,
     469             :                                             snum,
     470             :                                             path,
     471             :                                             session_info);
     472          12 :         unbecome_root();
     473          12 :         return status;
     474             : }
     475             : 
     476        2192 : static void shuffle_strlist(char **list, int count)
     477             : {
     478           0 :         int i;
     479           0 :         uint32_t r;
     480           0 :         char *tmp;
     481             : 
     482        4292 :         for (i = count; i > 1; i--) {
     483        2100 :                 r = generate_random() % i;
     484             : 
     485        2100 :                 tmp = list[i-1];
     486        2100 :                 list[i-1] = list[r];
     487        2100 :                 list[r] = tmp;
     488             :         }
     489        2192 : }
     490             : 
     491             : /**********************************************************************
     492             :  Parse the contents of a symlink to verify if it is an msdfs referral
     493             :  A valid referral is of the form:
     494             : 
     495             :  msdfs:server1\share1,server2\share2
     496             :  msdfs:server1\share1\pathname,server2\share2\pathname
     497             :  msdfs:server1/share1,server2/share2
     498             :  msdfs:server1/share1/pathname,server2/share2/pathname.
     499             : 
     500             :  Note that the alternate paths returned here must be of the canonicalized
     501             :  form:
     502             : 
     503             :  \server\share or
     504             :  \server\share\path\to\file,
     505             : 
     506             :  even in posix path mode. This is because we have no knowledge if the
     507             :  server we're referring to understands posix paths.
     508             :  **********************************************************************/
     509             : 
     510        4448 : bool parse_msdfs_symlink(TALLOC_CTX *ctx,
     511             :                         bool shuffle_referrals,
     512             :                         const char *target,
     513             :                         struct referral **ppreflist,
     514             :                         size_t *prefcount)
     515             : {
     516        4448 :         char *temp = NULL;
     517           0 :         char *prot;
     518        4448 :         char **alt_path = NULL;
     519        4448 :         size_t count = 0, i;
     520        4448 :         struct referral *reflist = NULL;
     521           0 :         char *saveptr;
     522             : 
     523        4448 :         temp = talloc_strdup(ctx, target);
     524        4448 :         if (!temp) {
     525           0 :                 return false;
     526             :         }
     527        4448 :         prot = strtok_r(temp, ":", &saveptr);
     528        4448 :         if (!prot) {
     529           0 :                 DEBUG(0,("parse_msdfs_symlink: invalid path !\n"));
     530           0 :                 TALLOC_FREE(temp);
     531           0 :                 return false;
     532             :         }
     533             : 
     534        4448 :         alt_path = talloc_array(ctx, char *, MAX_REFERRAL_COUNT);
     535        4448 :         if (!alt_path) {
     536           0 :                 TALLOC_FREE(temp);
     537           0 :                 return false;
     538             :         }
     539             : 
     540             :         /* parse out the alternate paths */
     541       12886 :         while((count<MAX_REFERRAL_COUNT) &&
     542       12886 :               ((alt_path[count] = strtok_r(NULL, ",", &saveptr)) != NULL)) {
     543        8438 :                 count++;
     544             :         }
     545             : 
     546             :         /* shuffle alternate paths */
     547        4448 :         if (shuffle_referrals) {
     548        2192 :                 shuffle_strlist(alt_path, count);
     549             :         }
     550             : 
     551        4448 :         DBG_DEBUG("count=%zu\n", count);
     552             : 
     553        4448 :         if (count) {
     554        4448 :                 reflist = talloc_zero_array(ctx,
     555             :                                 struct referral, count);
     556        4448 :                 if(reflist == NULL) {
     557           0 :                         TALLOC_FREE(temp);
     558           0 :                         TALLOC_FREE(alt_path);
     559           0 :                         return false;
     560             :                 }
     561             :         } else {
     562           0 :                 reflist = NULL;
     563             :         }
     564             : 
     565       12886 :         for(i=0;i<count;i++) {
     566           0 :                 char *p;
     567             : 
     568             :                 /* Canonicalize link target.
     569             :                  * Replace all /'s in the path by a \ */
     570        8438 :                 string_replace(alt_path[i], '/', '\\');
     571             : 
     572             :                 /* Remove leading '\\'s */
     573        8438 :                 p = alt_path[i];
     574        8438 :                 while (*p && (*p == '\\')) {
     575           0 :                         p++;
     576             :                 }
     577             : 
     578        8438 :                 reflist[i].alternate_path = talloc_asprintf(reflist,
     579             :                                 "\\%s",
     580             :                                 p);
     581        8438 :                 if (!reflist[i].alternate_path) {
     582           0 :                         TALLOC_FREE(temp);
     583           0 :                         TALLOC_FREE(alt_path);
     584           0 :                         TALLOC_FREE(reflist);
     585           0 :                         return false;
     586             :                 }
     587             : 
     588        8438 :                 reflist[i].proximity = 0;
     589        8438 :                 reflist[i].ttl = REFERRAL_TTL;
     590        8438 :                 DBG_DEBUG("Created alt path: %s\n",
     591             :                         reflist[i].alternate_path);
     592             :         }
     593             : 
     594        4448 :         if (ppreflist != NULL) {
     595        4448 :                 *ppreflist = reflist;
     596             :         } else {
     597           0 :                 TALLOC_FREE(reflist);
     598             :         }
     599        4448 :         if (prefcount != NULL) {
     600        4448 :                 *prefcount = count;
     601             :         }
     602        4448 :         TALLOC_FREE(temp);
     603        4448 :         TALLOC_FREE(alt_path);
     604        4448 :         return true;
     605             : }
     606             : 
     607             : /**********************************************************************
     608             :  Returns true if the unix path is a valid msdfs symlink.
     609             : **********************************************************************/
     610             : 
     611         374 : bool is_msdfs_link(struct files_struct *dirfsp,
     612             :                    struct smb_filename *atname)
     613             : {
     614         374 :         NTSTATUS status = SMB_VFS_READ_DFS_PATHAT(dirfsp->conn,
     615             :                                         talloc_tos(),
     616             :                                         dirfsp,
     617             :                                         atname,
     618             :                                         NULL,
     619             :                                         NULL);
     620         374 :         return (NT_STATUS_IS_OK(status));
     621             : }
     622             : 
     623             : /*****************************************************************
     624             :  Used by other functions to decide if a dfs path is remote,
     625             :  and to get the list of referred locations for that remote path.
     626             : 
     627             :  consumedcntp: how much of the dfs path is being redirected. the client
     628             :  should try the remaining path on the redirected server.
     629             : *****************************************************************/
     630             : 
     631        4448 : static NTSTATUS dfs_path_lookup(TALLOC_CTX *ctx,
     632             :                 connection_struct *conn,
     633             :                 const char *dfspath, /* Incoming complete dfs path */
     634             :                 const char *reqpath, /* Parsed out remaining path. */
     635             :                 uint32_t ucf_flags,
     636             :                 size_t *consumedcntp,
     637             :                 struct referral **ppreflist,
     638             :                 size_t *preferral_count)
     639             : {
     640           0 :         NTSTATUS status;
     641        4448 :         struct smb_filename *parent_smb_fname = NULL;
     642        4448 :         struct smb_filename *smb_fname_rel = NULL;
     643        4448 :         NTTIME twrp = 0;
     644        4448 :         char *local_pathname = NULL;
     645        4448 :         char *last_component = NULL;
     646        4448 :         char *atname = NULL;
     647        4448 :         size_t removed_components = 0;
     648        4448 :         bool posix = (ucf_flags & UCF_POSIX_PATHNAMES);
     649        4448 :         char *p = NULL;
     650        4448 :         char *canon_dfspath = NULL;
     651             : 
     652        4448 :         DBG_DEBUG("Conn path = %s reqpath = %s\n", conn->connectpath, reqpath);
     653             : 
     654        4448 :         local_pathname = talloc_strdup(ctx, reqpath);
     655        4448 :         if (local_pathname == NULL) {
     656           0 :                 status = NT_STATUS_NO_MEMORY;
     657           0 :                 goto out;
     658             :         }
     659             : 
     660             :         /* We know reqpath isn't a DFS path. */
     661        4448 :         ucf_flags &= ~UCF_DFS_PATHNAME;
     662             : 
     663        4448 :         if (ucf_flags & UCF_GMT_PATHNAME) {
     664           0 :                 extract_snapshot_token(local_pathname, &twrp);
     665           0 :                 ucf_flags &= ~UCF_GMT_PATHNAME;
     666             :         }
     667             : 
     668             :         /*
     669             :          * We should have been given a DFS path to resolve.
     670             :          * This should return NT_STATUS_PATH_NOT_COVERED.
     671             :          *
     672             :          * Do a pathname walk, stripping off components
     673             :          * until we get NT_STATUS_OK instead of
     674             :          * NT_STATUS_PATH_NOT_COVERED.
     675             :          *
     676             :          * Fail on any other error.
     677             :          */
     678             : 
     679       45500 :         for (;;) {
     680       49948 :                 TALLOC_CTX *frame = NULL;
     681       49948 :                 struct files_struct *dirfsp = NULL;
     682       49948 :                 struct smb_filename *smb_fname_walk = NULL;
     683             : 
     684       49948 :                 TALLOC_FREE(parent_smb_fname);
     685             : 
     686             :                 /*
     687             :                  * Use a local stackframe as filename_convert_dirfsp()
     688             :                  * opens handles on the last two components in the path.
     689             :                  * Allow these to be freed as we step back through
     690             :                  * the local_pathname.
     691             :                  */
     692       49948 :                 frame = talloc_stackframe();
     693       49948 :                 status = filename_convert_dirfsp(frame,
     694             :                                                  conn,
     695             :                                                  local_pathname,
     696             :                                                  ucf_flags,
     697             :                                                  twrp,
     698             :                                                  &dirfsp,
     699             :                                                  &smb_fname_walk);
     700             :                 /* If we got a name, save it. */
     701       49948 :                 if (smb_fname_walk != NULL) {
     702        4448 :                         parent_smb_fname = talloc_move(ctx, &smb_fname_walk);
     703             :                 }
     704       49948 :                 TALLOC_FREE(frame);
     705             : 
     706       49948 :                 if (!NT_STATUS_EQUAL(status, NT_STATUS_PATH_NOT_COVERED)) {
     707             :                         /*
     708             :                          * For any other status than NT_STATUS_PATH_NOT_COVERED
     709             :                          * (including NT_STATUS_OK) we exit the walk.
     710             :                          * If it's an error we catch it outside the loop.
     711             :                          */
     712        4448 :                         break;
     713             :                 }
     714             : 
     715             :                 /* Step back one component and save it off as last_component. */
     716       45500 :                 TALLOC_FREE(last_component);
     717       45500 :                 p = strrchr(local_pathname, '/');
     718       45500 :                 if (p == NULL) {
     719             :                         /*
     720             :                          * We removed all components.
     721             :                          * Go around once more to make
     722             :                          * sure we can open the root '\0'.
     723             :                          */
     724        3990 :                         last_component = talloc_strdup(ctx, local_pathname);
     725        3990 :                         *local_pathname = '\0';
     726             :                 } else {
     727       41510 :                         last_component = talloc_strdup(ctx, p+1);
     728       41510 :                         *p = '\0';
     729             :                 }
     730       45500 :                 if (last_component == NULL) {
     731           0 :                         status = NT_STATUS_NO_MEMORY;
     732           0 :                         goto out;
     733             :                 }
     734             :                 /* Integer wrap check. */
     735       45500 :                 if (removed_components + 1 < removed_components) {
     736           0 :                         status = NT_STATUS_INVALID_PARAMETER;
     737           0 :                         goto out;
     738             :                 }
     739       45500 :                 removed_components++;
     740             :         }
     741             : 
     742        4448 :         if (!NT_STATUS_IS_OK(status)) {
     743           0 :                 DBG_DEBUG("dfspath = %s. reqpath = %s. Error %s.\n",
     744             :                         dfspath,
     745             :                         reqpath,
     746             :                         nt_errstr(status));
     747           0 :                 goto out;
     748             :         }
     749             : 
     750        4448 :         if (parent_smb_fname->fsp == NULL) {
     751             :                 /* Unable to open parent. */
     752           0 :                 DBG_DEBUG("dfspath = %s. reqpath = %s. "
     753             :                           "Unable to open parent directory (%s).\n",
     754             :                         dfspath,
     755             :                         reqpath,
     756             :                         smb_fname_str_dbg(parent_smb_fname));
     757           0 :                 status = NT_STATUS_OBJECT_PATH_NOT_FOUND;
     758           0 :                 goto out;
     759             :         }
     760             : 
     761        4448 :         if (removed_components == 0) {
     762             :                 /*
     763             :                  * We never got NT_STATUS_PATH_NOT_COVERED.
     764             :                  * There was no DFS redirect.
     765             :                  */
     766           0 :                 DBG_DEBUG("dfspath = %s. reqpath = %s. "
     767             :                         "No removed components.\n",
     768             :                         dfspath,
     769             :                         reqpath);
     770           0 :                 status = NT_STATUS_OBJECT_PATH_NOT_FOUND;
     771           0 :                 goto out;
     772             :         }
     773             : 
     774             :         /*
     775             :          * One of the removed_components was the MSDFS link
     776             :          * at the end. We need to count this in the resolved
     777             :          * path below, so remove one from removed_components.
     778             :          */
     779        4448 :         removed_components--;
     780             : 
     781             :         /*
     782             :          * Now parent_smb_fname->fsp is the parent directory dirfsp,
     783             :          * last_component is the untranslated MS-DFS link name.
     784             :          * Search for it in the parent directory to get the real
     785             :          * filename on disk.
     786             :          */
     787        4448 :         status = get_real_filename_at(parent_smb_fname->fsp,
     788             :                                       last_component,
     789             :                                       ctx,
     790             :                                       &atname);
     791             : 
     792        4448 :         if (!NT_STATUS_IS_OK(status)) {
     793           0 :                 DBG_DEBUG("dfspath = %s. reqpath = %s "
     794             :                         "get_real_filename_at(%s, %s) error (%s)\n",
     795             :                         dfspath,
     796             :                         reqpath,
     797             :                         smb_fname_str_dbg(parent_smb_fname),
     798             :                         last_component,
     799             :                         nt_errstr(status));
     800           0 :                 goto out;
     801             :         }
     802             : 
     803        4448 :         smb_fname_rel = synthetic_smb_fname(ctx,
     804             :                                 atname,
     805             :                                 NULL,
     806             :                                 NULL,
     807             :                                 twrp,
     808             :                                 posix ? SMB_FILENAME_POSIX_PATH : 0);
     809        4448 :         if (smb_fname_rel == NULL) {
     810           0 :                 status = NT_STATUS_NO_MEMORY;
     811           0 :                 goto out;
     812             :         }
     813             : 
     814             :         /* Get the referral to return. */
     815        4448 :         status = SMB_VFS_READ_DFS_PATHAT(conn,
     816             :                                          ctx,
     817             :                                          parent_smb_fname->fsp,
     818             :                                          smb_fname_rel,
     819             :                                          ppreflist,
     820             :                                          preferral_count);
     821        4448 :         if (!NT_STATUS_IS_OK(status)) {
     822           0 :                 DBG_DEBUG("dfspath = %s. reqpath = %s. "
     823             :                         "SMB_VFS_READ_DFS_PATHAT(%s, %s) error (%s)\n",
     824             :                         dfspath,
     825             :                         reqpath,
     826             :                         smb_fname_str_dbg(parent_smb_fname),
     827             :                         smb_fname_str_dbg(smb_fname_rel),
     828             :                         nt_errstr(status));
     829           0 :                 goto out;
     830             :         }
     831             : 
     832             :         /*
     833             :          * Now we must work out how much of the
     834             :          * given pathname we consumed.
     835             :          */
     836        4448 :         canon_dfspath = talloc_strdup(ctx, dfspath);
     837        4448 :         if (!canon_dfspath) {
     838           0 :                 status = NT_STATUS_NO_MEMORY;
     839           0 :                 goto out;
     840             :         }
     841             :         /* Canonicalize the raw dfspath. */
     842        4448 :         string_replace(canon_dfspath, '\\', '/');
     843             : 
     844             :         /*
     845             :          * reqpath comes out of parse_dfs_path(), so it has
     846             :          * no trailing backslash. Make sure that canon_dfspath hasn't either.
     847             :          */
     848        4448 :         trim_char(canon_dfspath, 0, '/');
     849             : 
     850        4448 :         DBG_DEBUG("Unconsumed path: %s\n", canon_dfspath);
     851             : 
     852       45500 :         while (removed_components > 0) {
     853       41052 :                 p = strrchr(canon_dfspath, '/');
     854       41052 :                 if (p != NULL) {
     855       41052 :                         *p = '\0';
     856             :                 }
     857       41052 :                 removed_components--;
     858       41052 :                 if (p == NULL && removed_components != 0) {
     859           0 :                         DBG_ERR("Component mismatch. path = %s, "
     860             :                                 "%zu components left\n",
     861             :                                 canon_dfspath,
     862             :                                 removed_components);
     863           0 :                         status = NT_STATUS_OBJECT_PATH_NOT_FOUND;
     864           0 :                         goto out;
     865             :                 }
     866             :         }
     867        4448 :         *consumedcntp = strlen(canon_dfspath);
     868        4448 :         DBG_DEBUG("Path consumed: %s (%zu)\n", canon_dfspath, *consumedcntp);
     869        4448 :         status = NT_STATUS_OK;
     870             : 
     871        4448 :   out:
     872             : 
     873        4448 :         TALLOC_FREE(parent_smb_fname);
     874        4448 :         TALLOC_FREE(local_pathname);
     875        4448 :         TALLOC_FREE(last_component);
     876        4448 :         TALLOC_FREE(atname);
     877        4448 :         TALLOC_FREE(smb_fname_rel);
     878        4448 :         TALLOC_FREE(canon_dfspath);
     879        4448 :         return status;
     880             : }
     881             : 
     882             : /**********************************************************************
     883             :  Return a self referral.
     884             : **********************************************************************/
     885             : 
     886        1456 : static NTSTATUS self_ref(TALLOC_CTX *ctx,
     887             :                         const char *dfs_path,
     888             :                         struct junction_map *jucn,
     889             :                         size_t *consumedcntp,
     890             :                         bool *self_referralp)
     891             : {
     892           0 :         struct referral *ref;
     893             : 
     894        1456 :         *self_referralp = True;
     895             : 
     896        1456 :         jucn->referral_count = 1;
     897        1456 :         if((ref = talloc_zero(ctx, struct referral)) == NULL) {
     898           0 :                 return NT_STATUS_NO_MEMORY;
     899             :         }
     900             : 
     901        1456 :         ref->alternate_path = talloc_strdup(ctx, dfs_path);
     902        1456 :         if (!ref->alternate_path) {
     903           0 :                 TALLOC_FREE(ref);
     904           0 :                 return NT_STATUS_NO_MEMORY;
     905             :         }
     906        1456 :         ref->proximity = 0;
     907        1456 :         ref->ttl = REFERRAL_TTL;
     908        1456 :         jucn->referral_list = ref;
     909        1456 :         *consumedcntp = strlen(dfs_path);
     910        1456 :         return NT_STATUS_OK;
     911             : }
     912             : 
     913             : /**********************************************************************
     914             :  Gets valid referrals for a dfs path and fills up the
     915             :  junction_map structure.
     916             : **********************************************************************/
     917             : 
     918       14942 : NTSTATUS get_referred_path(TALLOC_CTX *ctx,
     919             :                            struct auth_session_info *session_info,
     920             :                            const char *dfs_path,
     921             :                            const struct tsocket_address *remote_address,
     922             :                            const struct tsocket_address *local_address,
     923             :                            struct junction_map *jucn,
     924             :                            size_t *consumedcntp,
     925             :                            bool *self_referralp)
     926             : {
     927       14942 :         TALLOC_CTX *frame = talloc_stackframe();
     928           0 :         const struct loadparm_substitution *lp_sub =
     929       14942 :                 loadparm_s3_global_substitution();
     930       14942 :         struct conn_struct_tos *c = NULL;
     931       14942 :         struct connection_struct *conn = NULL;
     932       14942 :         char *servicename = NULL;
     933       14942 :         char *reqpath = NULL;
     934           0 :         int snum;
     935       14942 :         NTSTATUS status = NT_STATUS_NOT_FOUND;
     936             : 
     937       14942 :         *self_referralp = False;
     938             : 
     939       14942 :         status = parse_dfs_path_strict(
     940             :                                 frame,
     941             :                                 dfs_path,
     942             :                                 NULL, /* hostname */
     943             :                                 &servicename,
     944             :                                 &reqpath);
     945       14942 :         if (!NT_STATUS_IS_OK(status)) {
     946           2 :                 TALLOC_FREE(frame);
     947           2 :                 return status;
     948             :         }
     949             : 
     950             :         /* Path referrals are always non-POSIX. */
     951       14940 :         status = check_path_syntax(reqpath, false);
     952       14940 :         if (!NT_STATUS_IS_OK(status)) {
     953           0 :                 TALLOC_FREE(frame);
     954           0 :                 return status;
     955             :         }
     956             : 
     957       14940 :         jucn->service_name = talloc_strdup(ctx, servicename);
     958       14940 :         jucn->volume_name = talloc_strdup(ctx, reqpath);
     959       14940 :         if (!jucn->service_name || !jucn->volume_name) {
     960           0 :                 TALLOC_FREE(frame);
     961           0 :                 return NT_STATUS_NO_MEMORY;
     962             :         }
     963             : 
     964             :         /* Verify the share is a dfs root */
     965       14940 :         snum = lp_servicenumber(jucn->service_name);
     966       14940 :         if(snum < 0) {
     967          32 :                 char *service_name = NULL;
     968          32 :                 if ((snum = find_service(ctx, jucn->service_name, &service_name)) < 0) {
     969          30 :                         TALLOC_FREE(frame);
     970          30 :                         return NT_STATUS_NOT_FOUND;
     971             :                 }
     972           2 :                 if (!service_name) {
     973           0 :                         TALLOC_FREE(frame);
     974           0 :                         return NT_STATUS_NO_MEMORY;
     975             :                 }
     976           2 :                 TALLOC_FREE(jucn->service_name);
     977           2 :                 jucn->service_name = talloc_strdup(ctx, service_name);
     978           2 :                 if (!jucn->service_name) {
     979           0 :                         TALLOC_FREE(frame);
     980           0 :                         return NT_STATUS_NO_MEMORY;
     981             :                 }
     982             :         }
     983             : 
     984       14910 :         if (!lp_msdfs_root(snum) && (*lp_msdfs_proxy(talloc_tos(), lp_sub, snum) == '\0')) {
     985        9006 :                 DEBUG(3,("get_referred_path: |%s| in dfs path %s is not "
     986             :                         "a dfs root.\n",
     987             :                         servicename, dfs_path));
     988        9006 :                 TALLOC_FREE(frame);
     989        9006 :                 return NT_STATUS_NOT_FOUND;
     990             :         }
     991             : 
     992             :         /*
     993             :          * Self referrals are tested with a anonymous IPC connection and
     994             :          * a GET_DFS_REFERRAL call to \\server\share. (which means
     995             :          * dp.reqpath[0] points to an empty string). create_conn_struct cd's
     996             :          * into the directory and will fail if it cannot (as the anonymous
     997             :          * user). Cope with this.
     998             :          */
     999             : 
    1000        5904 :         if (reqpath[0] == '\0') {
    1001           0 :                 char *tmp;
    1002           0 :                 struct referral *ref;
    1003           0 :                 size_t refcount;
    1004             : 
    1005        1456 :                 if (*lp_msdfs_proxy(talloc_tos(), lp_sub, snum) == '\0') {
    1006        1456 :                         TALLOC_FREE(frame);
    1007        1456 :                         return self_ref(ctx,
    1008             :                                         dfs_path,
    1009             :                                         jucn,
    1010             :                                         consumedcntp,
    1011             :                                         self_referralp);
    1012             :                 }
    1013             : 
    1014             :                 /*
    1015             :                  * It's an msdfs proxy share. Redirect to
    1016             :                  * the configured target share.
    1017             :                  */
    1018             : 
    1019           0 :                 tmp = talloc_asprintf(frame, "msdfs:%s",
    1020             :                                       lp_msdfs_proxy(frame, lp_sub, snum));
    1021           0 :                 if (tmp == NULL) {
    1022           0 :                         TALLOC_FREE(frame);
    1023           0 :                         return NT_STATUS_NO_MEMORY;
    1024             :                 }
    1025             : 
    1026           0 :                 if (!parse_msdfs_symlink(ctx,
    1027           0 :                                 lp_msdfs_shuffle_referrals(snum),
    1028             :                                 tmp,
    1029             :                                 &ref,
    1030             :                                 &refcount)) {
    1031           0 :                         TALLOC_FREE(frame);
    1032           0 :                         return NT_STATUS_INVALID_PARAMETER;
    1033             :                 }
    1034           0 :                 jucn->referral_count = refcount;
    1035           0 :                 jucn->referral_list = ref;
    1036           0 :                 *consumedcntp = strlen(dfs_path);
    1037           0 :                 TALLOC_FREE(frame);
    1038           0 :                 return NT_STATUS_OK;
    1039             :         }
    1040             : 
    1041        4448 :         status = create_conn_struct_tos_cwd(global_messaging_context(),
    1042             :                                             snum,
    1043        4448 :                                             lp_path(frame, lp_sub, snum),
    1044             :                                             session_info,
    1045             :                                             &c);
    1046        4448 :         if (!NT_STATUS_IS_OK(status)) {
    1047           0 :                 TALLOC_FREE(frame);
    1048           0 :                 return status;
    1049             :         }
    1050        4448 :         conn = c->conn;
    1051             : 
    1052             :         /*
    1053             :          * TODO
    1054             :          *
    1055             :          * The remote and local address should be passed down to
    1056             :          * create_conn_struct_cwd.
    1057             :          */
    1058        4448 :         if (conn->sconn->remote_address == NULL) {
    1059        8896 :                 conn->sconn->remote_address =
    1060        4448 :                         tsocket_address_copy(remote_address, conn->sconn);
    1061        4448 :                 if (conn->sconn->remote_address == NULL) {
    1062           0 :                         TALLOC_FREE(frame);
    1063           0 :                         return NT_STATUS_NO_MEMORY;
    1064             :                 }
    1065             :         }
    1066        4448 :         if (conn->sconn->local_address == NULL) {
    1067        8896 :                 conn->sconn->local_address =
    1068        4448 :                         tsocket_address_copy(local_address, conn->sconn);
    1069        4448 :                 if (conn->sconn->local_address == NULL) {
    1070           0 :                         TALLOC_FREE(frame);
    1071           0 :                         return NT_STATUS_NO_MEMORY;
    1072             :                 }
    1073             :         }
    1074             : 
    1075        4448 :         status = dfs_path_lookup(ctx,
    1076             :                                 conn,
    1077             :                                 dfs_path,
    1078             :                                 reqpath,
    1079             :                                 0, /* ucf_flags */
    1080             :                                 consumedcntp,
    1081             :                                 &jucn->referral_list,
    1082             :                                 &jucn->referral_count);
    1083             : 
    1084        4448 :         if (!NT_STATUS_IS_OK(status)) {
    1085           0 :                 DBG_NOTICE("No valid referrals for path %s (%s)\n",
    1086             :                         dfs_path,
    1087             :                         nt_errstr(status));
    1088             :         }
    1089             : 
    1090        4448 :         TALLOC_FREE(frame);
    1091        4448 :         return status;
    1092             : }
    1093             : 
    1094             : /******************************************************************
    1095             :  Set up the DFS referral for the dfs pathname. This call returns
    1096             :  the amount of the path covered by this server, and where the
    1097             :  client should be redirected to. This is the meat of the
    1098             :  TRANS2_GET_DFS_REFERRAL call.
    1099             : ******************************************************************/
    1100             : 
    1101       14972 : int setup_dfs_referral(connection_struct *orig_conn,
    1102             :                         const char *dfs_path,
    1103             :                         int max_referral_level,
    1104             :                         char **ppdata, NTSTATUS *pstatus)
    1105             : {
    1106       14972 :         char *pdata = *ppdata;
    1107       14972 :         int reply_size = 0;
    1108           0 :         struct dfs_GetDFSReferral *r;
    1109       14972 :         DATA_BLOB blob = data_blob_null;
    1110           0 :         NTSTATUS status;
    1111           0 :         enum ndr_err_code ndr_err;
    1112             : 
    1113       14972 :         r = talloc_zero(talloc_tos(), struct dfs_GetDFSReferral);
    1114       14972 :         if (r == NULL) {
    1115           0 :                 *pstatus = NT_STATUS_NO_MEMORY;
    1116           0 :                 return -1;
    1117             :         }
    1118             : 
    1119       14972 :         r->in.req.max_referral_level = max_referral_level;
    1120       14972 :         r->in.req.servername = talloc_strdup(r, dfs_path);
    1121       14972 :         if (r->in.req.servername == NULL) {
    1122           0 :                 talloc_free(r);
    1123           0 :                 *pstatus = NT_STATUS_NO_MEMORY;
    1124           0 :                 return -1;
    1125             :         }
    1126             : 
    1127       14972 :         status = SMB_VFS_GET_DFS_REFERRALS(orig_conn, r);
    1128       14972 :         if (!NT_STATUS_IS_OK(status)) {
    1129        9042 :                 talloc_free(r);
    1130        9042 :                 *pstatus = status;
    1131        9042 :                 return -1;
    1132             :         }
    1133             : 
    1134        5930 :         ndr_err = ndr_push_struct_blob(&blob, r,
    1135        5930 :                                 r->out.resp,
    1136             :                                 (ndr_push_flags_fn_t)ndr_push_dfs_referral_resp);
    1137        5930 :         if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
    1138           0 :                 TALLOC_FREE(r);
    1139           0 :                 *pstatus = NT_STATUS_INVALID_PARAMETER;
    1140           0 :                 return -1;
    1141             :         }
    1142             : 
    1143        5930 :         pdata = (char *)SMB_REALLOC(pdata, blob.length);
    1144        5930 :         if(pdata == NULL) {
    1145           0 :                 TALLOC_FREE(r);
    1146           0 :                 DEBUG(0,("referral setup:"
    1147             :                          "malloc failed for Realloc!\n"));
    1148           0 :                 return -1;
    1149             :         }
    1150        5930 :         *ppdata = pdata;
    1151        5930 :         reply_size = blob.length;
    1152        5930 :         memcpy(pdata, blob.data, blob.length);
    1153        5930 :         TALLOC_FREE(r);
    1154             : 
    1155        5930 :         *pstatus = NT_STATUS_OK;
    1156        5930 :         return reply_size;
    1157             : }
    1158             : 
    1159             : /**********************************************************************
    1160             :  The following functions are called by the NETDFS RPC pipe functions
    1161             :  **********************************************************************/
    1162             : 
    1163             : /*********************************************************************
    1164             :  Creates a junction structure from a DFS pathname
    1165             : **********************************************************************/
    1166             : 
    1167           2 : bool create_junction(TALLOC_CTX *ctx,
    1168             :                 const char *dfs_path,
    1169             :                 struct junction_map *jucn)
    1170             : {
    1171           0 :         const struct loadparm_substitution *lp_sub =
    1172           2 :                 loadparm_s3_global_substitution();
    1173           0 :         int snum;
    1174           2 :         char *servicename = NULL;
    1175           2 :         char *reqpath = NULL;
    1176           0 :         NTSTATUS status;
    1177             : 
    1178           2 :         status = parse_dfs_path_strict(
    1179             :                                 ctx,
    1180             :                                 dfs_path,
    1181             :                                 NULL,
    1182             :                                 &servicename,
    1183             :                                 &reqpath);
    1184           2 :         if (!NT_STATUS_IS_OK(status)) {
    1185           0 :                 return False;
    1186             :         }
    1187             : 
    1188             :         /* Check for a non-DFS share */
    1189           2 :         snum = lp_servicenumber(servicename);
    1190             : 
    1191           2 :         if(snum < 0 || !lp_msdfs_root(snum)) {
    1192           0 :                 DEBUG(4,("create_junction: %s is not an msdfs root.\n",
    1193             :                         servicename));
    1194           0 :                 return False;
    1195             :         }
    1196             : 
    1197             :         /* Junction create paths are always non-POSIX. */
    1198           2 :         status = check_path_syntax(reqpath, false);
    1199           2 :         if (!NT_STATUS_IS_OK(status)) {
    1200           0 :                 return false;
    1201             :         }
    1202             : 
    1203           2 :         jucn->service_name = talloc_strdup(ctx, servicename);
    1204           2 :         jucn->volume_name = talloc_strdup(ctx, reqpath);
    1205           2 :         jucn->comment = lp_comment(ctx, lp_sub, snum);
    1206             : 
    1207           2 :         if (!jucn->service_name || !jucn->volume_name || ! jucn->comment) {
    1208           0 :                 return False;
    1209             :         }
    1210           2 :         return True;
    1211             : }
    1212             : 
    1213             : /**********************************************************************
    1214             :  Forms a valid Unix pathname from the junction
    1215             :  **********************************************************************/
    1216             : 
    1217           0 : static bool junction_to_local_path_tos(const struct junction_map *jucn,
    1218             :                                        struct auth_session_info *session_info,
    1219             :                                        char **pp_path_out,
    1220             :                                        connection_struct **conn_out)
    1221             : {
    1222           0 :         const struct loadparm_substitution *lp_sub =
    1223           0 :                 loadparm_s3_global_substitution();
    1224           0 :         struct conn_struct_tos *c = NULL;
    1225           0 :         int snum;
    1226           0 :         char *path_out = NULL;
    1227           0 :         NTSTATUS status;
    1228             : 
    1229           0 :         snum = lp_servicenumber(jucn->service_name);
    1230           0 :         if(snum < 0) {
    1231           0 :                 return False;
    1232             :         }
    1233           0 :         status = create_conn_struct_tos_cwd(global_messaging_context(),
    1234             :                                             snum,
    1235           0 :                                             lp_path(talloc_tos(), lp_sub, snum),
    1236             :                                             session_info,
    1237             :                                             &c);
    1238           0 :         if (!NT_STATUS_IS_OK(status)) {
    1239           0 :                 return False;
    1240             :         }
    1241             : 
    1242           0 :         path_out = talloc_asprintf(c,
    1243             :                         "%s/%s",
    1244             :                         lp_path(talloc_tos(), lp_sub, snum),
    1245           0 :                         jucn->volume_name);
    1246           0 :         if (path_out == NULL) {
    1247           0 :                 TALLOC_FREE(c);
    1248           0 :                 return False;
    1249             :         }
    1250           0 :         *pp_path_out = path_out;
    1251           0 :         *conn_out = c->conn;
    1252           0 :         return True;
    1253             : }
    1254             : 
    1255             : /*
    1256             :  * Create a msdfs string in Samba format we can store
    1257             :  * in a filesystem object (currently a symlink).
    1258             :  */
    1259             : 
    1260           0 : char *msdfs_link_string(TALLOC_CTX *ctx,
    1261             :                         const struct referral *reflist,
    1262             :                         size_t referral_count)
    1263             : {
    1264           0 :         char *refpath = NULL;
    1265           0 :         bool insert_comma = false;
    1266           0 :         char *msdfs_link = NULL;
    1267           0 :         size_t i;
    1268             : 
    1269             :         /* Form the msdfs_link contents */
    1270           0 :         msdfs_link = talloc_strdup(ctx, "msdfs:");
    1271           0 :         if (msdfs_link == NULL) {
    1272           0 :                 goto err;
    1273             :         }
    1274             : 
    1275           0 :         for( i= 0; i < referral_count; i++) {
    1276           0 :                 refpath = talloc_strdup(ctx, reflist[i].alternate_path);
    1277             : 
    1278           0 :                 if (refpath == NULL) {
    1279           0 :                         goto err;
    1280             :                 }
    1281             : 
    1282             :                 /* Alternate paths always use Windows separators. */
    1283           0 :                 trim_char(refpath, '\\', '\\');
    1284           0 :                 if (*refpath == '\0') {
    1285           0 :                         if (i == 0) {
    1286           0 :                                 insert_comma = false;
    1287             :                         }
    1288           0 :                         continue;
    1289             :                 }
    1290           0 :                 if (i > 0 && insert_comma) {
    1291           0 :                         msdfs_link = talloc_asprintf_append_buffer(msdfs_link,
    1292             :                                         ",%s",
    1293             :                                         refpath);
    1294             :                 } else {
    1295           0 :                         msdfs_link = talloc_asprintf_append_buffer(msdfs_link,
    1296             :                                         "%s",
    1297             :                                         refpath);
    1298             :                 }
    1299             : 
    1300           0 :                 if (msdfs_link == NULL) {
    1301           0 :                         goto err;
    1302             :                 }
    1303             : 
    1304           0 :                 if (!insert_comma) {
    1305           0 :                         insert_comma = true;
    1306             :                 }
    1307             : 
    1308           0 :                 TALLOC_FREE(refpath);
    1309             :         }
    1310             : 
    1311           0 :         return msdfs_link;
    1312             : 
    1313           0 :   err:
    1314             : 
    1315           0 :         TALLOC_FREE(refpath);
    1316           0 :         TALLOC_FREE(msdfs_link);
    1317           0 :         return NULL;
    1318             : }
    1319             : 
    1320           0 : bool create_msdfs_link(const struct junction_map *jucn,
    1321             :                        struct auth_session_info *session_info)
    1322             : {
    1323           0 :         TALLOC_CTX *frame = talloc_stackframe();
    1324           0 :         char *path = NULL;
    1325           0 :         connection_struct *conn;
    1326           0 :         struct smb_filename *smb_fname = NULL;
    1327           0 :         struct smb_filename *parent_fname = NULL;
    1328           0 :         struct smb_filename *at_fname = NULL;
    1329           0 :         bool ok;
    1330           0 :         NTSTATUS status;
    1331           0 :         bool ret = false;
    1332             : 
    1333           0 :         ok = junction_to_local_path_tos(jucn, session_info, &path, &conn);
    1334           0 :         if (!ok) {
    1335           0 :                 goto out;
    1336             :         }
    1337             : 
    1338           0 :         if (!CAN_WRITE(conn)) {
    1339           0 :                 const struct loadparm_substitution *lp_sub =
    1340           0 :                         loadparm_s3_global_substitution();
    1341           0 :                 int snum = lp_servicenumber(jucn->service_name);
    1342             : 
    1343           0 :                 DBG_WARNING("Can't create DFS entry on read-only share %s\n",
    1344             :                         lp_servicename(frame, lp_sub, snum));
    1345           0 :                 goto out;
    1346             :         }
    1347             : 
    1348           0 :         smb_fname = synthetic_smb_fname(frame,
    1349             :                                 path,
    1350             :                                 NULL,
    1351             :                                 NULL,
    1352             :                                 0,
    1353             :                                 0);
    1354           0 :         if (smb_fname == NULL) {
    1355           0 :                 goto out;
    1356             :         }
    1357             : 
    1358           0 :         status = parent_pathref(frame,
    1359           0 :                                 conn->cwd_fsp,
    1360             :                                 smb_fname,
    1361             :                                 &parent_fname,
    1362             :                                 &at_fname);
    1363           0 :         if (!NT_STATUS_IS_OK(status)) {
    1364           0 :                 goto out;
    1365             :         }
    1366             : 
    1367           0 :         status = SMB_VFS_CREATE_DFS_PATHAT(conn,
    1368             :                                 parent_fname->fsp,
    1369             :                                 at_fname,
    1370             :                                 jucn->referral_list,
    1371             :                                 jucn->referral_count);
    1372           0 :         if (!NT_STATUS_IS_OK(status)) {
    1373           0 :                 if (NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_NAME_COLLISION)) {
    1374           0 :                         int retval = SMB_VFS_UNLINKAT(conn,
    1375             :                                                 parent_fname->fsp,
    1376             :                                                 at_fname,
    1377             :                                                 0);
    1378           0 :                         if (retval != 0) {
    1379           0 :                                 goto out;
    1380             :                         }
    1381             :                 }
    1382           0 :                 status = SMB_VFS_CREATE_DFS_PATHAT(conn,
    1383             :                                 parent_fname->fsp,
    1384             :                                 at_fname,
    1385             :                                 jucn->referral_list,
    1386             :                                 jucn->referral_count);
    1387           0 :                 if (!NT_STATUS_IS_OK(status)) {
    1388           0 :                         DBG_WARNING("SMB_VFS_CREATE_DFS_PATHAT failed "
    1389             :                                 "%s - Error: %s\n",
    1390             :                                 path,
    1391             :                                 nt_errstr(status));
    1392           0 :                         goto out;
    1393             :                 }
    1394             :         }
    1395             : 
    1396           0 :         ret = true;
    1397             : 
    1398           0 : out:
    1399           0 :         TALLOC_FREE(frame);
    1400           0 :         return ret;
    1401             : }
    1402             : 
    1403           0 : bool remove_msdfs_link(const struct junction_map *jucn,
    1404             :                        struct auth_session_info *session_info)
    1405             : {
    1406           0 :         TALLOC_CTX *frame = talloc_stackframe();
    1407           0 :         char *path = NULL;
    1408           0 :         connection_struct *conn;
    1409           0 :         bool ret = False;
    1410           0 :         struct smb_filename *smb_fname;
    1411           0 :         struct smb_filename *parent_fname = NULL;
    1412           0 :         struct smb_filename *at_fname = NULL;
    1413           0 :         NTSTATUS status;
    1414           0 :         bool ok;
    1415           0 :         int retval;
    1416             : 
    1417           0 :         ok = junction_to_local_path_tos(jucn, session_info, &path, &conn);
    1418           0 :         if (!ok) {
    1419           0 :                 TALLOC_FREE(frame);
    1420           0 :                 return false;
    1421             :         }
    1422             : 
    1423           0 :         if (!CAN_WRITE(conn)) {
    1424           0 :                 const struct loadparm_substitution *lp_sub =
    1425           0 :                         loadparm_s3_global_substitution();
    1426           0 :                 int snum = lp_servicenumber(jucn->service_name);
    1427             : 
    1428           0 :                 DBG_WARNING("Can't remove DFS entry on read-only share %s\n",
    1429             :                         lp_servicename(frame, lp_sub, snum));
    1430           0 :                 TALLOC_FREE(frame);
    1431           0 :                 return false;
    1432             :         }
    1433             : 
    1434           0 :         smb_fname = synthetic_smb_fname(frame,
    1435             :                                         path,
    1436             :                                         NULL,
    1437             :                                         NULL,
    1438             :                                         0,
    1439             :                                         0);
    1440           0 :         if (smb_fname == NULL) {
    1441           0 :                 TALLOC_FREE(frame);
    1442           0 :                 errno = ENOMEM;
    1443           0 :                 return false;
    1444             :         }
    1445             : 
    1446           0 :         status = parent_pathref(frame,
    1447           0 :                                 conn->cwd_fsp,
    1448             :                                 smb_fname,
    1449             :                                 &parent_fname,
    1450             :                                 &at_fname);
    1451           0 :         if (!NT_STATUS_IS_OK(status)) {
    1452           0 :                 TALLOC_FREE(frame);
    1453           0 :                 return false;
    1454             :         }
    1455             : 
    1456           0 :         retval = SMB_VFS_UNLINKAT(conn,
    1457             :                         parent_fname->fsp,
    1458             :                         at_fname,
    1459             :                         0);
    1460           0 :         if (retval == 0) {
    1461           0 :                 ret = True;
    1462             :         }
    1463             : 
    1464           0 :         TALLOC_FREE(frame);
    1465           0 :         return ret;
    1466             : }
    1467             : 
    1468             : /*********************************************************************
    1469             :  Return the number of DFS links at the root of this share.
    1470             : *********************************************************************/
    1471             : 
    1472           0 : static size_t count_dfs_links(TALLOC_CTX *ctx,
    1473             :                               struct auth_session_info *session_info,
    1474             :                               int snum)
    1475             : {
    1476           0 :         TALLOC_CTX *frame = talloc_stackframe();
    1477           0 :         const struct loadparm_substitution *lp_sub =
    1478           0 :                 loadparm_s3_global_substitution();
    1479           0 :         size_t cnt = 0;
    1480           0 :         const char *dname = NULL;
    1481           0 :         char *talloced = NULL;
    1482           0 :         const char *connect_path = lp_path(frame, lp_sub, snum);
    1483           0 :         const char *msdfs_proxy = lp_msdfs_proxy(frame, lp_sub, snum);
    1484           0 :         struct conn_struct_tos *c = NULL;
    1485           0 :         connection_struct *conn = NULL;
    1486           0 :         NTSTATUS status;
    1487           0 :         struct smb_filename *smb_fname = NULL;
    1488           0 :         struct smb_Dir *dir_hnd = NULL;
    1489             : 
    1490           0 :         if(*connect_path == '\0') {
    1491           0 :                 TALLOC_FREE(frame);
    1492           0 :                 return 0;
    1493             :         }
    1494             : 
    1495             :         /*
    1496             :          * Fake up a connection struct for the VFS layer.
    1497             :          */
    1498             : 
    1499           0 :         status = create_conn_struct_tos_cwd(global_messaging_context(),
    1500             :                                             snum,
    1501             :                                             connect_path,
    1502             :                                             session_info,
    1503             :                                             &c);
    1504           0 :         if (!NT_STATUS_IS_OK(status)) {
    1505           0 :                 DEBUG(3, ("create_conn_struct failed: %s\n",
    1506             :                           nt_errstr(status)));
    1507           0 :                 TALLOC_FREE(frame);
    1508           0 :                 return 0;
    1509             :         }
    1510           0 :         conn = c->conn;
    1511             : 
    1512             :         /* Count a link for the msdfs root - convention */
    1513           0 :         cnt = 1;
    1514             : 
    1515             :         /* No more links if this is an msdfs proxy. */
    1516           0 :         if (*msdfs_proxy != '\0') {
    1517           0 :                 goto out;
    1518             :         }
    1519             : 
    1520           0 :         smb_fname = synthetic_smb_fname(frame,
    1521             :                                         ".",
    1522             :                                         NULL,
    1523             :                                         NULL,
    1524             :                                         0,
    1525             :                                         0);
    1526           0 :         if (smb_fname == NULL) {
    1527           0 :                 goto out;
    1528             :         }
    1529             : 
    1530             :         /* Now enumerate all dfs links */
    1531           0 :         status = OpenDir(frame,
    1532             :                          conn,
    1533             :                          smb_fname,
    1534             :                          NULL,
    1535             :                          0,
    1536             :                          &dir_hnd);
    1537           0 :         if (!NT_STATUS_IS_OK(status)) {
    1538           0 :                 errno = map_errno_from_nt_status(status);
    1539           0 :                 goto out;
    1540             :         }
    1541             : 
    1542           0 :         while ((dname = ReadDirName(dir_hnd, &talloced)) != NULL) {
    1543           0 :                 struct smb_filename *smb_dname =
    1544           0 :                         synthetic_smb_fname(frame,
    1545             :                                         dname,
    1546             :                                         NULL,
    1547             :                                         NULL,
    1548             :                                         0,
    1549             :                                         0);
    1550           0 :                 if (smb_dname == NULL) {
    1551           0 :                         goto out;
    1552             :                 }
    1553           0 :                 if (is_msdfs_link(dir_hnd_fetch_fsp(dir_hnd), smb_dname)) {
    1554           0 :                         if (cnt + 1 < cnt) {
    1555           0 :                                 cnt = 0;
    1556           0 :                                 goto out;
    1557             :                         }
    1558           0 :                         cnt++;
    1559             :                 }
    1560           0 :                 TALLOC_FREE(talloced);
    1561           0 :                 TALLOC_FREE(smb_dname);
    1562             :         }
    1563             : 
    1564           0 : out:
    1565           0 :         TALLOC_FREE(frame);
    1566           0 :         return cnt;
    1567             : }
    1568             : 
    1569             : /*********************************************************************
    1570             : *********************************************************************/
    1571             : 
    1572           0 : static int form_junctions(TALLOC_CTX *ctx,
    1573             :                           struct auth_session_info *session_info,
    1574             :                                 int snum,
    1575             :                                 struct junction_map *jucn,
    1576             :                                 size_t jn_remain)
    1577             : {
    1578           0 :         TALLOC_CTX *frame = talloc_stackframe();
    1579           0 :         const struct loadparm_substitution *lp_sub =
    1580           0 :                 loadparm_s3_global_substitution();
    1581           0 :         size_t cnt = 0;
    1582           0 :         const char *dname = NULL;
    1583           0 :         char *talloced = NULL;
    1584           0 :         const char *connect_path = lp_path(frame, lp_sub, snum);
    1585           0 :         char *service_name = lp_servicename(frame, lp_sub, snum);
    1586           0 :         const char *msdfs_proxy = lp_msdfs_proxy(frame, lp_sub, snum);
    1587           0 :         struct conn_struct_tos *c = NULL;
    1588           0 :         connection_struct *conn = NULL;
    1589           0 :         struct referral *ref = NULL;
    1590           0 :         struct smb_filename *smb_fname = NULL;
    1591           0 :         struct smb_Dir *dir_hnd = NULL;
    1592           0 :         NTSTATUS status;
    1593             : 
    1594           0 :         if (jn_remain == 0) {
    1595           0 :                 TALLOC_FREE(frame);
    1596           0 :                 return 0;
    1597             :         }
    1598             : 
    1599           0 :         if(*connect_path == '\0') {
    1600           0 :                 TALLOC_FREE(frame);
    1601           0 :                 return 0;
    1602             :         }
    1603             : 
    1604             :         /*
    1605             :          * Fake up a connection struct for the VFS layer.
    1606             :          */
    1607             : 
    1608           0 :         status = create_conn_struct_tos_cwd(global_messaging_context(),
    1609             :                                             snum,
    1610             :                                             connect_path,
    1611             :                                             session_info,
    1612             :                                             &c);
    1613           0 :         if (!NT_STATUS_IS_OK(status)) {
    1614           0 :                 DEBUG(3, ("create_conn_struct failed: %s\n",
    1615             :                           nt_errstr(status)));
    1616           0 :                 TALLOC_FREE(frame);
    1617           0 :                 return 0;
    1618             :         }
    1619           0 :         conn = c->conn;
    1620             : 
    1621             :         /* form a junction for the msdfs root - convention
    1622             :            DO NOT REMOVE THIS: NT clients will not work with us
    1623             :            if this is not present
    1624             :         */
    1625           0 :         jucn[cnt].service_name = talloc_strdup(ctx,service_name);
    1626           0 :         jucn[cnt].volume_name = talloc_strdup(ctx, "");
    1627           0 :         if (!jucn[cnt].service_name || !jucn[cnt].volume_name) {
    1628           0 :                 goto out;
    1629             :         }
    1630           0 :         jucn[cnt].comment = "";
    1631           0 :         jucn[cnt].referral_count = 1;
    1632             : 
    1633           0 :         ref = jucn[cnt].referral_list = talloc_zero(ctx, struct referral);
    1634           0 :         if (jucn[cnt].referral_list == NULL) {
    1635           0 :                 goto out;
    1636             :         }
    1637             : 
    1638           0 :         ref->proximity = 0;
    1639           0 :         ref->ttl = REFERRAL_TTL;
    1640           0 :         if (*msdfs_proxy != '\0') {
    1641           0 :                 ref->alternate_path = talloc_strdup(ctx,
    1642             :                                                 msdfs_proxy);
    1643             :         } else {
    1644           0 :                 ref->alternate_path = talloc_asprintf(ctx,
    1645             :                         "\\\\%s\\%s",
    1646             :                         get_local_machine_name(),
    1647             :                         service_name);
    1648             :         }
    1649             : 
    1650           0 :         if (!ref->alternate_path) {
    1651           0 :                 goto out;
    1652             :         }
    1653           0 :         cnt++;
    1654             : 
    1655             :         /* Don't enumerate if we're an msdfs proxy. */
    1656           0 :         if (*msdfs_proxy != '\0') {
    1657           0 :                 goto out;
    1658             :         }
    1659             : 
    1660           0 :         smb_fname = synthetic_smb_fname(frame,
    1661             :                                         ".",
    1662             :                                         NULL,
    1663             :                                         NULL,
    1664             :                                         0,
    1665             :                                         0);
    1666           0 :         if (smb_fname == NULL) {
    1667           0 :                 goto out;
    1668             :         }
    1669             : 
    1670             :         /* Now enumerate all dfs links */
    1671           0 :         status = OpenDir(frame,
    1672             :                          conn,
    1673             :                          smb_fname,
    1674             :                          NULL,
    1675             :                          0,
    1676             :                          &dir_hnd);
    1677           0 :         if (!NT_STATUS_IS_OK(status)) {
    1678           0 :                 errno = map_errno_from_nt_status(status);
    1679           0 :                 goto out;
    1680             :         }
    1681             : 
    1682           0 :         while ((dname = ReadDirName(dir_hnd, &talloced)) != NULL) {
    1683           0 :                 struct smb_filename *smb_dname = NULL;
    1684             : 
    1685           0 :                 if (cnt >= jn_remain) {
    1686           0 :                         DEBUG(2, ("form_junctions: ran out of MSDFS "
    1687             :                                 "junction slots\n"));
    1688           0 :                         TALLOC_FREE(talloced);
    1689           0 :                         goto out;
    1690             :                 }
    1691           0 :                 smb_dname = synthetic_smb_fname(talloc_tos(),
    1692             :                                 dname,
    1693             :                                 NULL,
    1694             :                                 NULL,
    1695             :                                 0,
    1696             :                                 0);
    1697           0 :                 if (smb_dname == NULL) {
    1698           0 :                         TALLOC_FREE(talloced);
    1699           0 :                         goto out;
    1700             :                 }
    1701             : 
    1702           0 :                 status = SMB_VFS_READ_DFS_PATHAT(conn,
    1703             :                                 ctx,
    1704             :                                 conn->cwd_fsp,
    1705             :                                 smb_dname,
    1706             :                                 &jucn[cnt].referral_list,
    1707             :                                 &jucn[cnt].referral_count);
    1708             : 
    1709           0 :                 if (NT_STATUS_IS_OK(status)) {
    1710           0 :                         jucn[cnt].service_name = talloc_strdup(ctx,
    1711             :                                                         service_name);
    1712           0 :                         jucn[cnt].volume_name = talloc_strdup(ctx, dname);
    1713           0 :                         if (!jucn[cnt].service_name || !jucn[cnt].volume_name) {
    1714           0 :                                 TALLOC_FREE(talloced);
    1715           0 :                                 goto out;
    1716             :                         }
    1717           0 :                         jucn[cnt].comment = "";
    1718           0 :                         cnt++;
    1719             :                 }
    1720           0 :                 TALLOC_FREE(talloced);
    1721           0 :                 TALLOC_FREE(smb_dname);
    1722             :         }
    1723             : 
    1724           0 : out:
    1725           0 :         TALLOC_FREE(frame);
    1726           0 :         return cnt;
    1727             : }
    1728             : 
    1729           0 : struct junction_map *enum_msdfs_links(TALLOC_CTX *ctx,
    1730             :                                       struct auth_session_info *session_info,
    1731             :                                       size_t *p_num_jn)
    1732             : {
    1733           0 :         struct junction_map *jn = NULL;
    1734           0 :         int i=0;
    1735           0 :         size_t jn_count = 0;
    1736           0 :         int sharecount = 0;
    1737             : 
    1738           0 :         *p_num_jn = 0;
    1739           0 :         if(!lp_host_msdfs()) {
    1740           0 :                 return NULL;
    1741             :         }
    1742             : 
    1743             :         /* Ensure all the usershares are loaded. */
    1744           0 :         become_root();
    1745           0 :         load_registry_shares();
    1746           0 :         sharecount = load_usershare_shares(NULL, connections_snum_used);
    1747           0 :         unbecome_root();
    1748             : 
    1749           0 :         for(i=0;i < sharecount;i++) {
    1750           0 :                 if(lp_msdfs_root(i)) {
    1751           0 :                         jn_count += count_dfs_links(ctx, session_info, i);
    1752             :                 }
    1753             :         }
    1754           0 :         if (jn_count == 0) {
    1755           0 :                 return NULL;
    1756             :         }
    1757           0 :         jn = talloc_array(ctx,  struct junction_map, jn_count);
    1758           0 :         if (!jn) {
    1759           0 :                 return NULL;
    1760             :         }
    1761           0 :         for(i=0; i < sharecount; i++) {
    1762           0 :                 if (*p_num_jn >= jn_count) {
    1763           0 :                         break;
    1764             :                 }
    1765           0 :                 if(lp_msdfs_root(i)) {
    1766           0 :                         *p_num_jn += form_junctions(ctx,
    1767             :                                         session_info,
    1768             :                                         i,
    1769           0 :                                         &jn[*p_num_jn],
    1770           0 :                                         jn_count - *p_num_jn);
    1771             :                 }
    1772             :         }
    1773           0 :         return jn;
    1774             : }

Generated by: LCOV version 1.14