📄 jniconnectionhandler.java
字号:
/* * $Header: /home/cvs/jakarta-tomcat/src/share/org/apache/tomcat/service/connector/Attic/JNIConnectionHandler.java,v 1.9.2.2 2000/12/12 09:41:44 hgomez Exp $ * $Revision: 1.9.2.2 $ * $Date: 2000/12/12 09:41:44 $ * * ==================================================================== * * The Apache Software License, Version 1.1 * * Copyright (c) 1999 The Apache Software Foundation. All rights * reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * 3. The end-user documentation included with the redistribution, if * any, must include the following acknowlegement: * "This product includes software developed by the * Apache Software Foundation (http://www.apache.org/)." * Alternately, this acknowlegement may appear in the software itself, * if and wherever such third-party acknowlegements normally appear. * * 4. The names "The Jakarta Project", "Tomcat", and "Apache Software * Foundation" must not be used to endorse or promote products derived * from this software without prior written permission. For written * permission, please contact apache@apache.org. * * 5. Products derived from this software may not be called "Apache" * nor may "Apache" appear in their names without prior written * permission of the Apache Group. * * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * ==================================================================== * * This software consists of voluntary contributions made by many * individuals on behalf of the Apache Software Foundation. For more * information on the Apache Software Foundation, please see * <http://www.apache.org/>. * * [Additional notices, if required by prior licensing conditions] * */package org.apache.tomcat.service.connector;import java.io.IOException;import org.apache.tomcat.core.*;import org.apache.tomcat.util.*;import javax.servlet.ServletInputStream;import java.util.Vector;import java.io.File;public class JNIConnectionHandler { ContextManager contextM; public JNIConnectionHandler() { } public void setServer(Object contextM) { this.contextM=(ContextManager)contextM; } public void setNativeLibrary(String lib) { // First try to load from the library path try { System.loadLibrary(lib); System.out.println("Library " + lib + " was loaded from the lib path"); return; } catch(UnsatisfiedLinkError usl) { //usl.printStackTrace(); System.err.println("Failed to loadLibrary() " + lib); } // Loading from the library path failed // Try to load assuming lib is a complete pathname. try { System.load(lib); System.out.println("Library " + lib + " loaded"); return; } catch(UnsatisfiedLinkError usl) { System.err.println("Failed to load() " + lib); //usl.printStackTrace(); } // OK, try to load from the default libexec // directory. // libexec directory = tomcat.home + / + libexec File f = new File(System.getProperties().getProperty("tomcat.home"), "libexec"); if(System.getProperty( "os.name" ).toLowerCase().indexOf("windows") >= 0) { f = new File(f, "jni_connect.dll"); } else { f = new File(f, "jni_connect.so"); } System.load(f.toString()); System.out.println("Library " + f.toString() + " loaded"); } static Vector pool=new Vector(); static boolean reuse=true; public void processConnection(long s, long l) { JNIRequestAdapter reqA=null; JNIResponseAdapter resA=null; try { if( reuse ) { synchronized( this ) { if( pool.size()==0 ) { reqA=new JNIRequestAdapter( contextM, this); resA=new JNIResponseAdapter( this ); contextM.initRequest( reqA, resA ); } else { reqA = (JNIRequestAdapter)pool.lastElement(); resA=(JNIResponseAdapter)reqA.getResponse(); pool.removeElement( reqA ); } } reqA.recycle(); resA.recycle(); } else { reqA = new JNIRequestAdapter(contextM, this); resA =new JNIResponseAdapter(this); contextM.initRequest( reqA , resA ); } resA.setRequestAttr(s, l); reqA.readNextRequest(s, l); if(reqA.shutdown ) return; if(resA.getStatus() >= 400) { resA.finish(); return; } int contentLength = reqA.getFacade().getIntHeader("content-length"); if (contentLength != -1) { BufferedServletInputStream sis = (BufferedServletInputStream)reqA.getInputStream(); sis.setLimit(contentLength); } contextM.service( reqA, resA ); } catch(Exception ex) { ex.printStackTrace(); } if( reuse ) { synchronized( this ) { pool.addElement( reqA ); } } } native int readEnvironment(long s, long l, String []env); native int getNumberOfHeaders(long s, long l); native int readHeaders(long s, long l, String []names, String []values); native int read(long s, long l, byte []buf, int from, int cnt); native int startReasponse(long s, long l, int sc, String msg, String []headerNames, String []headerValues, int headerCnt); native int write(long s, long l, byte []buf, int from, int cnt);}class JNIRequestAdapter extends RequestImpl { static StringManager sm = StringManager.getManager("org.apache.tomcat.service"); ContextManager contextM; boolean shutdown=false; JNIConnectionHandler h; long s; long l; public int doRead() throws IOException { byte []b = new byte[1]; int rc = doRead(b, 0, 1); if(rc <= 0) { return -1; } return ((int)b[0]) & 0x000000FF; } public int doRead(byte b[], int off, int len) throws IOException { int rc = 0; while(0 == rc) { rc = h.read(s, l, b, off, len); if(0 == rc) { Thread.currentThread().yield(); } } return rc; } public JNIRequestAdapter(ContextManager cm, JNIConnectionHandler h) { this.contextM = cm; this.h = h; } protected void readNextRequest(long s, long l) throws IOException { String []env = new String[15]; int i = 0; this.s = s; this.l = l; for(i = 0 ; i < 12 ; i++) { env[i] = null; } /* * Read the environment */ if(h.readEnvironment(s, l, env) > 0) { method = env[0]; requestURI = env[1]; queryString = env[2]; remoteAddr = env[3]; remoteHost = env[4]; serverName = env[5]; serverPort = Integer.parseInt(env[6]); authType = env[7]; remoteUser = env[8]; scheme = env[9]; protocol = env[10]; // response.setServerHeader(env[11]); if(scheme.equalsIgnoreCase("https")) { if(null != env[12]) { attributes.put("javax.servlet.request.X509Certificate", env[12]); } if(null != env[13]) { attributes.put("javax.servlet.request.cipher_suite", env[13]); } if(null != env[14]) { attributes.put("javax.servlet.request.ssl_session", env[14]); } } } else { throw new IOException("Error: JNI implementation error"); } /* * Read the headers */ int nheaders = h.getNumberOfHeaders(s, l); if(nheaders > 0) { String []names = new String[nheaders]; String []values = new String[nheaders]; if(h.readHeaders(s, l, names, values) > 0) { for(i = 0 ; i < nheaders ; i++) { headers.putHeader(names[i].toLowerCase(), values[i]); } } else { throw new IOException("Error: JNI implementation error"); } } // REQUEST_URI may include a query string int idQ= requestURI.indexOf("?"); if ( idQ > -1) { requestURI = requestURI.substring(0, idQ); } contentLength = headers.getIntHeader("content-length"); contentType = headers.getHeader("content-type"); } public ServletInputStream getInputStream() throws IOException { if(contentLength <= 0) { throw new IOException("Empty input stream"); } in = new BufferedServletInputStream(this); return in; }}// Ajp use Status: instead of Statusclass JNIResponseAdapter extends ResponseImpl { JNIConnectionHandler h; long s; long l; public JNIResponseAdapter(JNIConnectionHandler h) { this.h = h; } protected void setRequestAttr(long s, long l) throws IOException { this.s = s; this.l = l; } public void endHeaders() throws IOException { if(request.getProtocol()==null) // HTTP/0.9 return; super.endHeaders(); // Servlet Engine header will be set per/adapter - smarter adapters will // not send it every time ( have it in C side ), and we may also want // to add informations about the adapter used if( request.getContext() != null) setHeader("Servlet-Engine", request.getContext().getEngineHeader()); int hcnt = 0; String []headerNames = null; String []headerValues = null; headers.removeHeader("Status"); hcnt = headers.size(); headerNames = new String[hcnt]; headerValues = new String[hcnt]; for(int i = 0; i < hcnt; i++) { MimeHeaderField h = headers.getField(i); headerNames[i] = h.getName(); headerValues[i] = h.getValue(); } if(h.startReasponse(s, l, status, getMessage(status), headerNames, headerValues, hcnt) <= 0) { throw new IOException("Error: JNI startReasponse implementation error"); } } public void doWrite(byte buf[], int pos, int count) throws IOException { if(h.write(s, l, buf, pos, count) <= 0) { throw new IOException("Error: JNI implementation error"); } }}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -