summaryrefslogtreecommitdiffstats
path: root/libc
diff options
context:
space:
mode:
authorElliott Hughes <enh@google.com>2015-01-21 19:42:40 +0000
committerGerrit Code Review <noreply-gerritcodereview@google.com>2015-01-21 19:42:44 +0000
commit2a8c929aaf8d34d2b6e89ed9c8b6da163316143e (patch)
tree992748f27c1ee609a1dba97d7a9e19b0df7689c0 /libc
parent8476b8ed9e290d79cd00e7801c505268f0e8ec15 (diff)
parent8c4994bbc1a9a01e34ea92c91eb5b2d1a27bd074 (diff)
downloadbionic-2a8c929aaf8d34d2b6e89ed9c8b6da163316143e.zip
bionic-2a8c929aaf8d34d2b6e89ed9c8b6da163316143e.tar.gz
bionic-2a8c929aaf8d34d2b6e89ed9c8b6da163316143e.tar.bz2
Merge "Implement __fsetlocking."
Diffstat (limited to 'libc')
-rw-r--r--libc/stdio/fileext.h9
-rw-r--r--libc/stdio/local.h4
-rw-r--r--libc/stdio/stdio_ext.cpp25
3 files changed, 23 insertions, 15 deletions
diff --git a/libc/stdio/fileext.h b/libc/stdio/fileext.h
index 25b7bda..75230cd 100644
--- a/libc/stdio/fileext.h
+++ b/libc/stdio/fileext.h
@@ -33,6 +33,7 @@
#define _FILEEXT_H_
#include <pthread.h>
+#include <stdbool.h>
__BEGIN_DECLS
@@ -40,9 +41,10 @@ __BEGIN_DECLS
* file extension
*/
struct __sfileext {
- struct __sbuf _ub; /* ungetc buffer */
+ struct __sbuf _ub; /* ungetc buffer */
struct wchar_io_data _wcio; /* wide char io status */
- pthread_mutex_t _lock; /* file lock */
+ pthread_mutex_t _lock; /* file lock */
+ bool _stdio_handles_locking; /* __fsetlocking support */
};
#define _EXT(fp) ((struct __sfileext *)((fp)->_ext._base))
@@ -54,7 +56,8 @@ do { \
_UB(fp)._base = NULL; \
_UB(fp)._size = 0; \
WCIO_INIT(fp); \
- _FLOCK(fp).value = __PTHREAD_RECURSIVE_MUTEX_INIT_VALUE; \
+ _FLOCK(fp).value = __PTHREAD_RECURSIVE_MUTEX_INIT_VALUE; \
+ _EXT(fp)->_stdio_handles_locking = true; \
} while (0)
#define _FILEEXT_SETUP(f, fext) \
diff --git a/libc/stdio/local.h b/libc/stdio/local.h
index 46b11f1..7033eda 100644
--- a/libc/stdio/local.h
+++ b/libc/stdio/local.h
@@ -111,8 +111,8 @@ extern void __atexit_register_cleanup(void (*)(void));
(fp)->_lb._base = NULL; \
}
-#define FLOCKFILE(fp) flockfile(fp)
-#define FUNLOCKFILE(fp) funlockfile(fp)
+#define FLOCKFILE(fp) if (_EXT(fp)->_stdio_handles_locking) flockfile(fp)
+#define FUNLOCKFILE(fp) if (_EXT(fp)->_stdio_handles_locking) funlockfile(fp)
#define FLOATING_POINT
#define PRINTF_WIDE_CHAR
diff --git a/libc/stdio/stdio_ext.cpp b/libc/stdio/stdio_ext.cpp
index bfdecb8..fea44f6 100644
--- a/libc/stdio/stdio_ext.cpp
+++ b/libc/stdio/stdio_ext.cpp
@@ -27,13 +27,10 @@
*/
#include <stdio_ext.h>
+#include <stdlib.h>
-#include <stdio.h>
#include "local.h"
-
-#define FSETLOCKING_QUERY 0
-#define FSETLOCKING_INTERNAL 1
-#define FSETLOCKING_BYCALLER 2
+#include "private/libc_logging.h"
size_t __fbufsize(FILE* fp) {
return fp->_bf._size;
@@ -76,11 +73,19 @@ void _flushlbf() {
fflush(NULL);
}
-int __fsetlocking(FILE*, int) {
- // We don't currently have an implementation that would obey this,
- // so make setting the state a no-op and always return "we handle locking for you".
- // http://b/17154740 suggests ways we could fix this.
- return FSETLOCKING_INTERNAL;
+int __fsetlocking(FILE* fp, int type) {
+ int old_state = _EXT(fp)->_stdio_handles_locking ? FSETLOCKING_INTERNAL : FSETLOCKING_BYCALLER;
+ if (type == FSETLOCKING_QUERY) {
+ return old_state;
+ }
+
+ if (type != FSETLOCKING_INTERNAL && type != FSETLOCKING_BYCALLER) {
+ // The API doesn't let us report an error, so blow up.
+ __libc_fatal("Bad type (%d) passed to __fsetlocking", type);
+ }
+
+ _EXT(fp)->_stdio_handles_locking = (type == FSETLOCKING_INTERNAL);
+ return old_state;
}
void clearerr_unlocked(FILE* fp) {