android_net_NetUtils.cpp revision a47c40193fcbe9ca07facea78b2828afdae5e025
1/* 2 * Copyright 2008, The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17#define LOG_TAG "NetUtils" 18 19#include "jni.h" 20#include <utils/misc.h> 21#include <android_runtime/AndroidRuntime.h> 22#include <utils/Log.h> 23#include <arpa/inet.h> 24#include <cutils/properties.h> 25 26extern "C" { 27int ifc_enable(const char *ifname); 28int ifc_disable(const char *ifname); 29int ifc_add_route(const char *ifname, const char *destStr, uint32_t prefixLen, const char *gwStr); 30int ifc_remove_host_routes(const char *ifname); 31int ifc_get_default_route(const char *ifname); 32int ifc_remove_default_route(const char *ifname); 33int ifc_reset_connections(const char *ifname); 34 35int dhcp_do_request(const char *ifname, 36 const char *ipaddr, 37 const char *gateway, 38 uint32_t *prefixLength, 39 const char *dns1, 40 const char *dns2, 41 const char *server, 42 uint32_t *lease); 43 44int dhcp_do_request_renew(const char *ifname, 45 const char *ipaddr, 46 const char *gateway, 47 uint32_t *prefixLength, 48 const char *dns1, 49 const char *dns2, 50 const char *server, 51 uint32_t *lease); 52 53int dhcp_stop(const char *ifname); 54int dhcp_release_lease(const char *ifname); 55char *dhcp_get_errmsg(); 56} 57 58#define NETUTILS_PKG_NAME "android/net/NetworkUtils" 59 60namespace android { 61 62/* 63 * The following remembers the jfieldID's of the fields 64 * of the DhcpInfo Java object, so that we don't have 65 * to look them up every time. 66 */ 67static struct fieldIds { 68 jclass dhcpInfoInternalClass; 69 jmethodID constructorId; 70 jfieldID ipaddress; 71 jfieldID prefixLength; 72 jfieldID dns1; 73 jfieldID dns2; 74 jfieldID serverAddress; 75 jfieldID leaseDuration; 76} dhcpInfoInternalFieldIds; 77 78static jint android_net_utils_enableInterface(JNIEnv* env, jobject clazz, jstring ifname) 79{ 80 int result; 81 82 const char *nameStr = env->GetStringUTFChars(ifname, NULL); 83 result = ::ifc_enable(nameStr); 84 env->ReleaseStringUTFChars(ifname, nameStr); 85 return (jint)result; 86} 87 88static jint android_net_utils_disableInterface(JNIEnv* env, jobject clazz, jstring ifname) 89{ 90 int result; 91 92 const char *nameStr = env->GetStringUTFChars(ifname, NULL); 93 result = ::ifc_disable(nameStr); 94 env->ReleaseStringUTFChars(ifname, nameStr); 95 return (jint)result; 96} 97 98static jint android_net_utils_addRoute(JNIEnv* env, jobject clazz, jstring ifname, 99 jstring dst, jint prefixLength, jstring gw) 100{ 101 int result; 102 103 const char *nameStr = env->GetStringUTFChars(ifname, NULL); 104 const char *dstStr = env->GetStringUTFChars(dst, NULL); 105 const char *gwStr = NULL; 106 if (gw != NULL) { 107 gwStr = env->GetStringUTFChars(gw, NULL); 108 } 109 result = ::ifc_add_route(nameStr, dstStr, prefixLength, gwStr); 110 env->ReleaseStringUTFChars(ifname, nameStr); 111 env->ReleaseStringUTFChars(dst, dstStr); 112 if (gw != NULL) { 113 env->ReleaseStringUTFChars(gw, gwStr); 114 } 115 return (jint)result; 116} 117 118static jint android_net_utils_removeHostRoutes(JNIEnv* env, jobject clazz, jstring ifname) 119{ 120 int result; 121 122 const char *nameStr = env->GetStringUTFChars(ifname, NULL); 123 result = ::ifc_remove_host_routes(nameStr); 124 env->ReleaseStringUTFChars(ifname, nameStr); 125 return (jint)result; 126} 127 128static jint android_net_utils_getDefaultRoute(JNIEnv* env, jobject clazz, jstring ifname) 129{ 130 int result; 131 132 const char *nameStr = env->GetStringUTFChars(ifname, NULL); 133 result = ::ifc_get_default_route(nameStr); 134 env->ReleaseStringUTFChars(ifname, nameStr); 135 return (jint)result; 136} 137 138static jint android_net_utils_removeDefaultRoute(JNIEnv* env, jobject clazz, jstring ifname) 139{ 140 int result; 141 142 const char *nameStr = env->GetStringUTFChars(ifname, NULL); 143 result = ::ifc_remove_default_route(nameStr); 144 env->ReleaseStringUTFChars(ifname, nameStr); 145 return (jint)result; 146} 147 148static jint android_net_utils_resetConnections(JNIEnv* env, jobject clazz, jstring ifname) 149{ 150 int result; 151 152 const char *nameStr = env->GetStringUTFChars(ifname, NULL); 153 result = ::ifc_reset_connections(nameStr); 154 env->ReleaseStringUTFChars(ifname, nameStr); 155 return (jint)result; 156} 157 158static jboolean android_net_utils_runDhcpCommon(JNIEnv* env, jobject clazz, jstring ifname, 159 jobject info, bool renew) 160{ 161 int result; 162 char ipaddr[PROPERTY_VALUE_MAX]; 163 uint32_t prefixLength; 164 char gateway[PROPERTY_VALUE_MAX]; 165 char dns1[PROPERTY_VALUE_MAX]; 166 char dns2[PROPERTY_VALUE_MAX]; 167 char server[PROPERTY_VALUE_MAX]; 168 uint32_t lease; 169 170 const char *nameStr = env->GetStringUTFChars(ifname, NULL); 171 if (nameStr == NULL) return (jboolean)false; 172 173 if (renew) { 174 result = ::dhcp_do_request_renew(nameStr, ipaddr, gateway, &prefixLength, 175 dns1, dns2, server, &lease); 176 } else { 177 result = ::dhcp_do_request(nameStr, ipaddr, gateway, &prefixLength, 178 dns1, dns2, server, &lease); 179 } 180 181 env->ReleaseStringUTFChars(ifname, nameStr); 182 if (result == 0 && dhcpInfoInternalFieldIds.dhcpInfoInternalClass != NULL) { 183 env->SetObjectField(info, dhcpInfoInternalFieldIds.ipaddress, env->NewStringUTF(ipaddr)); 184 185 // set the gateway 186 jclass cls = env->FindClass("java/net/InetAddress"); 187 jmethodID method = env->GetStaticMethodID(cls, "getByName", 188 "(Ljava/lang/String;)Ljava/net/InetAddress;"); 189 jvalue args[1]; 190 args[0].l = env->NewStringUTF(gateway); 191 jobject inetAddressObject = env->CallStaticObjectMethodA(cls, method, args); 192 193 if (!env->ExceptionOccurred()) { 194 cls = env->FindClass("android/net/RouteInfo"); 195 method = env->GetMethodID(cls, "<init>", "(Ljava/net/InetAddress;)V"); 196 args[0].l = inetAddressObject; 197 jobject routeInfoObject = env->NewObjectA(cls, method, args); 198 199 cls = env->FindClass("android/net/DhcpInfoInternal"); 200 method = env->GetMethodID(cls, "addRoute", "(Landroid/net/RouteInfo;)V"); 201 args[0].l = routeInfoObject; 202 env->CallVoidMethodA(info, method, args); 203 } else { 204 // if we have an exception (host not found perhaps), just don't add the route 205 env->ExceptionClear(); 206 } 207 208 env->SetIntField(info, dhcpInfoInternalFieldIds.prefixLength, prefixLength); 209 env->SetObjectField(info, dhcpInfoInternalFieldIds.dns1, env->NewStringUTF(dns1)); 210 env->SetObjectField(info, dhcpInfoInternalFieldIds.dns2, env->NewStringUTF(dns2)); 211 env->SetObjectField(info, dhcpInfoInternalFieldIds.serverAddress, 212 env->NewStringUTF(server)); 213 env->SetIntField(info, dhcpInfoInternalFieldIds.leaseDuration, lease); 214 } 215 return (jboolean)(result == 0); 216} 217 218static jboolean android_net_utils_runDhcp(JNIEnv* env, jobject clazz, jstring ifname, jobject info) 219{ 220 return android_net_utils_runDhcpCommon(env, clazz, ifname, info, false); 221} 222 223static jboolean android_net_utils_runDhcpRenew(JNIEnv* env, jobject clazz, jstring ifname, jobject info) 224{ 225 return android_net_utils_runDhcpCommon(env, clazz, ifname, info, true); 226} 227 228 229static jboolean android_net_utils_stopDhcp(JNIEnv* env, jobject clazz, jstring ifname) 230{ 231 int result; 232 233 const char *nameStr = env->GetStringUTFChars(ifname, NULL); 234 result = ::dhcp_stop(nameStr); 235 env->ReleaseStringUTFChars(ifname, nameStr); 236 return (jboolean)(result == 0); 237} 238 239static jboolean android_net_utils_releaseDhcpLease(JNIEnv* env, jobject clazz, jstring ifname) 240{ 241 int result; 242 243 const char *nameStr = env->GetStringUTFChars(ifname, NULL); 244 result = ::dhcp_release_lease(nameStr); 245 env->ReleaseStringUTFChars(ifname, nameStr); 246 return (jboolean)(result == 0); 247} 248 249static jstring android_net_utils_getDhcpError(JNIEnv* env, jobject clazz) 250{ 251 return env->NewStringUTF(::dhcp_get_errmsg()); 252} 253 254// ---------------------------------------------------------------------------- 255 256/* 257 * JNI registration. 258 */ 259static JNINativeMethod gNetworkUtilMethods[] = { 260 /* name, signature, funcPtr */ 261 262 { "enableInterface", "(Ljava/lang/String;)I", (void *)android_net_utils_enableInterface }, 263 { "disableInterface", "(Ljava/lang/String;)I", (void *)android_net_utils_disableInterface }, 264 { "addRoute", "(Ljava/lang/String;Ljava/lang/String;ILjava/lang/String;)I", 265 (void *)android_net_utils_addRoute }, 266 { "removeHostRoutes", "(Ljava/lang/String;)I", (void *)android_net_utils_removeHostRoutes }, 267 { "getDefaultRouteNative", "(Ljava/lang/String;)I", 268 (void *)android_net_utils_getDefaultRoute }, 269 { "removeDefaultRoute", "(Ljava/lang/String;)I", (void *)android_net_utils_removeDefaultRoute }, 270 { "resetConnections", "(Ljava/lang/String;)I", (void *)android_net_utils_resetConnections }, 271 { "runDhcp", "(Ljava/lang/String;Landroid/net/DhcpInfoInternal;)Z", (void *)android_net_utils_runDhcp }, 272 { "runDhcpRenew", "(Ljava/lang/String;Landroid/net/DhcpInfoInternal;)Z", (void *)android_net_utils_runDhcpRenew }, 273 { "stopDhcp", "(Ljava/lang/String;)Z", (void *)android_net_utils_stopDhcp }, 274 { "releaseDhcpLease", "(Ljava/lang/String;)Z", (void *)android_net_utils_releaseDhcpLease }, 275 { "getDhcpError", "()Ljava/lang/String;", (void*) android_net_utils_getDhcpError }, 276}; 277 278int register_android_net_NetworkUtils(JNIEnv* env) 279{ 280 jclass netutils = env->FindClass(NETUTILS_PKG_NAME); 281 LOG_FATAL_IF(netutils == NULL, "Unable to find class " NETUTILS_PKG_NAME); 282 283 dhcpInfoInternalFieldIds.dhcpInfoInternalClass = env->FindClass("android/net/DhcpInfoInternal"); 284 if (dhcpInfoInternalFieldIds.dhcpInfoInternalClass != NULL) { 285 dhcpInfoInternalFieldIds.constructorId = env->GetMethodID(dhcpInfoInternalFieldIds.dhcpInfoInternalClass, "<init>", "()V"); 286 dhcpInfoInternalFieldIds.ipaddress = env->GetFieldID(dhcpInfoInternalFieldIds.dhcpInfoInternalClass, "ipAddress", "Ljava/lang/String;"); 287 dhcpInfoInternalFieldIds.prefixLength = env->GetFieldID(dhcpInfoInternalFieldIds.dhcpInfoInternalClass, "prefixLength", "I"); 288 dhcpInfoInternalFieldIds.dns1 = env->GetFieldID(dhcpInfoInternalFieldIds.dhcpInfoInternalClass, "dns1", "Ljava/lang/String;"); 289 dhcpInfoInternalFieldIds.dns2 = env->GetFieldID(dhcpInfoInternalFieldIds.dhcpInfoInternalClass, "dns2", "Ljava/lang/String;"); 290 dhcpInfoInternalFieldIds.serverAddress = env->GetFieldID(dhcpInfoInternalFieldIds.dhcpInfoInternalClass, "serverAddress", "Ljava/lang/String;"); 291 dhcpInfoInternalFieldIds.leaseDuration = env->GetFieldID(dhcpInfoInternalFieldIds.dhcpInfoInternalClass, "leaseDuration", "I"); 292 } 293 294 return AndroidRuntime::registerNativeMethods(env, 295 NETUTILS_PKG_NAME, gNetworkUtilMethods, NELEM(gNetworkUtilMethods)); 296} 297 298}; // namespace android 299