IdTCPClient on Android: Connecting to non-active network - Printable Version +- Atozed Forums (https://www.atozed.com/forums) +-- Forum: Indy (https://www.atozed.com/forums/forum-8.html) +--- Forum: Indy General Discussion (https://www.atozed.com/forums/forum-9.html) +--- Thread: IdTCPClient on Android: Connecting to non-active network (/thread-126.html) Pages:
1
2
|
RE: IdTCPClient on Android: Connecting to non-active network - rlebeau - 05-08-2018 (05-08-2018, 08:55 AM)BartKindt Wrote: But now I get an AV: An AV at a memory address near 0 usually means you are accessing a nil pointer. My guess would be the TIdTCPClient.Socket.Binding object hasn't been created yet when you call setsockopt(). Try moving that call into the TIdTCPClient.OnSocketAllocated or TIdTCPClient.OnAfterBind event instead. That ensures the Socket.Binding object and its socket Handle are both ready for use. RE: IdTCPClient on Android: Connecting to non-active network - BartKindt - 05-09-2018 (05-08-2018, 05:12 PM)rlebeau Wrote: An AV at a memory address near 0 usually means you are accessing a nil pointer. My guess would be the TIdTCPClient's Socket.Binding object hasn't been created yet when you call setsockopt(). Try moving that call into the TIdTCPClient's OnSocketAllocated or OnAfterBind event instead. That ensures the Socket.Binding object and its socket Handle are both ready for use. Okay. Fixed the AV. Now I have a socket error: Code: procedure TTCPThread.IdTCPClient1OnAfterBind(ASender: TObject); This generates: IdTCPClient1.Socket.Binding.Handle=42 ERROR: TCPThread: GStack failed: WIFIInterface=wlan0 Length(WIFIInterface)=5: Socket Error # 1 I do hope we are not in a situation where Android somehow blocks any attempt to connect to a 'non-active network'. But if this is at Linux level, how could that be? So, the WIFI IS connected (enabled) and the name 'wlan0' is valid, as I can see a lot of data passing through the Android debug window PS, this is what the unfiltered Android Log looks like. There is no other issue reported by Android during the call with GStack. You can see the WIFI is connected to my test WIFI (with no Internet) called "SARTrack_SAT" on interface "wlan0" Code: 45: I/info(16249): SARTrack Service: [S] D3: IdTCPClient: OnSocketAllocated: Bart RE: IdTCPClient on Android: Connecting to non-active network - rlebeau - 05-09-2018 (05-09-2018, 01:36 PM)BartKindt Wrote: You don't need that check. The GStack pointer is always valid whenever an Indy component is alive. (05-09-2018, 01:36 PM)BartKindt Wrote: Not sure if this applies to Android's flavor of Linux, but on some Linux platforms, SO_BINDTODEVICE takes an ifreq struct as input instead of a null-terminated string pointer, eg: Code: struct ifreq ifr; Which would translate to Delphi as (roughly): Code: uses (05-09-2018, 01:36 PM)BartKindt Wrote: ERROR: TCPThread: GStack failed: WIFIInterface=wlan0 Length(WIFIInterface)=5: Socket Error # 1 Error #1 is EPERM ("Operation not permitted"). You likely need root/sudo access to your Android device to bind to a specific interface by name. RE: IdTCPClient on Android: Connecting to non-active network - BartKindt - 05-10-2018 I am getting quite desperate. I managed to get your new code to work, but I still cannot get connected to the WIFI network, when the Android 'Active Network' is set to MOBILE. Code: uses Posix.SysSocket, Posix.NetIf, Posix.Termios, Posix.StrOpts 1) When Android sets the Active Network to 'wlan0', and I use same name in the ioctl, this function gives a result of '0'. AND I now get connected to the WIFI based server. 2) When Android sets its Active Network to the MOBILE network, the ioctl return still '0', but I do NOT get any connection to local WIFI server, nothing goes out over the WIFI network. 3) Using situation (1), but now I use a non-valid interface of 'wlan1', the ioctl gives an error result '-1' as it should. But then afterwards, I still get connected to the WIFI server: that is, after the failure of the ioctl, it reverts back to the 'default' network of 'wlan0' (which is active at that moment). This means that the ioctl works correctly. But I still do net get access to the 'wlan0' interface, unless Android has set this to the "Active Network". And note that the ioctl return a successful '0'. I am not sure however, if there is another error hidden somewhere in the interface What else can I do? Can you please explain to me how Indy knows to which of the two networks to connect all by itself? There are TWO working interfaces available. But Indy automatically select the one which Android has set to the 'Active Network'. How does this work then? As Indy works directly with the Linux interfaces? Why would Indy use the one Android selects at Linux level? E.g. What makes it a "default"?? Thanks, Bart RE: IdTCPClient on Android: Connecting to non-active network - rlebeau - 05-10-2018 (05-10-2018, 05:12 PM)BartKindt Wrote: I managed to get your new code to work, but I still cannot get connected to the WIFI network, when the Android 'Active Network' is set to MOBILE. I don't know what else to tell you then. Maybe Android is just not physically connected to the WiFi when it is connected to the Mobile network? I'm not an Android expert. There are only so many options available to tell the low-level POSIX socket API which network to use, and we have pretty much exhausted them at this point. The only thing I have left to suggest is to not use Indy at all, but use Android's own Socket API instead. The Network.bindSocket() method can bind a Socket to a specific network. Or, you can create a Socket on a specific network using the Network.getSocketFactory() and SocketFactory.createSocket() methods. (05-10-2018, 05:12 PM)BartKindt Wrote: Can you please explain to me how Indy knows to which of the two networks to connect all by itself? Indy doesn't pick either one. If you don't explicitly bind the socket to a specific local IP or device name before the socket connection is attempted, the OS picks the appropriate network while connecting the socket to the target host/IP. It picks that network based on available network routes. (05-10-2018, 05:12 PM)BartKindt Wrote: But Indy automatically select the one which Android has set to the 'Active Network'. Indy is not the one making that decision, Android/Linux itself is. RE: IdTCPClient on Android: Connecting to non-active network - BartKindt - 05-10-2018 Okay Remy, I was hoping I could bypass the Android system and use Indy's way of selecting the interface. It looks like it is not possible. I will try the Android-based ideas you gave me and see what I can and cannot do with that. The biggest problem is that I cannot find anywhere a forum where I can actually ask Delphi-Android based questions, and get actually a real answer. I think I have been too spoiled over the many years with the incredible amount of help I got with my Windows based development, and mostly from you... Many thanks again for all your help, Bart RE: IdTCPClient on Android: Connecting to non-active network - rlebeau - 05-10-2018 (05-10-2018, 07:16 PM)BartKindt Wrote: I was hoping I could bypass the Android system and use Indy's way of selecting the interface. That would be nice. But Android is a managed platform, Google doesn't really want users going down to the Linux level unless they really need to. But, since Delphi Android apps operate inside the Java native NDK, Indy operates at the Linux POSIX level, not at the Android Java level. (05-10-2018, 07:16 PM)BartKindt Wrote: I will try the Android-based ideas you gave me and see what I can and cannot do with that. See Connecting your App to a Wi-Fi Device on the Android Developers Blog. However, despite what is documented, I see from Android's source code that:
And according to Linux's documentation: Quote:SO_MARK (since Linux 2.6.25) So again, you might need root/sudo access to bind a socket to a specific network. Unless Android's INTERNET or NET_ADMIN permissions enable CAP_NET_ADMIN, but I don't think they do. So, I'm still at a loss of how to proceed here. You could try what the blog says, or you could try using SO_MARK directly, and hope it somehow works out OK in your app without messing around with permissions too much. (05-10-2018, 07:16 PM)BartKindt Wrote: The biggest problem is that I cannot find anywhere a forum where I can actually ask Delphi-Android based questions, and get actually a real answer. In the absence of the Embarcadero forums, you can ask Delphi-related questions in the Delphi section of this same forum server, or in the Delphi section of StackOverflow, or any number of other Delphi forums scattered online (though I only frequent the ones I have mentioned). RE: IdTCPClient on Android: Connecting to non-active network - BartKindt - 05-12-2018 Thanks Remy, I will work my way through the options you mentioned. Bart |