jnr.ffi.provider.converters.StringBuilderParameterConverter Maven / Gradle / Ivy
/*
* Copyright (C) 2012 Wayne Meissner
*
* This file is part of the JNR project.
*
* 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.
*/
package jnr.ffi.provider.converters;
import jnr.ffi.mapper.ToNativeContext;
import jnr.ffi.mapper.ToNativeConverter;
import jnr.ffi.provider.ParameterFlags;
import java.lang.ref.Reference;
import java.nio.ByteBuffer;
import java.nio.CharBuffer;
import java.nio.charset.CharacterCodingException;
import java.nio.charset.Charset;
import java.nio.charset.CharsetDecoder;
import java.nio.charset.CharsetEncoder;
import java.nio.charset.CoderResult;
import static jnr.ffi.provider.converters.StringUtil.getCharset;
import static jnr.ffi.provider.converters.StringUtil.getDecoder;
import static jnr.ffi.provider.converters.StringUtil.getEncoder;
import static jnr.ffi.provider.converters.StringUtil.stringLength;
import static jnr.ffi.provider.converters.StringUtil.terminatorWidth;
import static jnr.ffi.provider.converters.StringUtil.throwException;
@ToNativeConverter.NoContext
@ToNativeConverter.Cacheable
public class StringBuilderParameterConverter implements ToNativeConverter, ToNativeConverter.PostInvocation {
private final ThreadLocal> localEncoder = new ThreadLocal>();
private final ThreadLocal> localDecoder = new ThreadLocal>();
private final Charset charset;
private final int parameterFlags;
private final int terminatorWidth;
private StringBuilderParameterConverter(Charset charset, int parameterFlags) {
this.charset = charset;
this.parameterFlags = parameterFlags;
this.terminatorWidth = terminatorWidth(charset);
}
public Class nativeType() {
return ByteBuffer.class;
}
public static StringBuilderParameterConverter getInstance(int parameterFlags, ToNativeContext toNativeContext) {
return new StringBuilderParameterConverter(getCharset(toNativeContext), parameterFlags);
}
public static StringBuilderParameterConverter getInstance(Charset charset, int parameterFlags, ToNativeContext toNativeContext) {
return new StringBuilderParameterConverter(charset, parameterFlags);
}
public ByteBuffer toNative(StringBuilder parameter, ToNativeContext context) {
if (parameter == null) {
return null;
} else {
CharsetEncoder encoder = getEncoder(charset, localEncoder);
ByteBuffer byteBuffer = ByteBuffer.wrap(new byte[parameter.capacity() * (int) Math.ceil(encoder.maxBytesPerChar()) + 4]);
if (ParameterFlags.isIn(parameterFlags)) {
byteBuffer.mark();
encoder.reset();
CoderResult result = encoder.encode(CharBuffer.wrap(parameter), byteBuffer, true);
if (result.isUnderflow()) result = encoder.flush(byteBuffer);
if (result.isError()) throwException(result);
byteBuffer.reset();
}
return byteBuffer;
}
}
public void postInvoke(StringBuilder stringBuilder, ByteBuffer buf, ToNativeContext context) {
//
// Copy the string back out if its an OUT parameter
//
if (ParameterFlags.isOut(parameterFlags) && stringBuilder != null && buf != null) {
buf.limit(stringLength(buf, terminatorWidth));
try {
stringBuilder.delete(0, stringBuilder.length()).append(getDecoder(charset, localDecoder).reset().decode(buf));
} catch (CharacterCodingException cce) {
throw new RuntimeException(cce);
}
}
}
}