os/ossrv/ssl/libcrypto/src/crypto/lpdir_vms.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/* $LP: LPlib/source/LPdir_vms.c,v 1.20 2004/08/26 13:36:05 _cvs_levitte Exp $ */
sl@0
     2
/*
sl@0
     3
 * Copyright (c) 2004, Richard Levitte <richard@levitte.org>
sl@0
     4
 * All rights reserved.
sl@0
     5
 *
sl@0
     6
 * Redistribution and use in source and binary forms, with or without
sl@0
     7
 * modification, are permitted provided that the following conditions
sl@0
     8
 * are met:
sl@0
     9
 * 1. Redistributions of source code must retain the above copyright
sl@0
    10
 *    notice, this list of conditions and the following disclaimer.
sl@0
    11
 * 2. Redistributions in binary form must reproduce the above copyright
sl@0
    12
 *    notice, this list of conditions and the following disclaimer in the
sl@0
    13
 *    documentation and/or other materials provided with the distribution.
sl@0
    14
 * 
sl@0
    15
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
sl@0
    16
 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
sl@0
    17
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
sl@0
    18
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
sl@0
    19
 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
sl@0
    20
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
sl@0
    21
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
sl@0
    22
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
sl@0
    23
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
sl@0
    24
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
sl@0
    25
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
sl@0
    26
 */
sl@0
    27
sl@0
    28
#include <stddef.h>
sl@0
    29
#include <stdlib.h>
sl@0
    30
#include <string.h>
sl@0
    31
#include <errno.h>
sl@0
    32
#include <descrip.h>
sl@0
    33
#include <namdef.h>
sl@0
    34
#include <rmsdef.h>
sl@0
    35
#include <libfildef.h>
sl@0
    36
#include <lib$routines.h>
sl@0
    37
#include <strdef.h>
sl@0
    38
#include <str$routines.h>
sl@0
    39
#include <stsdef.h>
sl@0
    40
#ifndef LPDIR_H
sl@0
    41
#include "LPdir.h"
sl@0
    42
#endif
sl@0
    43
sl@0
    44
/* Because some compiler options hide this macor */
sl@0
    45
#ifndef EVMSERR
sl@0
    46
#define EVMSERR		65535  /* error for non-translatable VMS errors */
sl@0
    47
#endif
sl@0
    48
sl@0
    49
struct LP_dir_context_st
sl@0
    50
{
sl@0
    51
  unsigned long VMS_context;
sl@0
    52
#ifdef NAML$C_MAXRSS
sl@0
    53
  char filespec[NAML$C_MAXRSS+1];
sl@0
    54
  char result[NAML$C_MAXRSS+1];
sl@0
    55
#else
sl@0
    56
  char filespec[256];
sl@0
    57
  char result[256];
sl@0
    58
#endif
sl@0
    59
  struct dsc$descriptor_d filespec_dsc;
sl@0
    60
  struct dsc$descriptor_d result_dsc;
sl@0
    61
};
sl@0
    62
sl@0
    63
