summaryrefslogtreecommitdiff
path: root/libavformat/os_support.c
diff options
context:
space:
mode:
authorMartin Storsjö <martin@martin.st>2010-01-11 17:27:07 +0000
committerRonald S. Bultje <rsbultje@gmail.com>2010-01-11 17:27:07 +0000
commitfa053ca725358e728f6ad70f01be7a4dedeebd57 (patch)
treea9c47640db1f72c5539d29020b31709146209722 /libavformat/os_support.c
parentc0018613513f283138e3afa7dc5438993d635bea (diff)
Provide fallback implementations of getaddrinfo() and freeaddrinfo().
Patch by Martin Storsjö <$firstname()$firstname,st>. Originally committed as revision 21145 to svn://svn.ffmpeg.org/ffmpeg/trunk
Diffstat (limited to 'libavformat/os_support.c')
-rw-r--r--libavformat/os_support.c70
1 files changed, 70 insertions, 0 deletions
diff --git a/libavformat/os_support.c b/libavformat/os_support.c
index 87481d674e..62fd183afd 100644
--- a/libavformat/os_support.c
+++ b/libavformat/os_support.c
@@ -60,6 +60,76 @@ int inet_aton (const char * str, struct in_addr * add)
}
#endif /* !HAVE_INET_ATON */
+#if !HAVE_GETADDRINFO
+int ff_getaddrinfo(const char *node, const char *service,
+ const struct addrinfo *hints, struct addrinfo **res)
+{
+ struct hostent *h = NULL;
+ struct addrinfo *ai;
+ struct sockaddr_in *sin;
+
+ sin = av_mallocz(sizeof(struct sockaddr_in));
+ if (!sin)
+ return EAI_FAIL;
+ sin->sin_family = AF_INET;
+
+ if (node) {
+ if (!inet_aton(node, &sin->sin_addr)) {
+ if (hints && (hints->ai_flags & AI_NUMERICHOST)) {
+ av_free(sin);
+ return EAI_FAIL;
+ }
+ h = gethostbyname(node);
+ if (!h) {
+ av_free(sin);
+ return EAI_FAIL;
+ }
+ memcpy(&sin->sin_addr, h->h_addr_list[0], sizeof(struct in_addr));
+ }
+ } else {
+ if (hints && (hints->ai_flags & AI_PASSIVE)) {
+ sin->sin_addr.s_addr = INADDR_ANY;
+ } else
+ sin->sin_addr.s_addr = INADDR_LOOPBACK;
+ }
+
+ /* Note: getaddrinfo allows service to be a string, which
+ * should be looked up using getservbyname. */
+ if (service)
+ sin->sin_port = htons(atoi(service));
+
+ ai = av_mallocz(sizeof(struct addrinfo));
+ if (!ai) {
+ av_free(sin);
+ return EAI_FAIL;
+ }
+
+ *res = ai;
+ ai->ai_family = AF_INET;
+ ai->ai_socktype = hints ? hints->ai_socktype : 0;
+ switch (ai->ai_socktype) {
+ case SOCK_STREAM: ai->ai_protocol = IPPROTO_TCP; break;
+ case SOCK_DGRAM: ai->ai_protocol = IPPROTO_UDP; break;
+ default: ai->ai_protocol = 0; break;
+ }
+
+ ai->ai_addr = (struct sockaddr *)sin;
+ ai->ai_addrlen = sizeof(struct sockaddr_in);
+ if (hints && (hints->ai_flags & AI_CANONNAME))
+ ai->ai_canonname = h ? av_strdup(h->h_name) : NULL;
+
+ ai->ai_next = NULL;
+ return 0;
+}
+
+void ff_freeaddrinfo(struct addrinfo *res)
+{
+ av_free(res->ai_canonname);
+ av_free(res->ai_addr);
+ av_free(res);
+}
+#endif
+
/* resolve host with also IP address parsing */
int resolve_host(struct in_addr *sin_addr, const char *hostname)
{