io_util.c 6.1 KB
Newer Older
D
duke 已提交
1
/*
X
xdono 已提交
2
 * Copyright 1994-2008 Sun Microsystems, Inc.  All Rights Reserved.
D
duke 已提交
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Sun designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Sun in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
 * CA 95054 USA or visit www.sun.com if you need additional information or
 * have any questions.
 */

#include <stdlib.h>
#include <string.h>
28
#include <stddef.h>
D
duke 已提交
29 30 31 32 33 34 35 36 37

#include "jni.h"
#include "jni_util.h"
#include "jvm.h"
#include "io_util.h"
#include "io_util_md.h"

/* IO helper functions */

38
jint
D
duke 已提交
39
readSingle(JNIEnv *env, jobject this, jfieldID fid) {
40
    jint nread;
D
duke 已提交
41 42 43
    char ret;
    FD fd = GET_FD(this, fid);
    if (fd == -1) {
44
        JNU_ThrowIOException(env, "Stream Closed");
D
duke 已提交
45 46 47 48 49 50 51 52
        return -1;
    }
    nread = IO_Read(fd, &ret, 1);
    if (nread == 0) { /* EOF */
        return -1;
    } else if (nread == JVM_IO_ERR) { /* error */
        JNU_ThrowIOExceptionWithLastError(env, "Read error");
    } else if (nread == JVM_IO_INTR) {
53
        JNU_ThrowByName(env, "java/io/InterruptedIOException", NULL);
D
duke 已提交
54 55 56 57 58 59 60 61
    }
    return ret & 0xFF;
}

/* The maximum size of a stack-allocated buffer.
 */
#define BUF_SIZE 8192

62 63 64 65 66 67 68 69 70 71 72 73
/*
 * Returns true if the array slice defined by the given offset and length
 * is out of bounds.
 */
static int
outOfBounds(JNIEnv *env, jint off, jint len, jbyteArray array) {
    return ((off < 0) ||
            (len < 0) ||
            // We are very careful to avoid signed integer overflow,
            // the result of which is undefined in C.
            ((*env)->GetArrayLength(env, array) - off < len));
}
D
duke 已提交
74

75
jint
D
duke 已提交
76 77 78
readBytes(JNIEnv *env, jobject this, jbyteArray bytes,
          jint off, jint len, jfieldID fid)
{
79
    jint nread;
D
duke 已提交
80
    char stackBuf[BUF_SIZE];
81
    char *buf = NULL;
D
duke 已提交
82 83 84
    FD fd;

    if (IS_NULL(bytes)) {
85
        JNU_ThrowNullPointerException(env, NULL);
D
duke 已提交
86 87 88
        return -1;
    }

89
    if (outOfBounds(env, off, len, bytes)) {
90
        JNU_ThrowByName(env, "java/lang/IndexOutOfBoundsException", NULL);
D
duke 已提交
91 92 93 94 95 96 97
        return -1;
    }

    if (len == 0) {
        return 0;
    } else if (len > BUF_SIZE) {
        buf = malloc(len);
98 99
        if (buf == NULL) {
            JNU_ThrowOutOfMemoryError(env, NULL);
D
duke 已提交
100 101 102 103 104 105 106 107
            return 0;
        }
    } else {
        buf = stackBuf;
    }

    fd = GET_FD(this, fid);
    if (fd == -1) {
108
        JNU_ThrowIOException(env, "Stream Closed");
D
duke 已提交
109
        nread = -1;
110 111 112 113 114 115
    } else {
        nread = IO_Read(fd, buf, len);
        if (nread > 0) {
            (*env)->SetByteArrayRegion(env, bytes, off, nread, (jbyte *)buf);
        } else if (nread == JVM_IO_ERR) {
            JNU_ThrowIOExceptionWithLastError(env, "Read error");
116 117
        } else if (nread == JVM_IO_INTR) {
            JNU_ThrowByName(env, "java/io/InterruptedIOException", NULL);
118 119 120
        } else { /* EOF */
            nread = -1;
        }
D
duke 已提交
121 122 123 124 125 126 127 128 129 130
    }

    if (buf != stackBuf) {
        free(buf);
    }
    return nread;
}