const char *LP_find_file(LP_DIR_CTX **ctx, const char *directory)
sl@0
    64
{
sl@0
    65
  int status;
sl@0
    66
  char *p, *r;
sl@0
    67
  size_t l;
sl@0
    68
  unsigned long flags = 0;
sl@0
    69
#ifdef NAML$C_MAXRSS
sl@0
    70
  flags |= LIB$M_FIL_LONG_NAMES;
sl@0
    71
#endif
sl@0
    72
sl@0
    73
  if (ctx == NULL || directory == NULL)
sl@0
    74
    {
sl@0
    75
      errno = EINVAL;
sl@0
    76
      return 0;
sl@0
    77
    }
sl@0
    78
sl@0
    79
  errno = 0;
sl@0
    80
  if (*ctx == NULL)
sl@0
    81
    {
sl@0
    82
      size_t filespeclen = strlen(directory);
sl@0
    83
      char *filespec = NULL;
sl@0
    84
sl@0
    85
      /* MUST be a VMS directory specification!  Let's estimate if it is. */
sl@0
    86
      if (directory[filespeclen-1] != ']'
sl@0
    87
	  && directory[filespeclen-1] != '>'
sl@0
    88
	  && directory[filespeclen-1] != ':')
sl@0
    89
	{
sl@0
    90
	  errno = EINVAL;
sl@0
    91
	  return 0;
sl@0
    92
	}
sl@0
    93
sl@0
    94
      filespeclen += 4;		/* "*.*;" */
sl@0
    95
sl@0
    96
      if (filespeclen >
sl@0
    97
#ifdef NAML$C_MAXRSS
sl@0
    98
	  NAML$C_MAXRSS
sl@0
    99
#else
sl@0
   100
	  255
sl@0
   101
#endif
sl@0
   102
	  )
sl@0
   103
	{
sl@0
   104
	  errno = ENAMETOOLONG;
sl@0
   105
	  return 0;
sl@0
   106
	}
sl@0
   107
sl@0
   108
      *ctx = (LP_DIR_CTX *)malloc(sizeof(LP_DIR_CTX));
sl@0
   109
      if (*ctx == NULL)
sl@0
   110
	{
sl@0
   111
	  errno = ENOMEM;
sl@0
   112
	  return 0;
sl@0
   113
	}
sl@0
   114
      memset(*ctx, '\0', sizeof(LP_DIR_CTX));
sl@0
   115
sl@0
   116
      strcpy((*ctx)->filespec,directory);
sl@0
   117
      strcat((*ctx)->filespec,"*.*;");
sl@0
   118
      (*ctx)->filespec_dsc.dsc$w_length = filespeclen;
sl@0
   119
      (*ctx)->filespec_dsc.dsc$b_dtype = DSC$K_DTYPE_T;
sl@0
   120
      (*ctx)->filespec_dsc.dsc$b_class = DSC$K_CLASS_S;
sl@0
   121
      (*ctx)->filespec_dsc.dsc$a_pointer = (*ctx)->filespec;
sl@0
   122
      (*ctx)->result_dsc.dsc$w_length = 0;
sl@0
   123
      (*ctx)->result_dsc.dsc$b_dtype = DSC$K_DTYPE_T;
sl@0
   124
      (*ctx)->result_dsc.dsc$b_class = DSC$K_CLASS_D;
sl@0
   125
      (*ctx)->result_dsc.dsc$a_pointer = 0;
sl@0
   126
    }
sl@0
   127
sl@0
   128
  (*ctx)->result_dsc.dsc$w_length = 0;
sl@0
   129
  (*ctx)->result_dsc.dsc$b_dtype = DSC$K_DTYPE_T;
sl@0
   130
  (*ctx)->result_dsc.dsc$b_class = DSC$K_CLASS_D;
sl@0
   131
  (*ctx)->result_dsc.dsc$a_pointer = 0;
sl@0
   132
sl@0
   133
  status = lib$find_file(&(*ctx)->filespec_dsc, &(*ctx)->result_dsc,
sl@0
   134
			 &(*ctx)->VMS_context, 0, 0, 0, &flags);
sl@0
   135
sl@0
   136
  if (status == RMS$_NMF)
sl@0
   137
    {
sl@0
   138
      errno = 0;
sl@0
   139
      vaxc$errno = status;
sl@0
   140
      return NULL;
sl@0
   141
    }
sl@0
   142
sl@0
   143
  if(!$VMS_STATUS_SUCCESS(status))
sl@0
   144
    {
sl@0
   145
      errno = EVMSERR;
sl@0
   146
      vaxc$errno = status;
sl@0
   147
      return NULL;
sl@0
   148
    }
sl@0
   149
sl@0
   150
  /* Quick, cheap and dirty way to discard any device and directory,
sl@0
   151
     since we only want file names */
sl@0
   152
  l = (*ctx)->result_dsc.dsc$w_length;
sl@0
   153
  p = (*ctx)->result_dsc.dsc$a_pointer;
sl@0
   154
  r = p;
sl@0
   155
  for (; *p; p++)
sl@0
   156
    {
sl@0
   157
      if (*p == '^' && p[1] != '\0') /* Take care of ODS-5 escapes */
sl@0
   158
	{
sl@0
   159
	  p++;
sl@0
   160
	}
sl@0
   161
      else if (*p == ':' || *p == '>' || *p == ']')
sl@0
   162
	{
sl@0
   163
	  l -= p + 1 - r;
sl@0
   164
	  r = p + 1;
sl@0
   165
	}
sl@0
   166
      else if (*p == ';')
sl@0
   167
	{
sl@0
   168
	  l = p - r;
sl@0
   169
	  break;
sl@0
   170
	}
sl@0
   171
    }
sl@0
   172
sl@0
   173
  strncpy((*ctx)->result, r, l);
sl@0
   174
  (*ctx)->result[l] = '\0';
sl@0
   175
  str$free1_dx(&(*ctx)->result_dsc);
sl@0
   176
sl@0
   177
  return (*ctx)->result;
sl@0
   178
}
sl@0
   179
sl@0
   180
int LP_find_file_end(LP_DIR_CTX **ctx)
sl@0
   181
{
sl@0
   182
  if (ctx != NULL && *ctx != NULL)
sl@0
   183
    {
sl@0
   184
      int status = lib$find_file_end(&(*ctx)->VMS_context);
sl@0
   185
sl@0
   186
      free(*ctx);
sl@0
   187
sl@0
   188
      if(!$VMS_STATUS_SUCCESS(status))
sl@0
   189
	{
sl@0
   190
	  errno = EVMSERR;
sl@0
   191
	  vaxc$errno = status;
sl@0
   192
	  return 0;
sl@0
   193
	}
sl@0
   194
      return 1;
sl@0
   195
    }
sl@0
   196
  errno = EINVAL;
sl@0
   197
  return 0;
sl@0
   198
}
sl@0
   199