The header files ("Generating Header Files") created by javah tell us the names and type signatures of all the routines we need to define.
The extern routines defined in our javah generated header files are the C prototypes for the functions we need to implement. These map directly onto the native methods from our Java class. In order to actually implement the native methods, we need to include some other Java header files. We need these files, because they define Java internal data structures and function prototypes that are essential in manipulating Java class data structures. The unhand() macro, which returns the "data" portion of an Java object, is an example of this.
/* * Copyright (c) 1994, 1995 by Sun Microsystems, Inc. * All Rights Reserved. * * @(#)file.c 94/12/08 1.3 * * December 1994, Eugene Kuerner * * * This file includes the native code used to implement methods * in the demo package of classes. Specifically, classes * InputFile and OutputFile have native methods that interact * with the underlying system. * * The prototype for the non-static functions are found in the * javah generated header file for the class. Also, the javah * generated stubfile takes care of marshalling the function * parameters from the Java world into the C world. * * * version 1.0, 01 Dec 1994 * author Eugene Kuerner * */ /* * First off, we need to include the Java internal header * files for macros and function prototypes required to * manipulate Javadata structures and functions. * * SubPreamble.h includes the structure and macro definitions * needed to convert Java data structures into C data * structures. For example, macros such as "unhand" are defined * in StubPreamble.h. * * javaString.h defines the Java string maniuplation macros and * routines needed to convert between Java and C strings. * */ #include "StubPreamble.h" #include "javaString.h" /* * These headers are special in that they are generated by JAVAH. * These include the C structure definitions for the Java * classes. * */ #include "demo_InputFile.h" #include "demo_OutputFile.h" /* * These are some standard Unix headers we use to implement our * system dependent native methods. * */ #include <sys/types.h> #include <sys/param.h> #include <stdio.h> #include <fcntl.h> #include <errno.h> /* * Here we define constants used by this module. In our case we * need to define out local path separator, because it is * different from the one used in Java. Note that Java's * version path separator is defined in the JAVAH generated * header files. * */ #define LOCAL_PATH_SEPARATOR `/' /* * static void * convertPath(char *) * * This function provides C string path conversions from those * separated by the File separator character to the system * defined one. * */ static void convertPath(char *str) { while (*str != `\0') { if ((*str == demo_InputFile_separatorChar) || (*str == demo_OutputFile_separatorChar)) { *str = LOCAL_PATH_SEPARATOR; } str++; } return; } /* * long * demo_InputFile_open(struct Hdemo_InputFile *) * * This function implements the native method open in class * InputFile from package demo. This fact is evidenced by the * name. The function name is derived by concatenating the * package name with the class name with the method name. The * semantics for this method are defined in the class file. * * The parameter to this function can be thought of as a "handle * to a demo_InputFile object." The actual makeup of this is * the data portion of the class followed by the method table * for the class. * */ long demo_InputFile_open(struct Hdemo_InputFile *this) { int fd; char buf[MAXPATHLEN]; /* * we need to convert Java path string into a C string and * then change it to a system dependent path string. * note that "unhand" gets the data portion of the handle to * the C representation of the Java class InputFile. * */ javaString2CString(unhand(this)->path, buf, sizeof(buf)); convertPath(buf); /* * now we actually call the system dependent routines to * implement the method. note that we could throw an * exception via SignalError() instead of returning a * boolean. * */ fd = open(buf, O_RDONLY); if (fd < 0) return(FALSE); /* * on success we store the system fd into the class variable * for future calls to this library. * */ unhand(this)->fd = fd; return(TRUE); } /* * void * demo_InputFile_close(struct Hdemo_InputFile *) * * Implements the close method for class InputFile in package * demo. * */ void demo_InputFile_close(struct Hdemo_InputFile *this) { close(unhand(this)->fd); unhand(this)->fd = -1; return; } /* * long * demo_InputFile_read(struct Hdemo_InputFile *, HArrayOfByte * *, long ) * * Implements the read method for class InputFile in package * demo. * */ long demo_InputFile_read(struct Hdemo_InputFile *this, HArrayOfByte *buffer, long count) { /* * we use another macro here called obj_length which returns * the length of an Java array object. again, unhand of * the Array object gets us the data structure portion of * the object. * */ char *data = unhand(buffer)->body; int len = obj_length(buffer); int actual; if (len < count) { actual = len; } else { actual = count; } actual = read(unhand(this)->fd, data, actual); if (actual == 0) return(-1); return(actual); } /* * long * demo_OutputFile_open(struct Hdemo_OutputFile *) * * Implements the open method for class OutputFile in package * demo. It is virually identical to demo_InputFile_open. * Please, refer to the comments for that function for more * information. * */ long demo_OutputFile_open(struct Hdemo_OutputFile *this) { int fd; char buf[MAXPATHLEN]; javaString2CString(unhand(this)->path, buf, sizeof(buf)); convertPath(buf); fd = open(buf, O_RDWR|O_CREAT|O_TRUNC, 0644); if (fd < 0) return(FALSE); unhand(this)->fd = fd; return(TRUE); } /* * void * demo_OutputFile_close(struct Hdemo_OutputFile *) * * Implements the close method for class OutputFile in package demo. * */ void demo_OutputFile_close(struct Hdemo_OutputFile *this) { close(unhand(this)->fd); unhand(this)->fd = -1; return; } /* * long * demo_OutputFile_write(struct Hdemo_OutputFile *, HArrayOfByte *, long ) * * Implements the write method for class OutputFile in package * demo. * */ long demo_OutputFile_write(struct Hdemo_OutputFile *this, HArrayOfByte *buffer, long count) { char *data = unhand(buffer)->body; int len = obj_length(buffer); int actual; if (len < count) { actual = len; } else { actual = count; } actual = write(unhand(this)->fd, data, actual); return(actual); }As an absolute minimum interface this C file will work. It defines all the necessary symbols so we can build a dynamically loadable library (DLL) with it. The DLL is linked into the Java interpreter via the Java Linker class at runtime. This allows the successful invocation of the native methods without throwing exceptions.
Implementing Native Methods
Generated with CERN WebMaker