__fdopen.c 2.8 KB
Newer Older
Y
yaofeng wang 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14
/* Copyright (c) 2021-2022 Huawei Device Co., Ltd.
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *	 http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

15 16 17 18 19 20 21 22 23
#include "stdio_impl.h"
#include <stdlib.h>
#include <sys/ioctl.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#include <string.h>
#include "libc.h"

24
static size_t __get_bufsize(int fd)
25
{
Y
yaofeng wang 已提交
26 27 28 29 30 31 32 33 34 35 36 37
	struct stat st;
	size_t buf_size = 0;

	if (fstat(fd, &st) < 0) {
		buf_size = BUFSIZ;
	} else if (st.st_blksize == 0) {
		buf_size = BUFSIZ;
	} else {
		buf_size = st.st_blksize;
	}

	return buf_size;
38 39
}

40
int __falloc_buf(FILE *f)
41
{
42 43
	/* return if already allocated, or F_NOBUF set */
	if (f->buf != NULL || f->buf_size != 0 || f->flags & F_NOBUF) {
Y
yaofeng wang 已提交
44 45 46
		return 0;
	}

47 48 49
	/* Default,  base and buf are NULL,and buf_size = 0 */
	size_t buf_size = 0;

Y
yaofeng wang 已提交
50
	/* get buffer size via file stat */
51
	buf_size = __get_bufsize(f->fd);
Y
yaofeng wang 已提交
52

53 54 55 56 57
	/* alloc R/W buffer */
	f->base = (unsigned char *)malloc(UNGET + buf_size * sizeof(unsigned char));
	if (!f->base) {
		errno = -ENOMEM;
		return errno;
Y
yaofeng wang 已提交
58 59
	}

60 61 62 63 64 65
	/* reserve UNGET buffer */
	f->buf = f->base + UNGET;
	f->buf_size = buf_size;

	return 0;
}
Y
yaofeng wang 已提交
66

67 68 69 70 71 72 73 74 75 76
FILE *__fdopen(int fd, const char *mode)
{
	FILE *f = NULL;
	int file_flags = 0;
	int mode_flags = 0;

	/* Compute the flags to pass to open() */
	mode_flags = __fmodeflags(mode, &file_flags);
	if (mode_flags < 0) {
		return NULL;
Y
yaofeng wang 已提交
77 78
	}

79
	if (mode_flags & O_CLOEXEC) {
Y
yaofeng wang 已提交
80 81 82
		__syscall(SYS_fcntl, fd, F_SETFD, FD_CLOEXEC);
	}

83
	if (mode_flags & O_APPEND) {
Y
yaofeng wang 已提交
84 85 86 87 88
		int flags = __syscall(SYS_fcntl, fd, F_GETFL);
		if (!(flags & O_APPEND))
			__syscall(SYS_fcntl, fd, F_SETFL, flags | O_APPEND);
	}

89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111
	f = __fdopenx(fd, file_flags);
	if (f) {
		return f;
	}

	return NULL;
}
weak_alias(__fdopen, fdopen);

FILE *__fdopenx(int fd, int flags)
{
	FILE *f = 0;
	struct winsize wsz;

	/* Allocate FILE or fail */
	if (!(f = __ofl_alloc())) {
		return NULL;
	}

	/* Zero-fill only the struct, not the buffer */
	memset(f, 0, sizeof *f);

	f->flags = flags;
Y
yaofeng wang 已提交
112 113 114 115 116 117 118 119 120 121 122 123 124
	f->fd = fd;

	/* Activate line buffered mode for terminals */
	f->lbf = EOF;
	if (!(f->flags & F_NOWR) && !__syscall(SYS_ioctl, fd, TIOCGWINSZ, &wsz)) {
		f->lbf = '\n';
	}

	/* Initialize op ptrs. No problem if some are unneeded. */
	f->read = __stdio_read;
	f->write = __stdio_write;
	f->seek = __stdio_seek;
	f->close = __stdio_close;
R
robottoy 已提交
125
	f->readx = __stdio_readx;
Y
yaofeng wang 已提交
126 127 128 129 130 131 132

	if (!libc.threaded) {
		f->lock = -1;
	}

	/* Add new FILE to open file list */
	return __ofl_add(f);
133 134
}