LCOV - code coverage report
Current view: top level - nsswitch/libwbclient - wbclient.c (source / functions) Hit Total Coverage
Test: coverage report for master 98b443d9 Lines: 96 136 70.6 %
Date: 2024-05-31 13:13:24 Functions: 15 16 93.8 %

          Line data    Source code
       1             : /*
       2             :    Unix SMB/CIFS implementation.
       3             : 
       4             :    Winbind client API
       5             : 
       6             :    Copyright (C) Gerald (Jerry) Carter 2007
       7             :    Copyright (C) Matthew Newton 2015
       8             : 
       9             : 
      10             :    This library is free software; you can redistribute it and/or
      11             :    modify it under the terms of the GNU Lesser General Public
      12             :    License as published by the Free Software Foundation; either
      13             :    version 3 of the License, or (at your option) any later version.
      14             : 
      15             :    This library is distributed in the hope that it will be useful,
      16             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      17             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
      18             :    Library General Public License for more details.
      19             : 
      20             :    You should have received a copy of the GNU Lesser General Public License
      21             :    along with this program.  If not, see <http://www.gnu.org/licenses/>.
      22             : */
      23             : 
      24             : /* Required Headers */
      25             : 
      26             : #include "replace.h"
      27             : #include "libwbclient.h"
      28             : 
      29             : /* From wb_common.c */
      30             : 
      31             : struct winbindd_context;
      32             : 
      33             : NSS_STATUS winbindd_request_response(struct winbindd_context *wbctx,
      34             :                                      int req_type,
      35             :                                      struct winbindd_request *request,
      36             :                                      struct winbindd_response *response);
      37             : NSS_STATUS winbindd_priv_request_response(struct winbindd_context *wbctx,
      38             :                                           int req_type,
      39             :                                           struct winbindd_request *request,
      40             :                                           struct winbindd_response *response);
      41             : struct winbindd_context *winbindd_ctx_create(void);
      42             : void winbindd_ctx_free(struct winbindd_context *ctx);
      43             : 
      44             : /* Global context used for non-Ctx functions */
      45             : 
      46             : static struct wbcContext wbcGlobalCtx = {
      47             :         .winbindd_ctx = NULL,
      48             :         .pw_cache_size = 0,
      49             :         .pw_cache_idx = 0,
      50             :         .gr_cache_size = 0,
      51             :         .gr_cache_idx = 0
      52             : };
      53             : 
      54             : /*
      55             :  result == NSS_STATUS_UNAVAIL: winbind not around
      56             :  result == NSS_STATUS_NOTFOUND: winbind around, but domain missing
      57             : 
      58             :  Due to a bad API NSS_STATUS_NOTFOUND is returned both when winbind_off
      59             :  and when winbind return WINBINDD_ERROR. So the semantics of this
      60             :  routine depends on winbind_on. Grepping for winbind_off I just
      61             :  found 3 places where winbind is turned off, and this does not conflict
      62             :  (as far as I have seen) with the callers of is_trusted_domains.
      63             : 
      64             :  --Volker
      65             : */
      66             : 
      67      853256 : static wbcErr wbcRequestResponseInt(
      68             :         struct winbindd_context *wbctx,
      69             :         int cmd,
      70             :         struct winbindd_request *request,
      71             :         struct winbindd_response *response,
      72             :         NSS_STATUS (*fn)(struct winbindd_context *wbctx, int req_type,
      73             :                          struct winbindd_request *request,
      74             :                          struct winbindd_response *response))
      75             : {
      76      853256 :         wbcErr wbc_status = WBC_ERR_UNKNOWN_FAILURE;
      77        4347 :         NSS_STATUS nss_status;
      78             : 
      79             :         /* for some calls the request and/or response can be NULL */
      80             : 
      81      857603 :         nss_status = fn(wbctx, cmd, request, response);
      82             : 
      83      853255 :         switch (nss_status) {
      84      389334 :         case NSS_STATUS_SUCCESS:
      85      389334 :                 wbc_status = WBC_ERR_SUCCESS;
      86      389334 :                 break;
      87      452640 :         case NSS_STATUS_UNAVAIL:
      88      452640 :                 wbc_status = WBC_ERR_WINBIND_NOT_AVAILABLE;
      89      452640 :                 break;
      90        6934 :         case NSS_STATUS_NOTFOUND:
      91        6934 :                 wbc_status = WBC_ERR_DOMAIN_NOT_FOUND;
      92        6934 :                 break;
      93           0 :         default:
      94           0 :                 wbc_status = WBC_ERR_NSS_ERROR;
      95           0 :                 break;
      96             :         }
      97             : 
      98      853255 :         return wbc_status;
      99             : }
     100             : 
     101             : /**
     102             :  * @brief Wrapper around Winbind's send/receive API call
     103             :  *
     104             :  * @param ctx       Context
     105             :  * @param cmd       Winbind command operation to perform
     106             :  * @param request   Send structure
     107             :  * @param response  Receive structure
     108             :  *
     109             :  * @return #wbcErr
     110             :  */
     111             : _PUBLIC_ /* this is internal to wbclient_internal.h, but part of the ABI */
     112      827409 : wbcErr wbcRequestResponse(struct wbcContext *ctx, int cmd,
     113             :                           struct winbindd_request *request,
     114             :                           struct winbindd_response *response)
     115             : {
     116      827409 :         struct winbindd_context *wbctx = NULL;
     117             : 
     118      827409 :         if (ctx) {
     119        1224 :                 wbctx = ctx->winbindd_ctx;
     120             :         }
     121             : 
     122      827409 :         return wbcRequestResponseInt(wbctx, cmd, request, response,
     123             :                                      winbindd_request_response);
     124             : }
     125             : 
     126             : _PUBLIC_ /* this is internal to wbclient_internal.h, but part of the ABI */
     127       25847 : wbcErr wbcRequestResponsePriv(struct wbcContext *ctx, int cmd,
     128             :                               struct winbindd_request *request,
     129             :                               struct winbindd_response *response)
     130             : {
     131       25847 :         struct winbindd_context *wbctx = NULL;
     132             : 
     133       25847 :         if (ctx) {
     134           0 :                 wbctx = ctx->winbindd_ctx;
     135             :         }
     136             : 
     137       25847 :         return wbcRequestResponseInt(wbctx, cmd, request, response,
     138             :                                      winbindd_priv_request_response);
     139             : }
     140             : 
     141             : /** @brief Translate an error value into a string
     142             :  *
     143             :  * @param error
     144             :  *
     145             :  * @return a pointer to a static string
     146             :  **/
     147             : _PUBLIC_
     148         415 : const char *wbcErrorString(wbcErr error)
     149             : {
     150         415 :         switch (error) {
     151         220 :         case WBC_ERR_SUCCESS:
     152         220 :                 return "WBC_ERR_SUCCESS";
     153           0 :         case WBC_ERR_NOT_IMPLEMENTED:
     154           0 :                 return "WBC_ERR_NOT_IMPLEMENTED";
     155           0 :         case WBC_ERR_UNKNOWN_FAILURE:
     156           0 :                 return "WBC_ERR_UNKNOWN_FAILURE";
     157           0 :         case WBC_ERR_NO_MEMORY:
     158           0 :                 return "WBC_ERR_NO_MEMORY";
     159           0 :         case WBC_ERR_INVALID_SID:
     160           0 :                 return "WBC_ERR_INVALID_SID";
     161           0 :         case WBC_ERR_INVALID_PARAM:
     162           0 :                 return "WBC_ERR_INVALID_PARAM";
     163          12 :         case WBC_ERR_WINBIND_NOT_AVAILABLE:
     164          12 :                 return "WBC_ERR_WINBIND_NOT_AVAILABLE";
     165         107 :         case WBC_ERR_DOMAIN_NOT_FOUND:
     166         107 :                 return "WBC_ERR_DOMAIN_NOT_FOUND";
     167           0 :         case WBC_ERR_INVALID_RESPONSE:
     168           0 :                 return "WBC_ERR_INVALID_RESPONSE";
     169           0 :         case WBC_ERR_NSS_ERROR:
     170           0 :                 return "WBC_ERR_NSS_ERROR";
     171           0 :         case WBC_ERR_UNKNOWN_USER:
     172           0 :                 return "WBC_ERR_UNKNOWN_USER";
     173           0 :         case WBC_ERR_UNKNOWN_GROUP:
     174           0 :                 return "WBC_ERR_UNKNOWN_GROUP";
     175          76 :         case WBC_ERR_AUTH_ERROR:
     176          76 :                 return "WBC_ERR_AUTH_ERROR";
     177           0 :         case WBC_ERR_PWD_CHANGE_FAILED:
     178           0 :                 return "WBC_ERR_PWD_CHANGE_FAILED";
     179             :         }
     180             : 
     181           0 :         return "unknown wbcErr value";
     182             : }
     183             : 
     184             : #define WBC_MAGIC (0x7a2b0e1e)
     185             : #define WBC_MAGIC_FREE (0x875634fe)
     186             : 
     187             : struct wbcMemPrefix {
     188             :         uint32_t magic;
     189             :         void (*destructor)(void *ptr);
     190             : };
     191             : 
     192      402313 : static size_t wbcPrefixLen(void)
     193             : {
     194      402329 :         size_t result = sizeof(struct wbcMemPrefix);
     195      402313 :         return (result + 15) & ~15;
     196             : }
     197             : 
     198      133479 : static struct wbcMemPrefix *wbcMemToPrefix(void *ptr)
     199             : {
     200      133479 :         return (struct wbcMemPrefix *)(((char *)ptr) - wbcPrefixLen());
     201             : }
     202             : 
     203             : _PUBLIC_ /* this is internal to wbclient_internal.h, but part of the ABI */
     204      134425 : void *wbcAllocateMemory(size_t nelem, size_t elsize,
     205             :                         void (*destructor)(void *ptr))
     206             : {
     207           6 :         struct wbcMemPrefix *result;
     208             : 
     209      134425 :         if (nelem >= (2<<24)/elsize) {
     210             :                 /* basic protection against integer wrap */
     211           0 :                 return NULL;
     212             :         }
     213             : 
     214      134425 :         result = (struct wbcMemPrefix *)calloc(
     215      134425 :                 1, nelem*elsize + wbcPrefixLen());
     216      134425 :         if (result == NULL) {
     217           0 :                 return NULL;
     218             :         }
     219      134425 :         result->magic = WBC_MAGIC;
     220      134425 :         result->destructor = destructor;
     221      134425 :         return ((char *)result) + wbcPrefixLen();
     222             : }
     223             : 
     224             : /* Free library allocated memory */
     225             : _PUBLIC_
     226      146036 : void wbcFreeMemory(void *p)
     227             : {
     228           8 :         struct wbcMemPrefix *wbcMem;
     229             : 
     230      146036 :         if (p == NULL) {
     231       12553 :                 return;
     232             :         }
     233      133479 :         wbcMem = wbcMemToPrefix(p);
     234      133479 :         if (wbcMem->magic != WBC_MAGIC) {
     235           0 :                 return;
     236             :         }
     237             : 
     238             :         /* paranoid check to ensure we don't double free */
     239      133479 :         wbcMem->magic = WBC_MAGIC_FREE;
     240             : 
     241      133479 :         if (wbcMem->destructor != NULL) {
     242      128935 :                 wbcMem->destructor(p);
     243             :         }
     244      133479 :         free(wbcMem);
     245      133479 :         return;
     246             : }
     247             : 
     248             : _PUBLIC_ /* this is internal to wbclient_internal.h, but part of the ABI */
     249        1918 : char *wbcStrDup(const char *str)
     250             : {
     251           0 :         char *result;
     252           0 :         size_t len;
     253             : 
     254        1918 :         len = strlen(str);
     255        1918 :         result = (char *)wbcAllocateMemory(len+1, sizeof(char), NULL);
     256        1918 :         if (result == NULL) {
     257           0 :                 return NULL;
     258             :         }
     259        1918 :         memcpy(result, str, len+1);
     260        1918 :         return result;
     261             : }
     262             : 
     263          72 : static void wbcStringArrayDestructor(void *ptr)
     264             : {
     265          72 :         char **p = (char **)ptr;
     266        6560 :         while (*p != NULL) {
     267        6488 :                 free(*p);
     268        6488 :                 p += 1;
     269             :         }
     270          72 : }
     271             : 
     272             : _PUBLIC_ /* this is internal to wbclient_internal.h, but part of the ABI */
     273          72 : const char **wbcAllocateStringArray(int num_strings)
     274             : {
     275         144 :         return (const char **)wbcAllocateMemory(
     276          72 :                 num_strings + 1, sizeof(const char *),
     277             :                 wbcStringArrayDestructor);
     278             : }
     279             : 
     280             : _PUBLIC_
     281           4 : wbcErr wbcLibraryDetails(struct wbcLibraryDetails **_details)
     282             : {
     283           0 :         struct wbcLibraryDetails *info;
     284             : 
     285           4 :         info = (struct wbcLibraryDetails *)wbcAllocateMemory(
     286             :                 1, sizeof(struct wbcLibraryDetails), NULL);
     287             : 
     288           4 :         if (info == NULL) {
     289           0 :                 return WBC_ERR_NO_MEMORY;
     290             :         }
     291             : 
     292           4 :         info->major_version = WBCLIENT_MAJOR_VERSION;
     293           4 :         info->minor_version = WBCLIENT_MINOR_VERSION;
     294           4 :         info->vendor_version = WBCLIENT_VENDOR_VERSION;
     295             : 
     296           4 :         *_details = info;
     297           4 :         return WBC_ERR_SUCCESS;
     298             : }
     299             : 
     300             : /* Context handling functions */
     301             : 
     302         266 : static void wbcContextDestructor(void *ptr)
     303             : {
     304         266 :         struct wbcContext *ctx = (struct wbcContext *)ptr;
     305             : 
     306         266 :         winbindd_ctx_free(ctx->winbindd_ctx);
     307         266 : }
     308             : 
     309             : _PUBLIC_
     310         266 : struct wbcContext *wbcCtxCreate(void)
     311             : {
     312           0 :         struct wbcContext *ctx;
     313           0 :         struct winbindd_context *wbctx;
     314             : 
     315         266 :         ctx = (struct wbcContext *)wbcAllocateMemory(
     316             :                 1, sizeof(struct wbcContext), wbcContextDestructor);
     317             : 
     318         266 :         if (!ctx) {
     319           0 :                 return NULL;
     320             :         }
     321             : 
     322         266 :         wbctx = winbindd_ctx_create();
     323             : 
     324         266 :         if (!wbctx) {
     325           0 :                 wbcFreeMemory(ctx);
     326           0 :                 return NULL;
     327             :         }
     328             : 
     329         266 :         ctx->winbindd_ctx = wbctx;
     330             : 
     331         266 :         return ctx;
     332             : }
     333             : 
     334             : _PUBLIC_
     335         266 : void wbcCtxFree(struct wbcContext *ctx)
     336             : {
     337         266 :         wbcFreeMemory(ctx);
     338         266 : }
     339             : 
     340             : _PUBLIC_ /* this is internal to wbclient_internal.h, but part of the ABI */
     341           0 : struct wbcContext *wbcGetGlobalCtx(void)
     342             : {
     343           0 :         return &wbcGlobalCtx;
     344             : }

Generated by: LCOV version 1.14