Re: A NSOutputStream vs. native socket question
Re: A NSOutputStream vs. native socket question
- Subject: Re: A NSOutputStream vs. native socket question
- From: Motti Shneor <email@hidden>
- Date: Mon, 25 Nov 2013 20:40:24 +0200
Hello again, and thanks Josh for the clear info.
In the mean time I was able to both remove the offensive code that changed my read/write native socket's buffer size, and write a whole mechanism that tracks the contents of the native socket buffer and can tell me what messages (of our proprietary protocol) were already sent, and what is still waiting in the socket.
For my bandwidth-control needs, I only querying the native socket for its buffer content size every 100ms, using getsockopt(socket,SOL_SOCKET,SO_NWRITE, &value, &valueSize); I believe this is accepted.
Now I'm looking into options for cleaning up the older implementation, and I wanted to ask about the "dynamic buffer sizes" of the native sockets. Especially the output socket.
I was wondering if I could completely get rid of the application level message-queue (currently a 1MB cyclic buffer) and work directly with the socket buffer.
When I last checked my native socket output buffer size, getsockopt(socket,SOL_SOCKET,SO_SNDBUF,&value,&valueSize); It started at 132KByte.
If we use the output stream for ~350KBPS video messages, that's about 3 seconds of video. In my environment, I may need more buffering. Something like 5-6 seconds, or 256KB.
My questions.
1. Does the "Dynamic buffer" increase its size if I try to write data faster? Does it become bigger when network latency increases? Can I rely on any such behavior? Is it at all documented?
2. What performance issues I may experience if I will manually change the output-buffer size, this time for a larger value than the default? (say, the above 256KB) ???
Motti.
>>>>
On 15 בנוב 2013, at 01:14, Josh Graessley <email@hidden> wrote:
>
> On Nov 14, 2013, at 12:58 PM, Motti Shneor <email@hidden> wrote:
>
>>
>> On 14 בנוב 2013, at 21:10, Josh Graessley wrote:
>>
>>>
>>> On Nov 14, 2013, at 10:53 AM, Motti Shneor <email@hidden> wrote:
>>>
>>>> Hi Everyone. First post to this group.
>>>>
>>>> I'm controlling the output stream created by
>>>>
>>>> ::CFStreamCreatePairWithSocketToHost(kCFAllocatorDefault, (CFStringRef)hostAddress, portNum, &readStream, &writeStream);
>>>> ::CFWriteStreamSetProperty(writeStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
>>>> ::CFReadStreamSetProperty(readStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
>>>>
>>>> and later bridged to NSInputStream and NSOutputStream instances,
>>>> via its internal native socket, using the posix getsockopt and setsockopt calls.
>>>>
>>>> The properties I need to work with are the read (input) and write (output) buffer sizes, and the write (output) socket's buffer pending content size.
>>>>
>>>> Things like this...
>>>> oss = getsockopt(socket,SOL_SOCKET,SO_RCVBUF,&value,&valueSize);
>>>> oss = setsockopt(socket,SOL_SOCKET,SO_SNDBUF, &value, sizeof(value);
>>>> oss = getsockopt(socket,SOL_SOCKET,SO_NWRITE, &value, &valueSize);
>>>>
>>>> etc.
>>>>
>>>> To retrieve the native socket I'm using code like this:
>>>>
>>>> // Getting the amount of bytes in the output socket buffer which are waiting to be sent
>>>> - (int) getSocketOutputBufferPendingSize:(uint32_t*)bufSize {
>>>> int oss = -1;
>>>> CFDataRef data = NULL;
>>>> do {
>>>> if (_outputStream==nil)
>>>> break;
>>>>
>>>> // retrieve native socket handle from output stream.
>>>> if((data = (CFDataRef)::CFWriteStreamCopyProperty((CFWriteStreamRef)_outputStream, kCFStreamPropertySocketNativeHandle))== NULL)
>>>> break;
>>>>
>>>> CFSocketNativeHandle socket;
>>>> CFDataGetBytes(data, CFRangeMake(0, sizeof(CFSocketNativeHandle)), (UInt8*)&socket);
>>>> .
>>>> .
>>>> . // --- here I work with the native socket properties.
>>>> .
>>>> } while (false);
>>>>
>>>> if(data!= NULL)
>>>> ::CFRelease(data);
>>>> return oss;
>>>> }
>>>>
>>>> My questions are:
>>>>
>>>> 1. What is the lifespan of the native socket handle?
>>>> 2. Can I retrieve it once, when the socket is created or opened the first time, then use it until socket is closed, or should I retrieve it from the stream like this, every time I want to use it?
>>>> 3. What is the cost of the above lines that retrieve the native socket handle?
>>>
>>> I hate to answer a question with another question, but…
>>>
>>> Why are you modifying SO_SNDBUF? Modifying SO_SNDBUF and SO_RCVBUF means the system can’t dynamically adjust these values based on network conditions. If you’re hand tuning these values, that might have a small improvement in performance in one network environment but it is likely to have a negative performance impact in other environments.
>>>
>>> Fetching the native file descriptor out of the socket stream does work but it doesn’t come without cost.
>>>
>>> -josh
>>>
>>
>>
>> Hi Josh. I will answer your question, but could I please ask you to also provide some answer to mine?
>
> Sure:
>> 1. What is the lifespan of the native socket handle?
>
> The native socket handle should exist until the socket streams are released.
>
>> 2. Can I retrieve it once, when the socket is created or opened the first time, then use it until socket is closed, or should I retrieve it from the stream like this, every time I want to use it?
>
> To the best of my knowledge, you should be able to retrieve it once.
>
>> 3. What is the cost of the above lines that retrieve the native socket handle?
>
> It is not terribly expensive, but it’s better to cache the results and trust it won’t go away until the socket streams are closed.
>
> A potentially safer option is to get the native socket handle once and call dup on the socket you get back. Your dupd file descriptor will be yours to close when you see fit. The socket itself won’t be closed until all file descriptors that point to it are closed.
>
>> In the organization I'm working for (as a contractor), we're using an old, (and dare I say wracked) "socket" implementation which maintains its own message queue on a cyclic buffer, from which it draws data and writes it to the NSOutputStream's socket.
>>
>> This socket's design and most of its implementation are taken from an even-older C++ Windows (sigh) implementation over a windows native socket who is opaque and provides no information neither about the size of its buffers, nor of their current contents size.
>>
>> Under those historical windows-imposed considerations, the only way to have reasonable bandwidth measurements (how much was pushed through the network in the last second, and what is the "delay" of data in "our side" of the server connection) is by reducing the size of the socket's internal (opaque) buffers, in comparison with the external socket queue, and do the measurements on OUR queue.
>>
>> When the system socket's internal buffers are small and there is a network bandwidth problem --- we know it early, by failing to write new data into the system socket.
>
> The downside to this approach is that the performance problem could be the small socket buffer sizes. Both your send and receive buffers need to be at least the size of the bandwidth delay product (bandwidth multiplied by the round trip time). On a lower latency network, you won’t see a big performance hit for a small buffer, but when you move to a high latency network, the small buffer could become the limiting factor.
>
>> OF COURSE THIS IS A BAD IMPLEMENTATION.
>>
>> However, I'm a contractor, hired to debug old code, and I can only recommend on enhancing (actually deleting) this old socket code. This is nontrivial because this socket code is doing much more than what I said, and is tightly bound to the specific protocol running between the server and client. It also implements non-standard encryption, nonstandard proxy interactions, and most important -- same socket code runs on server (Windows) and all sorts of clients. In time, I may be called to replace it with better implementation.
>>
>> I know of a good way, using the SO_NWRITE property, to provide good network bandwidth measurements WITHOUT altering the native socket buffer sizes, but I will need to read this property pretty often --- hence my question. What can I do once-per-native-socket and what do I need to do regularly?
>
> Using SO_NWRITE is pretty fast. It’s a trip to the kernel, so I wouldn’t call it every millisecond, but if you’re calling it less often, it shouldn’t show up as a hot spot (but it’s always good to profile the code to confirm).
>
> Getting the send and receive buffer sizes and using SO_NWRITE and SO_NREAD to determine how much data is in the send and receive buffers is a good approach. If you can avoid setting the size of those buffers, you’re likely to see better network performance.
>
> -josh
>
Motti Shneor
----------------------------------------
Ceterum censeo Microsoftinem delendam esse
_______________________________________________
Do not post admin requests to the list. They will be ignored.
Macnetworkprog mailing list (email@hidden)
Help/Unsubscribe/Update your Subscription:
This email sent to email@hidden