Commit 3d5aea83 by Tom Tromey Committed by Tom Tromey

natIconv.cc (done): New methods.

2001-07-30  Tom Tromey  <tromey@redhat.com>
	    Corey Minyard  <minyard@acm.org>

	* gnu/gcj/convert/natIconv.cc (done): New methods.
	* gnu/gcj/convert/Output_iconv.java (done): New method.
	* gnu/gcj/convert/Input_iconv.java (done): New method.
	* gnu/gcj/convert/UnicodeToBytes.java (defaultEncodingClass):
	Removed.
	(getDefaultEncodingClass): Removed.
	(getDefaultEncoder): Use getEncoder.
	(done): New method.
	(defaultEncoding, CACHE_SIZE, encoderCache, currCachePos): New
	static fields.
	* gnu/gcj/convert/BytesToUnicode.java (defaultDecodingClass):
	Removed.
	(defaultEncoding, CACHE_SIZE, decoderCache, currCachePos): New
	static fields.
	(getDefaultDecodingClass): Removed.
	(getDefaultDecoder): Use getDecoder.
	(getDecoder): Look up decoder in cache.
	(done): New method.
	* java/lang/natString.cc (init): Call `done' on converter.
	(getBytes): Likewise.

Co-Authored-By: Corey Minyard <minyard@acm.org>

From-SVN: r44484
parent a08b2604
2001-07-30 Tom Tromey <tromey@redhat.com> 2001-07-30 Tom Tromey <tromey@redhat.com>
Corey Minyard <minyard@acm.org>
* gnu/gcj/convert/natIconv.cc (done): New methods.
* gnu/gcj/convert/Output_iconv.java (done): New method.
* gnu/gcj/convert/Input_iconv.java (done): New method.
* gnu/gcj/convert/UnicodeToBytes.java (defaultEncodingClass):
Removed.
(getDefaultEncodingClass): Removed.
(getDefaultEncoder): Use getEncoder.
(done): New method.
(defaultEncoding, CACHE_SIZE, encoderCache, currCachePos): New
static fields.
* gnu/gcj/convert/BytesToUnicode.java (defaultDecodingClass):
Removed.
(defaultEncoding, CACHE_SIZE, decoderCache, currCachePos): New
static fields.
(getDefaultDecodingClass): Removed.
(getDefaultDecoder): Use getDecoder.
(getDecoder): Look up decoder in cache.
(done): New method.
* java/lang/natString.cc (init): Call `done' on converter.
(getBytes): Likewise.
2001-07-30 Tom Tromey <tromey@redhat.com>
* java/lang/Integer.java: Merged with Classpath. * java/lang/Integer.java: Merged with Classpath.
......
/* Copyright (C) 1999, 2000 Free Software Foundation /* Copyright (C) 1999, 2000, 2001 Free Software Foundation
This file is part of libgcj. This file is part of libgcj.
...@@ -18,55 +18,79 @@ public abstract class BytesToUnicode extends IOConverter ...@@ -18,55 +18,79 @@ public abstract class BytesToUnicode extends IOConverter
/** End of valid bytes in buffer. */ /** End of valid bytes in buffer. */
public int inlength; public int inlength;
static Class defaultDecodingClass; // The name of the default encoding.
static String defaultEncoding;
static synchronized void getDefaultDecodingClass() /* These keep a small cache of decoders for reuse. The array holds
the actual decoders. The currCachePos is the next value we are
going to replace in the cache. We don't just throw the data away
if the cache is full, because if the cache filled up with stuff
we don't need then the cache would be worthless. We instead
circulate through the cache the implement kind of an LRU
algorithm. */
private static final int CACHE_SIZE = 4; // A power of 2 for speed
private static BytesToUnicode[] decoderCache
= new BytesToUnicode[CACHE_SIZE];
private static int currCachePos = 0;
public abstract String getName();
public static BytesToUnicode getDefaultDecoder()
{ {
// Test (defaultDecodingClass == null) again in case of race condition. try
if (defaultDecodingClass == null)
{ {
String encoding = canonicalize (System.getProperty("file.encoding")); synchronized (BytesToUnicode.class)
String className = "gnu.gcj.convert.Input_"+encoding; {
if (defaultEncoding == null)
{
String encoding
= canonicalize (System.getProperty("file.encoding",
"8859_1"));
String className = "gnu.gcj.convert.Input_" + encoding;
try try
{ {
defaultDecodingClass = Class.forName(className); Class defaultDecodingClass = Class.forName(className);
defaultEncoding = encoding;
} }
catch (ClassNotFoundException ex) catch (ClassNotFoundException ex)
{ {
throw new NoClassDefFoundError("missing default encoding " throw new NoClassDefFoundError("missing default encoding "
+ encoding + " (class " + encoding + " (class "
+ className + " not found)"); + className
+ " not found)");
} }
} }
} }
return getDecoder (defaultEncoding);
public abstract String getName();
public static BytesToUnicode getDefaultDecoder()
{
try
{
if (defaultDecodingClass == null)
getDefaultDecodingClass();
return (BytesToUnicode) defaultDecodingClass.newInstance();
} }
catch (Throwable ex) catch (Throwable ex)
{ {
try
{
return new Input_iconv (System.getProperty ("file.encoding"));
}
catch (Throwable ex2)
{
return new Input_8859_1(); return new Input_8859_1();
} }
} }
}
/** Get a byte-stream->char-stream converter given an encoding name. */ /** Get a byte-stream->char-stream converter given an encoding name. */
public static BytesToUnicode getDecoder (String encoding) public static BytesToUnicode getDecoder (String encoding)
throws java.io.UnsupportedEncodingException throws java.io.UnsupportedEncodingException
{ {
/* First hunt in our cache to see if we have a decoder that is
already allocated. */
synchronized (BytesToUnicode.class)
{
int i;
for (i = 0; i < decoderCache.length; ++i)
{
if (decoderCache[i] != null
&& encoding.equals(decoderCache[i].getName ()))
{
BytesToUnicode rv = decoderCache[i];
decoderCache[i] = null;
return rv;
}
}
}
// It's not in the cache, so now we have to do real work.
String className = "gnu.gcj.convert.Input_" + canonicalize (encoding); String className = "gnu.gcj.convert.Input_" + canonicalize (encoding);
Class decodingClass; Class decodingClass;
try try
...@@ -120,4 +144,22 @@ public abstract class BytesToUnicode extends IOConverter ...@@ -120,4 +144,22 @@ public abstract class BytesToUnicode extends IOConverter
* of the length parameter for a read request). * of the length parameter for a read request).
*/ */
public abstract int read (char[] outbuffer, int outpos, int count); public abstract int read (char[] outbuffer, int outpos, int count);
/** Indicate that the converter is resuable.
* This class keeps track of converters on a per-encoding basis.
* When done with an encoder you may call this method to indicate
* that it can be reused later.
*/
public void done ()
{
synchronized (BytesToUnicode.class)
{
this.inbuffer = null;
this.inpos = 0;
this.inlength = 0;
decoderCache[currCachePos] = this;
currCachePos = (currCachePos + 1) % CACHE_SIZE;
}
}
} }
// Input_iconv.java -- Java side of iconv() reader. // Input_iconv.java -- Java side of iconv() reader.
/* Copyright (C) 2000 Free Software Foundation /* Copyright (C) 2000, 2001 Free Software Foundation
This file is part of libgcj. This file is part of libgcj.
...@@ -33,6 +33,7 @@ public class Input_iconv extends BytesToUnicode ...@@ -33,6 +33,7 @@ public class Input_iconv extends BytesToUnicode
private native void init (String encoding) private native void init (String encoding)
throws UnsupportedEncodingException; throws UnsupportedEncodingException;
public native int read (char[] outbuffer, int outpos, int count); public native int read (char[] outbuffer, int outpos, int count);
public native void done ();
// The encoding we're using. // The encoding we're using.
private String encoding; private String encoding;
......
// Output_iconv.java -- Java side of iconv() writer. // Output_iconv.java -- Java side of iconv() writer.
/* Copyright (C) 2000 Free Software Foundation /* Copyright (C) 2000, 2001 Free Software Foundation
This file is part of libgcj. This file is part of libgcj.
...@@ -33,6 +33,7 @@ public class Output_iconv extends UnicodeToBytes ...@@ -33,6 +33,7 @@ public class Output_iconv extends UnicodeToBytes
private native void init (String encoding) private native void init (String encoding)
throws UnsupportedEncodingException; throws UnsupportedEncodingException;
public native int write (char[] inbuffer, int inpos, int count); public native int write (char[] inbuffer, int inpos, int count);
public native void done ();
// The encoding we're using. // The encoding we're using.
private String encoding; private String encoding;
......
/* Copyright (C) 1999, 2000 Free Software Foundation /* Copyright (C) 1999, 2000, 2001 Free Software Foundation
This file is part of libgcj. This file is part of libgcj.
...@@ -15,56 +15,79 @@ public abstract class UnicodeToBytes extends IOConverter ...@@ -15,56 +15,79 @@ public abstract class UnicodeToBytes extends IOConverter
public byte[] buf; public byte[] buf;
public int count; public int count;
static Class defaultEncodingClass; // The name of the default encoding.
static String defaultEncoding;
static synchronized void getDefaultEncodingClass() /* These keep a small cache of encoders for reuse. The array holds
the actual encoders. The currCachePos is the next value we are
going to replace in the cache. We don't just throw the data away
if the cache is full, because if the cache filled up with stuff we
don't need then the cache would be worthless. We instead
circulate through the cache the implement kind of an LRU
algorithm. */
private static final int CACHE_SIZE = 4; // A power of 2 for speed
private static UnicodeToBytes[] encoderCache
= new UnicodeToBytes[CACHE_SIZE];
private static int currCachePos = 0;
public abstract String getName();
public static UnicodeToBytes getDefaultEncoder()
{ {
// Test (defaultEncodingClass == null) again in case of race condition. try
if (defaultEncodingClass == null)
{ {
String encoding = canonicalize (System.getProperty("file.encoding")); synchronized (UnicodeToBytes.class)
String className = "gnu.gcj.convert.Output_"+encoding; {
if (defaultEncoding == null)
{
String encoding
= canonicalize (System.getProperty("file.encoding",
"8859_1"));
String className = "gnu.gcj.convert.Output_" + encoding;
try try
{ {
defaultEncodingClass = Class.forName(className); Class defaultEncodingClass = Class.forName(className);
defaultEncoding = encoding;
} }
catch (ClassNotFoundException ex) catch (ClassNotFoundException ex)
{ {
throw new NoClassDefFoundError("missing default encoding " throw new NoClassDefFoundError("missing default encoding "
+ encoding + " (class " + encoding + " (class "
+ className + " not found)"); + className
+ " not found)");
} }
} }
} }
public abstract String getName(); return getEncoder (defaultEncoding);
public static UnicodeToBytes getDefaultEncoder()
{
try
{
if (defaultEncodingClass == null)
getDefaultEncodingClass();
return (UnicodeToBytes) defaultEncodingClass.newInstance();
} }
catch (Throwable ex) catch (Throwable ex)
{ {
try
{
return new Output_iconv (System.getProperty ("file.encoding"));
}
catch (Throwable ex2)
{
return new Output_8859_1(); return new Output_8859_1();
} }
} }
}
/** Get a char-stream->byte-stream converter given an encoding name. */ /** Get a char-stream->byte-stream converter given an encoding name. */
public static UnicodeToBytes getEncoder (String encoding) public static UnicodeToBytes getEncoder (String encoding)
throws java.io.UnsupportedEncodingException throws java.io.UnsupportedEncodingException
{ {
/* First hunt in our cache to see if we have a encoder that is
already allocated. */
synchronized (UnicodeToBytes.class)
{
int i;
for (i = 0; i < encoderCache.length; ++i)
{
if (encoderCache[i] != null
&& encoding.equals(encoderCache[i].getName ()))
{
UnicodeToBytes rv = encoderCache[i];
encoderCache[i] = null;
return rv;
}
}
}
String className = "gnu.gcj.convert.Output_" + canonicalize (encoding); String className = "gnu.gcj.convert.Output_" + canonicalize (encoding);
Class encodingClass; Class encodingClass;
try try
...@@ -122,4 +145,21 @@ public abstract class UnicodeToBytes extends IOConverter ...@@ -122,4 +145,21 @@ public abstract class UnicodeToBytes extends IOConverter
str.getChars(inpos, srcEnd, work, 0); str.getChars(inpos, srcEnd, work, 0);
return write(work, inpos, inlength); return write(work, inpos, inlength);
} }
/** Indicate that the converter is resuable.
* This class keeps track of converters on a per-encoding basis.
* When done with an encoder you may call this method to indicate
* that it can be reused later.
*/
public void done ()
{
synchronized (UnicodeToBytes.class)
{
this.buf = null;
this.count = 0;
encoderCache[currCachePos] = this;
currCachePos = (currCachePos + 1) % CACHE_SIZE;
}
}
} }
...@@ -91,7 +91,7 @@ gnu::gcj::convert::Input_iconv::read (jcharArray outbuffer, ...@@ -91,7 +91,7 @@ gnu::gcj::convert::Input_iconv::read (jcharArray outbuffer,
if (r == (size_t) -1) if (r == (size_t) -1)
{ {
// Incomplete character. // Incomplete character.
if (errno == EINVAL) if (errno == EINVAL || errno == E2BIG)
return 0; return 0;
throw new java::io::CharConversionException (); throw new java::io::CharConversionException ();
} }
...@@ -116,6 +116,20 @@ gnu::gcj::convert::Input_iconv::read (jcharArray outbuffer, ...@@ -116,6 +116,20 @@ gnu::gcj::convert::Input_iconv::read (jcharArray outbuffer,
} }
void void
gnu::gcj::convert::Input_iconv::done ()
{
// 50 bytes should be enough for any reset sequence.
size_t avail = 50;
char tmp[avail];
char *p = tmp;
// Calling iconv() with a NULL INBUF pointer will cause iconv() to
// switch to its initial state. We don't care about the output that
// might be generated in that situation.
iconv_adapter (iconv, (iconv_t) handle, NULL, NULL, &p, &avail);
BytesToUnicode::done ();
}
void
gnu::gcj::convert::Output_iconv::init (jstring encoding) gnu::gcj::convert::Output_iconv::init (jstring encoding)
{ {
#ifdef HAVE_ICONV #ifdef HAVE_ICONV
...@@ -251,3 +265,17 @@ gnu::gcj::convert::IOConverter::iconv_init (void) ...@@ -251,3 +265,17 @@ gnu::gcj::convert::IOConverter::iconv_init (void)
#endif /* HAVE_ICONV */ #endif /* HAVE_ICONV */
return result; return result;
} }
void
gnu::gcj::convert::Output_iconv::done ()
{
// 50 bytes should be enough for any reset sequence.
size_t avail = 50;
char tmp[avail];
char *p = tmp;
// Calling iconv() with a NULL INBUF pointer will cause iconv() to
// switch to its initial state. We don't care about the output that
// might be generated in that situation.
iconv_adapter (iconv, (iconv_t) handle, NULL, NULL, &p, &avail);
UnicodeToBytes::done ();
}
...@@ -523,6 +523,7 @@ java::lang::String::init (jbyteArray bytes, jint offset, jint count, ...@@ -523,6 +523,7 @@ java::lang::String::init (jbyteArray bytes, jint offset, jint count,
avail -= done; avail -= done;
} }
} }
converter->done ();
this->data = array; this->data = array;
this->boffset = (char *) elements (array) - (char *) array; this->boffset = (char *) elements (array) - (char *) array;
this->count = outpos; this->count = outpos;
...@@ -604,6 +605,7 @@ java::lang::String::getBytes (jstring enc) ...@@ -604,6 +605,7 @@ java::lang::String::getBytes (jstring enc)
todo -= converted; todo -= converted;
} }
} }
converter->done ();
if (bufpos == buflen) if (bufpos == buflen)
return buffer; return buffer;
jbyteArray result = JvNewByteArray(bufpos); jbyteArray result = JvNewByteArray(bufpos);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment