1*71db0c75SAndroid Build Coastguard Worker //===-- Implementation of fgets -------------------------------------------===// 2*71db0c75SAndroid Build Coastguard Worker // 3*71db0c75SAndroid Build Coastguard Worker // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4*71db0c75SAndroid Build Coastguard Worker // See https://llvm.org/LICENSE.txt for license information. 5*71db0c75SAndroid Build Coastguard Worker // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6*71db0c75SAndroid Build Coastguard Worker // 7*71db0c75SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===// 8*71db0c75SAndroid Build Coastguard Worker 9*71db0c75SAndroid Build Coastguard Worker #include "src/stdio/fgets.h" 10*71db0c75SAndroid Build Coastguard Worker #include "src/__support/File/file.h" 11*71db0c75SAndroid Build Coastguard Worker 12*71db0c75SAndroid Build Coastguard Worker #include "hdr/types/FILE.h" 13*71db0c75SAndroid Build Coastguard Worker #include "src/__support/macros/config.h" 14*71db0c75SAndroid Build Coastguard Worker #include "src/errno/libc_errno.h" 15*71db0c75SAndroid Build Coastguard Worker #include <stddef.h> 16*71db0c75SAndroid Build Coastguard Worker 17*71db0c75SAndroid Build Coastguard Worker namespace LIBC_NAMESPACE_DECL { 18*71db0c75SAndroid Build Coastguard Worker 19*71db0c75SAndroid Build Coastguard Worker LLVM_LIBC_FUNCTION(char *, fgets, 20*71db0c75SAndroid Build Coastguard Worker (char *__restrict str, int count, 21*71db0c75SAndroid Build Coastguard Worker ::FILE *__restrict raw_stream)) { 22*71db0c75SAndroid Build Coastguard Worker if (count < 1) 23*71db0c75SAndroid Build Coastguard Worker return nullptr; 24*71db0c75SAndroid Build Coastguard Worker 25*71db0c75SAndroid Build Coastguard Worker unsigned char c = '\0'; 26*71db0c75SAndroid Build Coastguard Worker auto stream = reinterpret_cast<LIBC_NAMESPACE::File *__restrict>(raw_stream); 27*71db0c75SAndroid Build Coastguard Worker stream->lock(); 28*71db0c75SAndroid Build Coastguard Worker 29*71db0c75SAndroid Build Coastguard Worker // i is an int because it's frequently compared to count, which is also int. 30*71db0c75SAndroid Build Coastguard Worker int i = 0; 31*71db0c75SAndroid Build Coastguard Worker 32*71db0c75SAndroid Build Coastguard Worker for (; i < (count - 1) && c != '\n'; ++i) { 33*71db0c75SAndroid Build Coastguard Worker auto result = stream->read_unlocked(&c, 1); 34*71db0c75SAndroid Build Coastguard Worker size_t r = result.value; 35*71db0c75SAndroid Build Coastguard Worker if (result.has_error()) 36*71db0c75SAndroid Build Coastguard Worker libc_errno = result.error; 37*71db0c75SAndroid Build Coastguard Worker 38*71db0c75SAndroid Build Coastguard Worker if (r != 1) 39*71db0c75SAndroid Build Coastguard Worker break; 40*71db0c75SAndroid Build Coastguard Worker str[i] = c; 41*71db0c75SAndroid Build Coastguard Worker } 42*71db0c75SAndroid Build Coastguard Worker 43*71db0c75SAndroid Build Coastguard Worker bool has_error = stream->error_unlocked(); 44*71db0c75SAndroid Build Coastguard Worker bool has_eof = stream->iseof_unlocked(); 45*71db0c75SAndroid Build Coastguard Worker stream->unlock(); 46*71db0c75SAndroid Build Coastguard Worker 47*71db0c75SAndroid Build Coastguard Worker // If the requested read size makes no sense, an error occured, or no bytes 48*71db0c75SAndroid Build Coastguard Worker // were read due to an EOF, then return nullptr and don't write the null byte. 49*71db0c75SAndroid Build Coastguard Worker if (has_error || (i == 0 && has_eof)) 50*71db0c75SAndroid Build Coastguard Worker return nullptr; 51*71db0c75SAndroid Build Coastguard Worker 52*71db0c75SAndroid Build Coastguard Worker str[i] = '\0'; 53*71db0c75SAndroid Build Coastguard Worker return str; 54*71db0c75SAndroid Build Coastguard Worker } 55*71db0c75SAndroid Build Coastguard Worker 56*71db0c75SAndroid Build Coastguard Worker } // namespace LIBC_NAMESPACE_DECL 57