PageRenderTime 68ms CodeModel.GetById 25ms RepoModel.GetById 0ms app.codeStats 0ms

/frameworks/base/core/jni/android_net_LocalSocketImpl.cpp

https://gitlab.com/brian0218/rk3066_r-box_android4.2.2_sdk
C++ | 964 lines | 660 code | 212 blank | 92 comment | 166 complexity | 31bebe8631fd45dfaf98f05e7188e083 MD5 | raw file
  1. /*
  2. * Copyright (C) 2006 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. #define LOG_TAG "LocalSocketImpl"
  17. #include "JNIHelp.h"
  18. #include "jni.h"
  19. #include "utils/Log.h"
  20. #include "utils/misc.h"
  21. #include <stdio.h>
  22. #include <string.h>
  23. #include <sys/types.h>
  24. #include <sys/socket.h>
  25. #include <sys/un.h>
  26. #include <arpa/inet.h>
  27. #include <netinet/in.h>
  28. #include <stdlib.h>
  29. #include <errno.h>
  30. #include <unistd.h>
  31. #include <sys/ioctl.h>
  32. #include <cutils/sockets.h>
  33. #include <netinet/tcp.h>
  34. namespace android {
  35. static jfieldID field_inboundFileDescriptors;
  36. static jfieldID field_outboundFileDescriptors;
  37. static jclass class_Credentials;
  38. static jclass class_FileDescriptor;
  39. static jmethodID method_CredentialsInit;
  40. /*
  41. * private native FileDescriptor
  42. * create_native(boolean stream)
  43. * throws IOException;
  44. */
  45. static jobject
  46. socket_create (JNIEnv *env, jobject object, jboolean stream)
  47. {
  48. int ret;
  49. ret = socket(PF_LOCAL, stream ? SOCK_STREAM : SOCK_DGRAM, 0);
  50. if (ret < 0) {
  51. jniThrowIOException(env, errno);
  52. return NULL;
  53. }
  54. return jniCreateFileDescriptor(env,ret);
  55. }
  56. /* private native void connectLocal(FileDescriptor fd,
  57. * String name, int namespace) throws IOException
  58. */
  59. static void
  60. socket_connect_local(JNIEnv *env, jobject object,
  61. jobject fileDescriptor, jstring name, jint namespaceId)
  62. {
  63. int ret;
  64. const char *nameUtf8;
  65. int fd;
  66. nameUtf8 = env->GetStringUTFChars(name, NULL);
  67. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  68. if (env->ExceptionOccurred() != NULL) {
  69. return;
  70. }
  71. ret = socket_local_client_connect(
  72. fd,
  73. nameUtf8,
  74. namespaceId,
  75. SOCK_STREAM);
  76. env->ReleaseStringUTFChars(name, nameUtf8);
  77. if (ret < 0) {
  78. jniThrowIOException(env, errno);
  79. return;
  80. }
  81. }
  82. #define DEFAULT_BACKLOG 4
  83. /* private native void bindLocal(FileDescriptor fd, String name, namespace)
  84. * throws IOException;
  85. */
  86. static void
  87. socket_bind_local (JNIEnv *env, jobject object, jobject fileDescriptor,
  88. jstring name, jint namespaceId)
  89. {
  90. int ret;
  91. int fd;
  92. const char *nameUtf8;
  93. if (name == NULL) {
  94. jniThrowNullPointerException(env, NULL);
  95. }
  96. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  97. if (env->ExceptionOccurred() != NULL) {
  98. return;
  99. }
  100. nameUtf8 = env->GetStringUTFChars(name, NULL);
  101. ret = socket_local_server_bind(fd, nameUtf8, namespaceId);
  102. env->ReleaseStringUTFChars(name, nameUtf8);
  103. if (ret < 0) {
  104. jniThrowIOException(env, errno);
  105. return;
  106. }
  107. }
  108. /* private native void listen_native(int fd, int backlog) throws IOException; */
  109. static void
  110. socket_listen (JNIEnv *env, jobject object, jobject fileDescriptor, int backlog)
  111. {
  112. int ret;
  113. int fd;
  114. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  115. if (env->ExceptionOccurred() != NULL) {
  116. return;
  117. }
  118. ret = listen(fd, backlog);
  119. if (ret < 0) {
  120. jniThrowIOException(env, errno);
  121. return;
  122. }
  123. }
  124. /* private native FileDescriptor
  125. ** accept (FileDescriptor fd, LocalSocketImpl s)
  126. ** throws IOException;
  127. */
  128. static jobject
  129. socket_accept (JNIEnv *env, jobject object, jobject fileDescriptor, jobject s)
  130. {
  131. union {
  132. struct sockaddr address;
  133. struct sockaddr_un un_address;
  134. } sa;
  135. int ret;
  136. int retFD;
  137. int fd;
  138. socklen_t addrlen;
  139. if (s == NULL) {
  140. jniThrowNullPointerException(env, NULL);
  141. return NULL;
  142. }
  143. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  144. if (env->ExceptionOccurred() != NULL) {
  145. return NULL;
  146. }
  147. do {
  148. addrlen = sizeof(sa);
  149. ret = accept(fd, &(sa.address), &addrlen);
  150. } while (ret < 0 && errno == EINTR);
  151. if (ret < 0) {
  152. jniThrowIOException(env, errno);
  153. return NULL;
  154. }
  155. retFD = ret;
  156. return jniCreateFileDescriptor(env, retFD);
  157. }
  158. /* private native void shutdown(FileDescriptor fd, boolean shutdownInput) */
  159. static void
  160. socket_shutdown (JNIEnv *env, jobject object, jobject fileDescriptor,
  161. jboolean shutdownInput)
  162. {
  163. int ret;
  164. int fd;
  165. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  166. if (env->ExceptionOccurred() != NULL) {
  167. return;
  168. }
  169. ret = shutdown(fd, shutdownInput ? SHUT_RD : SHUT_WR);
  170. if (ret < 0) {
  171. jniThrowIOException(env, errno);
  172. return;
  173. }
  174. }
  175. static bool
  176. java_opt_to_real(int optID, int* opt, int* level)
  177. {
  178. switch (optID)
  179. {
  180. case 4098:
  181. *opt = SO_RCVBUF;
  182. *level = SOL_SOCKET;
  183. return true;
  184. case 4097:
  185. *opt = SO_SNDBUF;
  186. *level = SOL_SOCKET;
  187. return true;
  188. case 4102:
  189. *opt = SO_SNDTIMEO;
  190. *level = SOL_SOCKET;
  191. return true;
  192. case 128:
  193. *opt = SO_LINGER;
  194. *level = SOL_SOCKET;
  195. return true;
  196. case 1:
  197. *opt = TCP_NODELAY;
  198. *level = IPPROTO_TCP;
  199. return true;
  200. case 4:
  201. *opt = SO_REUSEADDR;
  202. *level = SOL_SOCKET;
  203. return true;
  204. }
  205. return false;
  206. }
  207. static jint
  208. socket_getOption(JNIEnv *env, jobject object, jobject fileDescriptor, int optID)
  209. {
  210. int ret, value;
  211. int opt, level;
  212. int fd;
  213. socklen_t size = sizeof(int);
  214. if (!java_opt_to_real(optID, &opt, &level)) {
  215. jniThrowIOException(env, -1);
  216. return 0;
  217. }
  218. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  219. if (env->ExceptionOccurred() != NULL) {
  220. return 0;
  221. }
  222. switch (opt)
  223. {
  224. case SO_LINGER:
  225. {
  226. struct linger lingr;
  227. size = sizeof(lingr);
  228. ret = getsockopt(fd, level, opt, &lingr, &size);
  229. if (!lingr.l_onoff) {
  230. value = -1;
  231. } else {
  232. value = lingr.l_linger;
  233. }
  234. break;
  235. }
  236. default:
  237. ret = getsockopt(fd, level, opt, &value, &size);
  238. break;
  239. }
  240. if (ret != 0) {
  241. jniThrowIOException(env, errno);
  242. return 0;
  243. }
  244. return value;
  245. }
  246. static void socket_setOption(
  247. JNIEnv *env, jobject object, jobject fileDescriptor, int optID,
  248. jint boolValue, jint intValue) {
  249. int ret;
  250. int optname;
  251. int level;
  252. int fd;
  253. if (!java_opt_to_real(optID, &optname, &level)) {
  254. jniThrowIOException(env, -1);
  255. return;
  256. }
  257. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  258. if (env->ExceptionOccurred() != NULL) {
  259. return;
  260. }
  261. switch (optname) {
  262. case SO_LINGER: {
  263. /*
  264. * SO_LINGER is special because it needs to use a special
  265. * "linger" struct as well as use the incoming boolean
  266. * argument specially.
  267. */
  268. struct linger lingr;
  269. lingr.l_onoff = boolValue ? 1 : 0; // Force it to be 0 or 1.
  270. lingr.l_linger = intValue;
  271. ret = setsockopt(fd, level, optname, &lingr, sizeof(lingr));
  272. break;
  273. }
  274. case SO_SNDTIMEO: {
  275. /*
  276. * SO_TIMEOUT from the core library gets converted to
  277. * SO_SNDTIMEO, but the option is supposed to set both
  278. * send and receive timeouts. Note: The incoming timeout
  279. * value is in milliseconds.
  280. */
  281. struct timeval timeout;
  282. timeout.tv_sec = intValue / 1000;
  283. timeout.tv_usec = (intValue % 1000) * 1000;
  284. ret = setsockopt(fd, SOL_SOCKET, SO_RCVTIMEO,
  285. (void *)&timeout, sizeof(timeout));
  286. if (ret == 0) {
  287. ret = setsockopt(fd, SOL_SOCKET, SO_SNDTIMEO,
  288. (void *)&timeout, sizeof(timeout));
  289. }
  290. break;
  291. }
  292. default: {
  293. /*
  294. * In all other cases, the translated option level and
  295. * optname may be used directly for a call to setsockopt().
  296. */
  297. ret = setsockopt(fd, level, optname, &intValue, sizeof(intValue));
  298. break;
  299. }
  300. }
  301. if (ret != 0) {
  302. jniThrowIOException(env, errno);
  303. return;
  304. }
  305. }
  306. static jint socket_available (JNIEnv *env, jobject object,
  307. jobject fileDescriptor)
  308. {
  309. int fd;
  310. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  311. if (env->ExceptionOccurred() != NULL) {
  312. return (jint)-1;
  313. }
  314. #if 1
  315. int avail;
  316. int ret = ioctl(fd, FIONREAD, &avail);
  317. // If this were a non-socket fd, there would be other cases to worry
  318. // about...
  319. if (ret < 0) {
  320. jniThrowIOException(env, errno);
  321. return (jint) 0;
  322. }
  323. return (jint)avail;
  324. #else
  325. // there appears to be a bionic bug that prevents this version from working.
  326. ssize_t ret;
  327. struct msghdr msg;
  328. memset(&msg, 0, sizeof(msg));
  329. do {
  330. ret = recvmsg(fd, &msg, MSG_PEEK | MSG_DONTWAIT | MSG_NOSIGNAL);
  331. } while (ret < 0 && errno == EINTR);
  332. // MSG_PEEK returns 0 on EOF and EWOULDBLOCK on none available
  333. if (ret < 0 && errno == EWOULDBLOCK) {
  334. return 0;
  335. } if (ret < 0) {
  336. jniThrowIOException(env, errno);
  337. return -1;
  338. }
  339. return (jint)ret;
  340. #endif
  341. }
  342. static void socket_close (JNIEnv *env, jobject object, jobject fileDescriptor)
  343. {
  344. int fd;
  345. int err;
  346. if (fileDescriptor == NULL) {
  347. jniThrowNullPointerException(env, NULL);
  348. return;
  349. }
  350. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  351. if (env->ExceptionOccurred() != NULL) {
  352. return;
  353. }
  354. do {
  355. err = close(fd);
  356. } while (err < 0 && errno == EINTR);
  357. if (err < 0) {
  358. jniThrowIOException(env, errno);
  359. return;
  360. }
  361. }
  362. /**
  363. * Processes ancillary data, handling only
  364. * SCM_RIGHTS. Creates appropriate objects and sets appropriate
  365. * fields in the LocalSocketImpl object. Returns 0 on success
  366. * or -1 if an exception was thrown.
  367. */
  368. static int socket_process_cmsg(JNIEnv *env, jobject thisJ, struct msghdr * pMsg)
  369. {
  370. struct cmsghdr *cmsgptr;
  371. for (cmsgptr = CMSG_FIRSTHDR(pMsg);
  372. cmsgptr != NULL; cmsgptr = CMSG_NXTHDR(pMsg, cmsgptr)) {
  373. if (cmsgptr->cmsg_level != SOL_SOCKET) {
  374. continue;
  375. }
  376. if (cmsgptr->cmsg_type == SCM_RIGHTS) {
  377. int *pDescriptors = (int *)CMSG_DATA(cmsgptr);
  378. jobjectArray fdArray;
  379. int count
  380. = ((cmsgptr->cmsg_len - CMSG_LEN(0)) / sizeof(int));
  381. if (count < 0) {
  382. jniThrowException(env, "java/io/IOException",
  383. "invalid cmsg length");
  384. }
  385. fdArray = env->NewObjectArray(count, class_FileDescriptor, NULL);
  386. if (fdArray == NULL) {
  387. return -1;
  388. }
  389. for (int i = 0; i < count; i++) {
  390. jobject fdObject
  391. = jniCreateFileDescriptor(env, pDescriptors[i]);
  392. if (env->ExceptionOccurred() != NULL) {
  393. return -1;
  394. }
  395. env->SetObjectArrayElement(fdArray, i, fdObject);
  396. if (env->ExceptionOccurred() != NULL) {
  397. return -1;
  398. }
  399. }
  400. env->SetObjectField(thisJ, field_inboundFileDescriptors, fdArray);
  401. if (env->ExceptionOccurred() != NULL) {
  402. return -1;
  403. }
  404. }
  405. }
  406. return 0;
  407. }
  408. /**
  409. * Reads data from a socket into buf, processing any ancillary data
  410. * and adding it to thisJ.
  411. *
  412. * Returns the length of normal data read, or -1 if an exception has
  413. * been thrown in this function.
  414. */
  415. static ssize_t socket_read_all(JNIEnv *env, jobject thisJ, int fd,
  416. void *buffer, size_t len)
  417. {
  418. ssize_t ret;
  419. ssize_t bytesread = 0;
  420. struct msghdr msg;
  421. struct iovec iv;
  422. unsigned char *buf = (unsigned char *)buffer;
  423. // Enough buffer for a pile of fd's. We throw an exception if
  424. // this buffer is too small.
  425. struct cmsghdr cmsgbuf[2*sizeof(cmsghdr) + 0x100];
  426. memset(&msg, 0, sizeof(msg));
  427. memset(&iv, 0, sizeof(iv));
  428. iv.iov_base = buf;
  429. iv.iov_len = len;
  430. msg.msg_iov = &iv;
  431. msg.msg_iovlen = 1;
  432. msg.msg_control = cmsgbuf;
  433. msg.msg_controllen = sizeof(cmsgbuf);
  434. do {
  435. ret = recvmsg(fd, &msg, MSG_NOSIGNAL);
  436. } while (ret < 0 && errno == EINTR);
  437. if (ret < 0 && errno == EPIPE) {
  438. // Treat this as an end of stream
  439. return 0;
  440. }
  441. if (ret < 0) {
  442. jniThrowIOException(env, errno);
  443. return -1;
  444. }
  445. if ((msg.msg_flags & (MSG_CTRUNC | MSG_OOB | MSG_ERRQUEUE)) != 0) {
  446. // To us, any of the above flags are a fatal error
  447. jniThrowException(env, "java/io/IOException",
  448. "Unexpected error or truncation during recvmsg()");
  449. return -1;
  450. }
  451. if (ret >= 0) {
  452. socket_process_cmsg(env, thisJ, &msg);
  453. }
  454. return ret;
  455. }
  456. /**
  457. * Writes all the data in the specified buffer to the specified socket.
  458. *
  459. * Returns 0 on success or -1 if an exception was thrown.
  460. */
  461. static int socket_write_all(JNIEnv *env, jobject object, int fd,
  462. void *buf, size_t len)
  463. {
  464. ssize_t ret;
  465. struct msghdr msg;
  466. unsigned char *buffer = (unsigned char *)buf;
  467. memset(&msg, 0, sizeof(msg));
  468. jobjectArray outboundFds
  469. = (jobjectArray)env->GetObjectField(
  470. object, field_outboundFileDescriptors);
  471. if (env->ExceptionOccurred() != NULL) {
  472. return -1;
  473. }
  474. struct cmsghdr *cmsg;
  475. int countFds = outboundFds == NULL ? 0 : env->GetArrayLength(outboundFds);
  476. int fds[countFds];
  477. char msgbuf[CMSG_SPACE(countFds)];
  478. // Add any pending outbound file descriptors to the message
  479. if (outboundFds != NULL) {
  480. if (env->ExceptionOccurred() != NULL) {
  481. return -1;
  482. }
  483. for (int i = 0; i < countFds; i++) {
  484. jobject fdObject = env->GetObjectArrayElement(outboundFds, i);
  485. if (env->ExceptionOccurred() != NULL) {
  486. return -1;
  487. }
  488. fds[i] = jniGetFDFromFileDescriptor(env, fdObject);
  489. if (env->ExceptionOccurred() != NULL) {
  490. return -1;
  491. }
  492. }
  493. // See "man cmsg" really
  494. msg.msg_control = msgbuf;
  495. msg.msg_controllen = sizeof msgbuf;
  496. cmsg = CMSG_FIRSTHDR(&msg);
  497. cmsg->cmsg_level = SOL_SOCKET;
  498. cmsg->cmsg_type = SCM_RIGHTS;
  499. cmsg->cmsg_len = CMSG_LEN(sizeof fds);
  500. memcpy(CMSG_DATA(cmsg), fds, sizeof fds);
  501. }
  502. // We only write our msg_control during the first write
  503. while (len > 0) {
  504. struct iovec iv;
  505. memset(&iv, 0, sizeof(iv));
  506. iv.iov_base = buffer;
  507. iv.iov_len = len;
  508. msg.msg_iov = &iv;
  509. msg.msg_iovlen = 1;
  510. do {
  511. ret = sendmsg(fd, &msg, MSG_NOSIGNAL);
  512. } while (ret < 0 && errno == EINTR);
  513. if (ret < 0) {
  514. jniThrowIOException(env, errno);
  515. return -1;
  516. }
  517. buffer += ret;
  518. len -= ret;
  519. // Wipes out any msg_control too
  520. memset(&msg, 0, sizeof(msg));
  521. }
  522. return 0;
  523. }
  524. static jint socket_read (JNIEnv *env, jobject object, jobject fileDescriptor)
  525. {
  526. int fd;
  527. int err;
  528. if (fileDescriptor == NULL) {
  529. jniThrowNullPointerException(env, NULL);
  530. return (jint)-1;
  531. }
  532. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  533. if (env->ExceptionOccurred() != NULL) {
  534. return (jint)0;
  535. }
  536. unsigned char buf;
  537. err = socket_read_all(env, object, fd, &buf, 1);
  538. if (err < 0) {
  539. jniThrowIOException(env, errno);
  540. return (jint)0;
  541. }
  542. if (err == 0) {
  543. // end of file
  544. return (jint)-1;
  545. }
  546. return (jint)buf;
  547. }
  548. static jint socket_readba (JNIEnv *env, jobject object,
  549. jbyteArray buffer, jint off, jint len, jobject fileDescriptor)
  550. {
  551. int fd;
  552. jbyte* byteBuffer;
  553. int ret;
  554. if (fileDescriptor == NULL || buffer == NULL) {
  555. jniThrowNullPointerException(env, NULL);
  556. return (jint)-1;
  557. }
  558. if (off < 0 || len < 0 || (off + len) > env->GetArrayLength(buffer)) {
  559. jniThrowException(env, "java/lang/ArrayIndexOutOfBoundsException", NULL);
  560. return (jint)-1;
  561. }
  562. if (len == 0) {
  563. // because socket_read_all returns 0 on EOF
  564. return 0;
  565. }
  566. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  567. if (env->ExceptionOccurred() != NULL) {
  568. return (jint)-1;
  569. }
  570. byteBuffer = env->GetByteArrayElements(buffer, NULL);
  571. if (NULL == byteBuffer) {
  572. // an exception will have been thrown
  573. return (jint)-1;
  574. }
  575. ret = socket_read_all(env, object,
  576. fd, byteBuffer + off, len);
  577. // A return of -1 above means an exception is pending
  578. env->ReleaseByteArrayElements(buffer, byteBuffer, 0);
  579. return (jint) ((ret == 0) ? -1 : ret);
  580. }
  581. static void socket_write (JNIEnv *env, jobject object,
  582. jint b, jobject fileDescriptor)
  583. {
  584. int fd;
  585. int err;
  586. if (fileDescriptor == NULL) {
  587. jniThrowNullPointerException(env, NULL);
  588. return;
  589. }
  590. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  591. if (env->ExceptionOccurred() != NULL) {
  592. return;
  593. }
  594. err = socket_write_all(env, object, fd, &b, 1);
  595. // A return of -1 above means an exception is pending
  596. }
  597. static void socket_writeba (JNIEnv *env, jobject object,
  598. jbyteArray buffer, jint off, jint len, jobject fileDescriptor)
  599. {
  600. int fd;
  601. int err;
  602. jbyte* byteBuffer;
  603. if (fileDescriptor == NULL || buffer == NULL) {
  604. jniThrowNullPointerException(env, NULL);
  605. return;
  606. }
  607. if (off < 0 || len < 0 || (off + len) > env->GetArrayLength(buffer)) {
  608. jniThrowException(env, "java/lang/ArrayIndexOutOfBoundsException", NULL);
  609. return;
  610. }
  611. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  612. if (env->ExceptionOccurred() != NULL) {
  613. return;
  614. }
  615. byteBuffer = env->GetByteArrayElements(buffer,NULL);
  616. if (NULL == byteBuffer) {
  617. // an exception will have been thrown
  618. return;
  619. }
  620. err = socket_write_all(env, object, fd,
  621. byteBuffer + off, len);
  622. // A return of -1 above means an exception is pending
  623. env->ReleaseByteArrayElements(buffer, byteBuffer, JNI_ABORT);
  624. }
  625. static jobject socket_get_peer_credentials(JNIEnv *env,
  626. jobject object, jobject fileDescriptor)
  627. {
  628. int err;
  629. int fd;
  630. if (fileDescriptor == NULL) {
  631. jniThrowNullPointerException(env, NULL);
  632. return NULL;
  633. }
  634. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  635. if (env->ExceptionOccurred() != NULL) {
  636. return NULL;
  637. }
  638. struct ucred creds;
  639. memset(&creds, 0, sizeof(creds));
  640. socklen_t szCreds = sizeof(creds);
  641. err = getsockopt(fd, SOL_SOCKET, SO_PEERCRED, &creds, &szCreds);
  642. if (err < 0) {
  643. jniThrowIOException(env, errno);
  644. return NULL;
  645. }
  646. if (szCreds == 0) {
  647. return NULL;
  648. }
  649. return env->NewObject(class_Credentials, method_CredentialsInit,
  650. creds.pid, creds.uid, creds.gid);
  651. }
  652. #if 0
  653. //TODO change this to return an instance of LocalSocketAddress
  654. static jobject socket_getSockName(JNIEnv *env,
  655. jobject object, jobject fileDescriptor)
  656. {
  657. int err;
  658. int fd;
  659. if (fileDescriptor == NULL) {
  660. jniThrowNullPointerException(env, NULL);
  661. return NULL;
  662. }
  663. fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
  664. if (env->ExceptionOccurred() != NULL) {
  665. return NULL;
  666. }
  667. union {
  668. struct sockaddr address;
  669. struct sockaddr_un un_address;
  670. } sa;
  671. memset(&sa, 0, sizeof(sa));
  672. socklen_t namelen = sizeof(sa);
  673. err = getsockname(fd, &(sa.address), &namelen);
  674. if (err < 0) {
  675. jniThrowIOException(env, errno);
  676. return NULL;
  677. }
  678. if (sa.address.sa_family != AF_UNIX) {
  679. // We think we're an impl only for AF_UNIX, so this should never happen.
  680. jniThrowIOException(env, EINVAL);
  681. return NULL;
  682. }
  683. if (sa.un_address.sun_path[0] == '\0') {
  684. } else {
  685. }
  686. }
  687. #endif
  688. /*
  689. * JNI registration.
  690. */
  691. static JNINativeMethod gMethods[] = {
  692. /* name, signature, funcPtr */
  693. {"getOption_native", "(Ljava/io/FileDescriptor;I)I", (void*)socket_getOption},
  694. {"setOption_native", "(Ljava/io/FileDescriptor;III)V", (void*)socket_setOption},
  695. {"create_native", "(Z)Ljava/io/FileDescriptor;", (void*)socket_create},
  696. {"connectLocal", "(Ljava/io/FileDescriptor;Ljava/lang/String;I)V",
  697. (void*)socket_connect_local},
  698. {"bindLocal", "(Ljava/io/FileDescriptor;Ljava/lang/String;I)V", (void*)socket_bind_local},
  699. {"listen_native", "(Ljava/io/FileDescriptor;I)V", (void*)socket_listen},
  700. {"accept", "(Ljava/io/FileDescriptor;Landroid/net/LocalSocketImpl;)Ljava/io/FileDescriptor;", (void*)socket_accept},
  701. {"shutdown", "(Ljava/io/FileDescriptor;Z)V", (void*)socket_shutdown},
  702. {"available_native", "(Ljava/io/FileDescriptor;)I", (void*) socket_available},
  703. {"close_native", "(Ljava/io/FileDescriptor;)V", (void*) socket_close},
  704. {"read_native", "(Ljava/io/FileDescriptor;)I", (void*) socket_read},
  705. {"readba_native", "([BIILjava/io/FileDescriptor;)I", (void*) socket_readba},
  706. {"writeba_native", "([BIILjava/io/FileDescriptor;)V", (void*) socket_writeba},
  707. {"write_native", "(ILjava/io/FileDescriptor;)V", (void*) socket_write},
  708. {"getPeerCredentials_native",
  709. "(Ljava/io/FileDescriptor;)Landroid/net/Credentials;",
  710. (void*) socket_get_peer_credentials}
  711. //,{"getSockName_native", "(Ljava/io/FileDescriptor;)Ljava/lang/String;",
  712. // (void *) socket_getSockName}
  713. };
  714. int register_android_net_LocalSocketImpl(JNIEnv *env)
  715. {
  716. jclass clazz;
  717. clazz = env->FindClass("android/net/LocalSocketImpl");
  718. if (clazz == NULL) {
  719. goto error;
  720. }
  721. field_inboundFileDescriptors = env->GetFieldID(clazz,
  722. "inboundFileDescriptors", "[Ljava/io/FileDescriptor;");
  723. if (field_inboundFileDescriptors == NULL) {
  724. goto error;
  725. }
  726. field_outboundFileDescriptors = env->GetFieldID(clazz,
  727. "outboundFileDescriptors", "[Ljava/io/FileDescriptor;");
  728. if (field_outboundFileDescriptors == NULL) {
  729. goto error;
  730. }
  731. class_Credentials = env->FindClass("android/net/Credentials");
  732. if (class_Credentials == NULL) {
  733. goto error;
  734. }
  735. class_Credentials = (jclass)env->NewGlobalRef(class_Credentials);
  736. class_FileDescriptor = env->FindClass("java/io/FileDescriptor");
  737. if (class_FileDescriptor == NULL) {
  738. goto error;
  739. }
  740. class_FileDescriptor = (jclass)env->NewGlobalRef(class_FileDescriptor);
  741. method_CredentialsInit
  742. = env->GetMethodID(class_Credentials, "<init>", "(III)V");
  743. if (method_CredentialsInit == NULL) {
  744. goto error;
  745. }
  746. return jniRegisterNativeMethods(env,
  747. "android/net/LocalSocketImpl", gMethods, NELEM(gMethods));
  748. error:
  749. ALOGE("Error registering android.net.LocalSocketImpl");
  750. return -1;
  751. }
  752. };