compat.c 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. /*
  2. * Dropbear - a SSH2 server
  3. *
  4. * Copyright (c) 2002,2003 Matt Johnston
  5. * All rights reserved.
  6. *
  7. * Permission is hereby granted, free of charge, to any person obtaining a copy
  8. * of this software and associated documentation files (the "Software"), to deal
  9. * in the Software without restriction, including without limitation the rights
  10. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  11. * copies of the Software, and to permit persons to whom the Software is
  12. * furnished to do so, subject to the following conditions:
  13. *
  14. * The above copyright notice and this permission notice shall be included in
  15. * all copies or substantial portions of the Software.
  16. *
  17. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  18. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  19. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  20. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  21. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  22. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  23. * SOFTWARE.
  24. *
  25. * strlcat() is copyright as follows:
  26. * Copyright (c) 1998 Todd C. Miller <Todd.Miller@courtesan.com>
  27. * All rights reserved.
  28. *
  29. * Redistribution and use in source and binary forms, with or without
  30. * modification, are permitted provided that the following conditions
  31. * are met:
  32. * 1. Redistributions of source code must retain the above copyright
  33. * notice, this list of conditions and the following disclaimer.
  34. * 2. Redistributions in binary form must reproduce the above copyright
  35. * notice, this list of conditions and the following disclaimer in the
  36. * documentation and/or other materials provided with the distribution.
  37. * 3. The name of the author may not be used to endorse or promote products
  38. * derived from this software without specific prior written permission.
  39. *
  40. * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
  41. * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
  42. * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
  43. * THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
  44. * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
  45. * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
  46. * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
  47. * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
  48. * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
  49. * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  50. *
  51. * daemon() and getusershell() is copyright as follows:
  52. *
  53. * Copyright (c) 1990, 1993
  54. * The Regents of the University of California. All rights reserved.
  55. *
  56. * Redistribution and use in source and binary forms, with or without
  57. * modification, are permitted provided that the following conditions
  58. * are met:
  59. * 1. Redistributions of source code must retain the above copyright
  60. * notice, this list of conditions and the following disclaimer.
  61. * 2. Redistributions in binary form must reproduce the above copyright
  62. * notice, this list of conditions and the following disclaimer in the
  63. * documentation and/or other materials provided with the distribution.
  64. * 3. Neither the name of the University nor the names of its contributors
  65. * may be used to endorse or promote products derived from this software
  66. * without specific prior written permission.
  67. *
  68. * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
  69. * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  70. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  71. * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
  72. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  73. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  74. * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  75. * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  76. * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  77. * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  78. * SUCH DAMAGE.
  79. *
  80. * Modifications for Dropbear to getusershell() are by Paul Marinceu
  81. */
  82. #include "includes.h"
  83. #ifndef HAVE_GETUSERSHELL
  84. static char **curshell, **shells, *strings;
  85. static char **initshells();
  86. #endif
  87. #ifndef HAVE_STRLCPY
  88. /* Implemented by matt as specified in freebsd 4.7 manpage.
  89. * We don't require great speed, is simply for use with sshpty code */
  90. size_t strlcpy(char *dst, const char *src, size_t size) {
  91. size_t i;
  92. /* this is undefined, though size==0 -> return 0 */
  93. if (size < 1) {
  94. return 0;
  95. }
  96. for (i = 0; i < size-1; i++) {
  97. if (src[i] == '\0') {
  98. break;
  99. } else {
  100. dst[i] = src[i];
  101. }
  102. }
  103. dst[i] = '\0';
  104. return strlen(src);
  105. }
  106. #endif /* HAVE_STRLCPY */
  107. #ifndef HAVE_STRLCAT
  108. /* taken from openbsd-compat for OpenSSH 7.2p2 */
  109. /* "$OpenBSD: strlcat.c,v 1.13 2005/08/08 08:05:37 espie Exp $"
  110. *
  111. * Appends src to string dst of size siz (unlike strncat, siz is the
  112. * full size of dst, not space left). At most siz-1 characters
  113. * will be copied. Always NUL terminates (unless siz <= strlen(dst)).
  114. * Returns strlen(src) + MIN(siz, strlen(initial dst)).
  115. * If retval >= siz, truncation occurred.
  116. */
  117. size_t
  118. strlcat(char *dst, const char *src, size_t siz)
  119. {
  120. char *d = dst;
  121. const char *s = src;
  122. size_t n = siz;
  123. size_t dlen;
  124. /* Find the end of dst and adjust bytes left but don't go past end */
  125. while (n-- != 0 && *d != '\0')
  126. d++;
  127. dlen = d - dst;
  128. n = siz - dlen;
  129. if (n == 0)
  130. return(dlen + strlen(s));
  131. while (*s != '\0') {
  132. if (n != 1) {
  133. *d++ = *s;
  134. n--;
  135. }
  136. s++;
  137. }
  138. *d = '\0';
  139. return(dlen + (s - src)); /* count does not include NUL */
  140. }
  141. #endif /* HAVE_STRLCAT */
  142. #ifndef HAVE_DAEMON
  143. /* From NetBSD - daemonise a process */
  144. int daemon(int nochdir, int noclose) {
  145. int fd;
  146. switch (fork()) {
  147. case -1:
  148. return (-1);
  149. case 0:
  150. break;
  151. default:
  152. _exit(0);
  153. }
  154. if (setsid() == -1)
  155. return -1;
  156. if (!nochdir)
  157. (void)chdir("/");
  158. if (!noclose && (fd = open(DROPBEAR_PATH_DEVNULL, O_RDWR, 0)) != -1) {
  159. (void)dup2(fd, STDIN_FILENO);
  160. (void)dup2(fd, STDOUT_FILENO);
  161. (void)dup2(fd, STDERR_FILENO);
  162. if (fd > STDERR_FILENO)
  163. (void)close(fd);
  164. }
  165. return 0;
  166. }
  167. #endif /* HAVE_DAEMON */
  168. #ifndef HAVE_BASENAME
  169. char *basename(const char *path) {
  170. char *foo = strrchr(path, '/');
  171. if (!foo)
  172. {
  173. return path;
  174. }
  175. return ++foo;
  176. }
  177. #endif /* HAVE_BASENAME */
  178. #ifndef HAVE_GETUSERSHELL
  179. /*
  180. * Get a list of shells from /etc/shells, if it exists.
  181. */
  182. char * getusershell() {
  183. char *ret;
  184. if (curshell == NULL)
  185. curshell = initshells();
  186. ret = *curshell;
  187. if (ret != NULL)
  188. curshell++;
  189. return (ret);
  190. }
  191. void endusershell() {
  192. if (shells != NULL)
  193. free(shells);
  194. shells = NULL;
  195. if (strings != NULL)
  196. free(strings);
  197. strings = NULL;
  198. curshell = NULL;
  199. }
  200. void setusershell() {
  201. curshell = initshells();
  202. }
  203. static char **initshells() {
  204. /* don't touch this list. */
  205. static const char *okshells[] = { "/bin/sh", "/bin/csh", NULL };
  206. register char **sp, *cp;
  207. register FILE *fp;
  208. struct stat statb;
  209. int flen;
  210. if (shells != NULL)
  211. free(shells);
  212. shells = NULL;
  213. if (strings != NULL)
  214. free(strings);
  215. strings = NULL;
  216. if ((fp = fopen("/etc/shells", "rc")) == NULL)
  217. return (char **) okshells;
  218. if (fstat(fileno(fp), &statb) == -1) {
  219. (void)fclose(fp);
  220. return (char **) okshells;
  221. }
  222. if ((strings = malloc((u_int)statb.st_size + 1)) == NULL) {
  223. (void)fclose(fp);
  224. return (char **) okshells;
  225. }
  226. shells = calloc((unsigned)statb.st_size / 3, sizeof (char *));
  227. if (shells == NULL) {
  228. (void)fclose(fp);
  229. free(strings);
  230. strings = NULL;
  231. return (char **) okshells;
  232. }
  233. sp = shells;
  234. cp = strings;
  235. flen = statb.st_size;
  236. while (fgets(cp, flen - (cp - strings), fp) != NULL) {
  237. while (*cp != '#' && *cp != '/' && *cp != '\0')
  238. cp++;
  239. if (*cp == '#' || *cp == '\0')
  240. continue;
  241. *sp++ = cp;
  242. while (!isspace(*cp) && *cp != '#' && *cp != '\0')
  243. cp++;
  244. *cp++ = '\0';
  245. }
  246. *sp = NULL;
  247. (void)fclose(fp);
  248. return (shells);
  249. }
  250. #endif /* HAVE_GETUSERSHELL */