[sgen] Add missing memory barrier
[mono.git] / mcs / class / System / System.Net / ResponseStream.cs
1 //
2 // System.Net.ResponseStream
3 //
4 // Author:
5 //      Gonzalo Paniagua Javier (gonzalo@novell.com)
6 //
7 // Copyright (c) 2005 Novell, Inc. (http://www.novell.com)
8 //
9 // Permission is hereby granted, free of charge, to any person obtaining
10 // a copy of this software and associated documentation files (the
11 // "Software"), to deal in the Software without restriction, including
12 // without limitation the rights to use, copy, modify, merge, publish,
13 // distribute, sublicense, and/or sell copies of the Software, and to
14 // permit persons to whom the Software is furnished to do so, subject to
15 // the following conditions:
16 // 
17 // The above copyright notice and this permission notice shall be
18 // included in all copies or substantial portions of the Software.
19 // 
20 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
27 //
28
29 #if SECURITY_DEP
30
31 using System.IO;
32 using System.Net.Sockets;
33 using System.Text;
34 using System.Runtime.InteropServices;
35 namespace System.Net {
36         // FIXME: Does this buffer the response until Close?
37         // Update: we send a single packet for the first non-chunked Write
38         // What happens when we set content-length to X and write X-1 bytes then close?
39         // what if we don't set content-length at all?
40         class ResponseStream : Stream
41         {
42                 HttpListenerResponse response;
43                 bool ignore_errors;
44                 bool disposed;
45                 bool trailer_sent;
46                 Stream stream;
47
48                 internal ResponseStream (Stream stream, HttpListenerResponse response, bool ignore_errors)
49                 {
50                         this.response = response;
51                         this.ignore_errors = ignore_errors;
52                         this.stream = stream;
53                 }
54
55                 public override bool CanRead {
56                         get { return false; }
57                 }
58
59                 public override bool CanSeek {
60                         get { return false; }
61                 }
62
63                 public override bool CanWrite {
64                         get { return true; }
65                 }
66
67                 public override long Length {
68                         get { throw new NotSupportedException (); }
69                 }
70
71                 public override long Position {
72                         get { throw new NotSupportedException (); }
73                         set { throw new NotSupportedException (); }
74                 }
75
76
77                 public override void Close ()
78                 {
79                         if (disposed == false) {
80                                 disposed = true;
81                                 byte [] bytes = null;
82                                 MemoryStream ms = GetHeaders (true);
83                                 bool chunked = response.SendChunked;
84                                 if (ms != null) {
85                                         long start = ms.Position;
86                                         if (chunked && !trailer_sent) {
87                                                 bytes = GetChunkSizeBytes (0, true);
88                                                 ms.Position = ms.Length;
89                                                 ms.Write (bytes, 0, bytes.Length);
90                                         }
91                                         InternalWrite (ms.GetBuffer (), (int) start, (int) (ms.Length - start));
92                                         trailer_sent = true;
93                                 } else if (chunked && !trailer_sent) {
94                                         bytes = GetChunkSizeBytes (0, true);
95                                         InternalWrite (bytes, 0, bytes.Length);
96                                         trailer_sent = true;
97                                 }
98                                 response.Close ();
99                         }
100                 }
101
102                 MemoryStream GetHeaders (bool closing)
103                 {
104                         // SendHeaders works on shared headers
105                         lock (response.headers_lock) {
106                                 if (response.HeadersSent)
107                                         return null;
108                                 MemoryStream ms = new MemoryStream ();
109                                 response.SendHeaders (closing, ms);
110                                 return ms;
111                         }
112                 }
113
114                 public override void Flush ()
115                 {
116                 }
117
118                 static byte [] crlf = new byte [] { 13, 10 };
119                 static byte [] GetChunkSizeBytes (int size, bool final)
120                 {
121                         string str = String.Format ("{0:x}\r\n{1}", size, final ? "\r\n" : "");
122                         return Encoding.ASCII.GetBytes (str);
123                 }
124
125                 internal void InternalWrite (byte [] buffer, int offset, int count)
126                 {
127                         if (ignore_errors) {
128                                 try {
129                                         stream.Write (buffer, offset, count);
130                                 } catch { }
131                         } else {
132                                 stream.Write (buffer, offset, count);
133                         }
134                 }
135
136                 public override void Write (byte [] buffer, int offset, int count)
137                 {
138                         if (disposed)
139                                 throw new ObjectDisposedException (GetType ().ToString ());
140
141                         byte [] bytes = null;
142                         MemoryStream ms = GetHeaders (false);
143                         bool chunked = response.SendChunked;
144                         if (ms != null) {
145                                 long start = ms.Position; // After the possible preamble for the encoding
146                                 ms.Position = ms.Length;
147                                 if (chunked) {
148                                         bytes = GetChunkSizeBytes (count, false);
149                                         ms.Write (bytes, 0, bytes.Length);
150                                 }
151
152                                 int new_count = Math.Min (count, 16384 - (int) ms.Position + (int) start);
153                                 ms.Write (buffer, offset, new_count);
154                                 count -= new_count;
155                                 offset += new_count;
156                                 InternalWrite (ms.GetBuffer (), (int) start, (int) (ms.Length - start));
157                                 ms.SetLength (0);
158                                 ms.Capacity = 0; // 'dispose' the buffer in ms.
159                         } else if (chunked) {
160                                 bytes = GetChunkSizeBytes (count, false);
161                                 InternalWrite (bytes, 0, bytes.Length);
162                         }
163
164                         if (count > 0)
165                                 InternalWrite (buffer, offset, count);
166                         if (chunked)
167                                 InternalWrite (crlf, 0, 2);
168                 }
169
170                 public override IAsyncResult BeginWrite (byte [] buffer, int offset, int count,
171                                                         AsyncCallback cback, object state)
172                 {
173                         if (disposed)
174                                 throw new ObjectDisposedException (GetType ().ToString ());
175
176                         byte [] bytes = null;
177                         MemoryStream ms = GetHeaders (false);
178                         bool chunked = response.SendChunked;
179                         if (ms != null) {
180                                 long start = ms.Position;
181                                 ms.Position = ms.Length;
182                                 if (chunked) {
183                                         bytes = GetChunkSizeBytes (count, false);
184                                         ms.Write (bytes, 0, bytes.Length);
185                                 }
186                                 ms.Write (buffer, offset, count);
187                                 buffer = ms.GetBuffer ();
188                                 offset = (int) start;
189                                 count = (int) (ms.Position - start);
190                         } else if (chunked) {
191                                 bytes = GetChunkSizeBytes (count, false);
192                                 InternalWrite (bytes, 0, bytes.Length);
193                         }
194
195                         return stream.BeginWrite (buffer, offset, count, cback, state);
196                 }
197
198                 public override void EndWrite (IAsyncResult ares)
199                 {
200                         if (disposed)
201                                 throw new ObjectDisposedException (GetType ().ToString ());
202
203                         if (ignore_errors) {
204                                 try {
205                                         stream.EndWrite (ares);
206                                         if (response.SendChunked)
207                                                 stream.Write (crlf, 0, 2);
208                                 } catch { }
209                         } else {
210                                 stream.EndWrite (ares);
211                                 if (response.SendChunked)
212                                         stream.Write (crlf, 0, 2);
213                         }
214                 }
215
216                 public override int Read ([In,Out] byte[] buffer, int offset, int count)
217                 {
218                         throw new NotSupportedException ();
219                 }
220
221                 public override IAsyncResult BeginRead (byte [] buffer, int offset, int count,
222                                                         AsyncCallback cback, object state)
223                 {
224                         throw new NotSupportedException ();
225                 }
226
227                 public override int EndRead (IAsyncResult ares)
228                 {
229                         throw new NotSupportedException ();
230                 }
231
232                 public override long Seek (long offset, SeekOrigin origin)
233                 {
234                         throw new NotSupportedException ();
235                 }
236
237                 public override void SetLength (long value)
238                 {
239                         throw new NotSupportedException ();
240                 }
241         }
242 }
243 #endif
244