void
writeSingle(JNIEnv *env, jobject this, jint byte, jfieldID fid) {
131 132 133
    // Discard the 24 high-order bits of byte. See OutputStream#write(int)
    char c = (char) byte;
    jint n;
D
duke 已提交
134 135
    FD fd = GET_FD(this, fid);
    if (fd == -1) {
136
        JNU_ThrowIOException(env, "Stream Closed");
D
duke 已提交
137 138 139 140 141 142
        return;
    }
    n = IO_Write(fd, &c, 1);
    if (n == JVM_IO_ERR) {
        JNU_ThrowIOExceptionWithLastError(env, "Write error");
    } else if (n == JVM_IO_INTR) {
143
        JNU_ThrowByName(env, "java/io/InterruptedIOException", NULL);
D
duke 已提交
144 145 146 147 148
    }
}

void
writeBytes(JNIEnv *env, jobject this, jbyteArray bytes,
149
           jint off, jint len, jfieldID fid)
D
duke 已提交
150
{
151
    jint n;
D
duke 已提交
152
    char stackBuf[BUF_SIZE];
153
    char *buf = NULL;
D
duke 已提交
154 155 156
    FD fd;

    if (IS_NULL(bytes)) {
157
        JNU_ThrowNullPointerException(env, NULL);
D
duke 已提交
158 159 160
        return;
    }

161
    if (outOfBounds(env, off, len, bytes)) {
162
        JNU_ThrowByName(env, "java/lang/IndexOutOfBoundsException", NULL);
D
duke 已提交
163 164 165 166 167 168 169
        return;
    }

    if (len == 0) {
        return;
    } else if (len > BUF_SIZE) {
        buf = malloc(len);
170 171
        if (buf == NULL) {
            JNU_ThrowOutOfMemoryError(env, NULL);
D
duke 已提交
172 173 174 175 176 177 178 179 180 181 182 183 184
            return;
        }
    } else {
        buf = stackBuf;
    }

    (*env)->GetByteArrayRegion(env, bytes, off, len, (jbyte *)buf);

    if (!(*env)->ExceptionOccurred(env)) {
        off = 0;
        while (len > 0) {
            fd = GET_FD(this, fid);
            if (fd == -1) {
185 186
                JNU_ThrowIOException(env, "Stream Closed");
                break;
D
duke 已提交
187 188 189 190 191 192
            }
            n = IO_Write(fd, buf+off, len);
            if (n == JVM_IO_ERR) {
                JNU_ThrowIOExceptionWithLastError(env, "Write error");
                break;
            } else if (n == JVM_IO_INTR) {
193
                JNU_ThrowByName(env, "java/io/InterruptedIOException", NULL);
D
duke 已提交
194 195 196 197 198 199 200 201 202 203 204 205 206 207 208
                break;
            }
            off += n;
            len -= n;
        }
    }
    if (buf != stackBuf) {
        free(buf);
    }
}

void
throwFileNotFoundException(JNIEnv *env, jstring path)
{
    char buf[256];
209
    jint n;
D
duke 已提交
210 211 212 213 214
    jobject x;
    jstring why = NULL;

    n = JVM_GetLastErrorString(buf, sizeof(buf));
    if (n > 0) {
215
        why = JNU_NewStringPlatform(env, buf);
D
duke 已提交
216 217
    }
    x = JNU_NewObjectByName(env,
218 219 220
                            "java/io/FileNotFoundException",
                            "(Ljava/lang/String;Ljava/lang/String;)V",
                            path, why);
D
duke 已提交
221
    if (x != NULL) {
222
        (*env)->Throw(env, x);
D
duke 已提交
223 224
    }
}