ViewVC Help
View File | Revision Log | Show Annotations | Download File | View Changeset | Root Listing
root/public/ibx/trunk/runtime/IBBlob.pas
(Generate patch)

Comparing ibx/trunk/runtime/IBBlob.pas (file contents):
Revision 1 by tony, Mon Jul 31 16:43:00 2000 UTC vs.
Revision 17 by tony, Sat Dec 28 19:22:24 2013 UTC

# Line 1 | Line 1
1 < {************************************************************************}
2 < {                                                                        }
3 < {       Borland Delphi Visual Component Library                          }
4 < {       InterBase Express core components                                }
5 < {                                                                        }
6 < {       Copyright (c) 1998-2000 Inprise Corporation                      }
7 < {                                                                        }
8 < {    InterBase Express is based in part on the product                   }
9 < {    Free IB Components, written by Gregory H. Deatz for                 }
10 < {    Hoagland, Longo, Moran, Dunst & Doukas Company.                     }
11 < {    Free IB Components is used under license.                           }
12 < {                                                                        }
13 < {    The contents of this file are subject to the InterBase              }
14 < {    Public License Version 1.0 (the "License"); you may not             }
15 < {    use this file except in compliance with the License. You            }
16 < {    may obtain a copy of the License at http://www.Inprise.com/IPL.html }
17 < {    Software distributed under the License is distributed on            }
18 < {    an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either              }
19 < {    express or implied. See the License for the specific language       }
20 < {    governing rights and limitations under the License.                 }
21 < {    The Original Code was created by InterBase Software Corporation     }
22 < {       and its successors.                                              }
23 < {    Portions created by Inprise Corporation are Copyright (C) Inprise   }
24 < {       Corporation. All Rights Reserved.                                }
25 < {    Contributor(s): Jeff Overcash                                       }
26 < {                                                                        }
27 < {************************************************************************}
28 <
29 < unit IBBlob;
30 <
31 < interface
32 <
33 < uses
34 <  Windows, SysUtils, Classes, Forms, IBHeader, IBErrorCodes, IBExternals,
35 <  DB, IB, IBDatabase, IBUtils;
36 <
37 < const
38 <  DefaultBlobSegmentSize = 16 * 1024;
39 <
40 < type
41 <  { TIBBlobStream }
42 <  TIBBlobStream = class(TStream)
43 <  private
44 <    FBase: TIBBase;
45 <    FBlobID: TISC_QUAD;
46 <    FBlobMaxSegmentSize,
47 <    FBlobNumSegments,
48 <    FBlobSize: Long;
49 <    FBlobType: Short;  { 0 = segmented, 1 = streamed }
50 <    FBuffer: PChar;
51 <    FBlobInitialized: Boolean;
52 <    FHandle: TISC_BLOB_HANDLE;
53 <    FMode: TBlobStreamMode;
54 <    FModified: Boolean;
55 <    FPosition: Long;
56 <  protected
57 <    procedure CloseBlob;
58 <    procedure CreateBlob;
59 <    procedure EnsureBlobInitialized;
60 <    procedure GetBlobInfo;
61 <    function GetDatabase: TIBDatabase;
62 <    function GetDBHandle: PISC_DB_HANDLE;
63 <    function GetTransaction: TIBTransaction;
64 <    function GetTRHandle: PISC_TR_HANDLE;
65 <    procedure OpenBlob;
66 <    procedure SetBlobID(Value: TISC_QUAD);
67 <    procedure SetDatabase(Value: TIBDatabase);
68 <    procedure SetMode(Value: TBlobStreamMode);
69 <    procedure SetTransaction(Value: TIBTransaction);
70 <  public
71 <    constructor Create;
72 <    destructor Destroy; override;
73 <    function Call(ErrCode: ISC_STATUS; RaiseError: Boolean): ISC_STATUS;
74 <    procedure CheckReadable;
75 <    procedure CheckWritable;
76 <    procedure Finalize;
77 <    procedure LoadFromFile(Filename: string);
78 <    procedure LoadFromStream(Stream: TStream);
79 <    function Read(var Buffer; Count: Longint): Longint; override;
80 <    procedure SaveToFile(Filename: string);
81 <    procedure SaveToStream(Stream: TStream);
82 <    function Seek(Offset: Longint; Origin: Word): Longint; override;
83 <    procedure SetSize(NewSize: Long); override;
84 <    procedure Truncate;
85 <    function Write(const Buffer; Count: Longint): Longint; override;
86 <    property Handle: TISC_BLOB_HANDLE read FHandle;
87 <    property BlobID: TISC_QUAD read FBlobID write SetBlobID;
88 <    property BlobMaxSegmentSize: Long read FBlobMaxSegmentSize;
89 <    property BlobNumSegments: Long read FBlobNumSegments;
90 <    property BlobSize: Long read FBlobSize;
91 <    property BlobType: Short read FBlobType;
92 <    property Database: TIBDatabase read GetDatabase write SetDatabase;
93 <    property DBHandle: PISC_DB_HANDLE read GetDBHandle;
94 <    property Mode: TBlobStreamMode read FMode write SetMode;
95 <    property Modified: Boolean read FModified;
96 <    property Transaction: TIBTransaction read GetTransaction write SetTransaction;
97 <    property TRHandle: PISC_TR_HANDLE read GetTRHandle;
98 <  end;
99 <
100 <  procedure GetBlobInfo(hBlobHandle: PISC_BLOB_HANDLE; var NumSegments, MaxSegmentSize,
101 <                       TotalSize: Long; var BlobType: Short);
102 <  procedure ReadBlob(hBlobHandle: PISC_BLOB_HANDLE; Buffer: PChar; BlobSize: Long);
103 <  procedure WriteBlob(hBlobHandle: PISC_BLOB_HANDLE; Buffer: PChar; BlobSize: Long);
104 <
105 < implementation
106 <
107 < uses IBIntf, IBCustomDataSet;
108 <
109 < procedure GetBlobInfo(hBlobHandle: PISC_BLOB_HANDLE; var NumSegments, MaxSegmentSize,
110 <                      TotalSize: Long; var BlobType: Short);
111 < var
112 <  items: array[0..3] of Char;
113 <  results: array[0..99] of Char;
114 <  i, item_length: Integer;
115 <  item: Integer;
116 < begin
117 <  items[0] := Char(isc_info_blob_num_segments);
118 <  items[1] := Char(isc_info_blob_max_segment);
119 <  items[2] := Char(isc_info_blob_total_length);
120 <  items[3] := Char(isc_info_blob_type);
121 <
122 <  if isc_blob_info(StatusVector, hBlobHandle, 4, @items[0], SizeOf(results),
123 <                    @results[0]) > 0 then
124 <    IBDatabaseError;
125 <
126 <  i := 0;
127 <  while (i < SizeOf(results)) and (results[i] <> Char(isc_info_end)) do
128 <  begin
129 <    item := Integer(results[i]); Inc(i);
130 <    item_length := isc_vax_integer(@results[i], 2); Inc(i, 2);
131 <    case item of
132 <      isc_info_blob_num_segments:
133 <        NumSegments := isc_vax_integer(@results[i], item_length);
134 <      isc_info_blob_max_segment:
135 <        MaxSegmentSize := isc_vax_integer(@results[i], item_length);
136 <      isc_info_blob_total_length:
137 <        TotalSize := isc_vax_integer(@results[i], item_length);
138 <      isc_info_blob_type:
139 <        BlobType := isc_vax_integer(@results[i], item_length);
140 <    end;
141 <    Inc(i, item_length);
142 <  end;
143 < end;
144 <
145 < procedure ReadBlob(hBlobHandle: PISC_BLOB_HANDLE; Buffer: PChar; BlobSize: Long);
146 < var
147 <  CurPos: Long;
148 <  BytesRead, SegLen: UShort;
149 <  LocalBuffer: PChar;
150 < begin
151 <  CurPos := 0;
152 <  LocalBuffer := Buffer;
153 <  SegLen := UShort(DefaultBlobSegmentSize);
154 <  while (CurPos < BlobSize) do
155 <  begin
156 <    if (CurPos + SegLen > BlobSize) then
157 <      SegLen := BlobSize - CurPos;
158 <    if not ((isc_get_segment(StatusVector, hBlobHandle, @BytesRead, SegLen,
159 <                             LocalBuffer) = 0) or
160 <            (StatusVectorArray[1] = isc_segment)) then
161 <      IBDatabaseError;
162 <    Inc(LocalBuffer, BytesRead);
163 <    Inc(CurPos, BytesRead);
164 <    BytesRead := 0;
165 <  end;
166 < end;
167 <
168 < procedure WriteBlob(hBlobHandle: PISC_BLOB_HANDLE; Buffer: PChar;
169 <  BlobSize: Long);
170 < var
171 <  CurPos, SegLen: Long;
172 < begin
173 <  CurPos := 0;
174 <  SegLen := DefaultBlobSegmentSize;
175 <  while (CurPos < BlobSize) do
176 <  begin
177 <    if (CurPos + SegLen > BlobSize) then
178 <      SegLen := BlobSize - CurPos;
179 <    if isc_put_segment(StatusVector, hBlobHandle, SegLen,
180 <         PChar(@Buffer[CurPos])) > 0 then
181 <      IBDatabaseError;
182 <    Inc(CurPos, SegLen);
183 <  end;
184 < end;
185 <
186 <
187 < { TIBBlobStream }
188 < constructor TIBBlobStream.Create;
189 < begin
190 <  inherited Create;
191 <  FBase := TIBBase.Create(Self);
192 <  FBuffer := nil;
193 <  FBlobSize := 0;
194 < end;
195 <
196 < destructor TIBBlobStream.Destroy;
197 < begin
198 <  if (FHandle <> nil) and
199 <     (Call(isc_close_blob(StatusVector, @FHandle), False) > 0) then
200 <    IBDataBaseError;
201 <  FBase.Free;
202 <  SetSize(0);
203 <  inherited Destroy;
204 < end;
205 <
206 < function TIBBlobStream.Call(ErrCode: ISC_STATUS; RaiseError: Boolean): ISC_STATUS;
207 < begin
208 <  result := 0;
209 <  if Transaction <> nil then
210 <    result := Transaction.Call(ErrCode, RaiseError)
211 <  else if RaiseError and (ErrCode > 0) then
212 <    IBDataBaseError;
213 < end;
214 <
215 < procedure TIBBlobStream.CheckReadable;
216 < begin
217 <  if FMode = bmWrite then IBError(ibxeBlobCannotBeRead, [nil]);
218 < end;
219 <
220 < procedure TIBBlobStream.CheckWritable;
221 < begin
222 <  if FMode = bmRead then IBError(ibxeBlobCannotBeWritten, [nil]);
223 < end;
224 <
225 < procedure TIBBlobStream.CloseBlob;
226 < begin
227 <  Finalize;
228 <  if (FHandle <> nil) and
229 <     (Call(isc_close_blob(StatusVector, @FHandle), False) > 0) then
230 <    IBDataBaseError;
231 < end;
232 <
233 < procedure TIBBlobStream.CreateBlob;
234 < begin
235 <  CheckWritable;
236 <  FBlobID.gds_quad_high := 0;
237 <  FBlobID.gds_quad_low := 0;
238 <  Truncate;
239 < end;
240 <
241 < procedure TIBBlobStream.EnsureBlobInitialized;
242 < begin
243 <  if not FBlobInitialized then
244 <    case FMode of
245 <      bmWrite:
246 <        CreateBlob;
247 <      bmReadWrite: begin
248 <        if (FBlobID.gds_quad_high = 0) and
249 <           (FBlobID.gds_quad_low = 0) then
250 <          CreateBlob
251 <        else
252 <          OpenBlob;
253 <      end;
254 <      else
255 <        OpenBlob;
256 <    end;
257 <  FBlobInitialized := True;
258 < end;
259 <
260 < procedure TIBBlobStream.Finalize;
261 < begin
262 <  if (not FBlobInitialized) or (FMode = bmRead) or (not FModified) then
263 <    exit;
264 <  { need to start writing to a blob, create one }
265 <  Call(isc_create_blob2(StatusVector, DBHandle, TRHandle, @FHandle, @FBlobID,
266 <                       0, nil), True);
267 <  IBBlob.WriteBlob(@FHandle, FBuffer, FBlobSize);
268 <  Call(isc_close_blob(StatusVector, @FHandle), True);
269 <  FModified := False;
270 < end;
271 <
272 < procedure TIBBlobStream.GetBlobInfo;
273 < var
274 <  iBlobSize: Long;
275 < begin
276 <  IBBlob.GetBlobInfo(@FHandle, FBlobNumSegments, FBlobMaxSegmentSize,
277 <    iBlobSize, FBlobType);
278 <  SetSize(iBlobSize);
279 < end;
280 <
281 < function TIBBlobStream.GetDatabase: TIBDatabase;
282 < begin
283 <  result := FBase.Database;
284 < end;
285 <
286 < function TIBBlobStream.GetDBHandle: PISC_DB_HANDLE;
287 < begin
288 <  result := FBase.DBHandle;
289 < end;
290 <
291 < function TIBBlobStream.GetTransaction: TIBTransaction;
292 < begin
293 <  result := FBase.Transaction;
294 < end;
295 <
296 < function TIBBlobStream.GetTRHandle: PISC_TR_HANDLE;
297 < begin
298 <  result := FBase.TRHandle;
299 < end;
300 <
301 < procedure TIBBlobStream.LoadFromFile(Filename: string);
302 < var
303 <  Stream: TStream;
304 < begin
305 <  Stream := TFileStream.Create(FileName, fmOpenRead);
306 <  try
307 <    LoadFromStream(Stream);
308 <  finally
309 <    Stream.Free;
310 <  end;
311 < end;
312 <
313 < procedure TIBBlobStream.LoadFromStream(Stream: TStream);
314 < begin
315 <  CheckWritable;
316 <  EnsureBlobInitialized;
317 <  Stream.Position := 0;
318 <  SetSize(Stream.Size);
319 <  if FBlobSize <> 0 then
320 <    Stream.ReadBuffer(FBuffer^, FBlobSize);
321 <  FModified := True;
322 < end;
323 <
324 < procedure TIBBlobStream.OpenBlob;
325 < begin
326 <  CheckReadable;
327 <  Call(isc_open_blob2(StatusVector, DBHandle, TRHandle, @FHandle,
328 <                     @FBlobID, 0, nil), True);
329 <  try
330 <    GetBlobInfo;
331 <    SetSize(FBlobSize);
332 <    IBBlob.ReadBlob(@FHandle, FBuffer, FBlobSize);
333 <  except
334 <    Call(isc_close_blob(StatusVector, @FHandle), False);
335 <    raise;
336 <  end;
337 <  Call(isc_close_blob(StatusVector, @FHandle), True);
338 < end;
339 <
340 < function TIBBlobStream.Read(var Buffer; Count: Longint): Longint;
341 < begin
342 <  CheckReadable;
343 <  EnsureBlobInitialized;
344 <  if (Count <= 0) then
345 <  begin
346 <    result := 0;
347 <    exit;
348 <  end;
349 <  if (FPosition + Count > FBlobSize) then
350 <    result := FBlobSize - FPosition
351 <  else
352 <    result := Count;
353 <  Move(FBuffer[FPosition], Buffer, result);
354 <  Inc(FPosition, Result);
355 < end;
356 <
357 < procedure TIBBlobStream.SaveToFile(Filename: string);
358 < var
359 <  Stream: TStream;
360 < begin
361 <  Stream := TFileStream.Create(FileName, fmCreate);
362 <  try
363 <    SaveToStream(Stream);
364 <  finally
365 <    Stream.Free;
366 <  end;
367 < end;
368 <
369 < procedure TIBBlobStream.SaveToStream(Stream: TStream);
370 < begin
371 <  CheckReadable;
372 <  EnsureBlobInitialized;
373 <  if FBlobSize <> 0 then
374 <  begin
375 <    Seek(0, soFromBeginning);
376 <    Stream.WriteBuffer(FBuffer^, FBlobSize);
377 <  end;
378 < end;
379 <
380 < function TIBBlobStream.Seek(Offset: Longint; Origin: Word): Longint;
381 < begin
382 <  EnsureBlobInitialized;
383 <  case Origin of
384 <    soFromBeginning     : FPosition := Offset;
385 <    soFromCurrent       : Inc(FPosition, Offset);
386 <    soFromEnd           : FPosition := FBlobSize + Offset;
387 <  end;
388 <  result := FPosition;
389 < end;
390 <
391 < procedure TIBBlobStream.SetBlobID(Value: TISC_QUAD);
392 < begin
393 <  System.Move(Value, FBlobID, SizeOf(TISC_QUAD));
394 <  FBlobInitialized := False;
395 < end;
396 <
397 < procedure TIBBlobStream.SetDatabase(Value: TIBDatabase);
398 < begin
399 <  FBase.Database := Value;
400 <  FBlobInitialized := False;
401 < end;
402 <
403 < procedure TIBBlobStream.SetMode(Value: TBlobStreamMode);
404 < begin
405 <  FMode := Value;
406 <  FBlobInitialized := False;
407 < end;
408 <
409 < procedure TIBBlobStream.SetSize(NewSize: Long);
410 < begin
411 <  if (NewSize <> FBlobSize) then
412 <  begin
413 <    ReallocMem(FBuffer, NewSize);
414 <    FBlobSize := NewSize;
415 <    if NewSize = 0 then
416 <      FBuffer := nil;
417 <  end;
418 < end;
419 <
420 < procedure TIBBlobStream.SetTransaction(Value: TIBTransaction);
421 < begin
422 <  FBase.Transaction := Value;
423 <  FBlobInitialized := False;
424 < end;
425 <
426 < procedure TIBBlobStream.Truncate;
427 < begin
428 <  SetSize(0);
429 < end;
430 <
431 < function TIBBlobStream.Write(const Buffer; Count: Longint): Longint;
432 < begin
433 <  CheckWritable;
434 <  EnsureBlobInitialized;
435 <  result := Count;
436 <  if Count <= 0 then
437 <    exit;
438 <  if (FPosition + Count > FBlobSize) then
439 <    SetSize(FPosition + Count);
440 <  Move(Buffer, FBuffer[FPosition], Count);
441 <  Inc(FPosition, Count);
442 <  FModified := True;
443 < end;
444 <
445 < end.
1 > {************************************************************************}
2 > {                                                                        }
3 > {       Borland Delphi Visual Component Library                          }
4 > {       InterBase Express core components                                }
5 > {                                                                        }
6 > {       Copyright (c) 1998-2000 Inprise Corporation                      }
7 > {                                                                        }
8 > {    InterBase Express is based in part on the product                   }
9 > {    Free IB Components, written by Gregory H. Deatz for                 }
10 > {    Hoagland, Longo, Moran, Dunst & Doukas Company.                     }
11 > {    Free IB Components is used under license.                           }
12 > {                                                                        }
13 > {    The contents of this file are subject to the InterBase              }
14 > {    Public License Version 1.0 (the "License"); you may not             }
15 > {    use this file except in compliance with the License. You            }
16 > {    may obtain a copy of the License at http://www.Inprise.com/IPL.html }
17 > {    Software distributed under the License is distributed on            }
18 > {    an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either              }
19 > {    express or implied. See the License for the specific language       }
20 > {    governing rights and limitations under the License.                 }
21 > {    The Original Code was created by InterBase Software Corporation     }
22 > {       and its successors.                                              }
23 > {    Portions created by Inprise Corporation are Copyright (C) Inprise   }
24 > {       Corporation. All Rights Reserved.                                }
25 > {    Contributor(s): Jeff Overcash                                       }
26 > {                                                                        }
27 > {    IBX For Lazarus (Firebird Express)                                  }
28 > {    Contributor: Tony Whyman, MWA Software http://www.mwasoftware.co.uk }
29 > {    Portions created by MWA Software are copyright McCallum Whyman      }
30 > {    Associates Ltd 2011                                                 }
31 > {************************************************************************}
32 >
33 > unit IBBlob;
34 >
35 > {$Mode Delphi}
36 >
37 > interface
38 >
39 > uses
40 >  SysUtils, Classes, IBHeader, IBErrorCodes, IBExternals, DB, IB, IBDatabase;
41 >
42 >
43 > const
44 >  DefaultBlobSegmentSize = 16 * 1024;
45 >
46 > type
47 >  { TIBBlobStream }
48 >  TIBBlobStream = class(TStream)
49 >  private
50 >    FBase: TIBBase;
51 >    FBlobID: TISC_QUAD;
52 >    FBlobMaxSegmentSize: Int64;
53 >    FBlobNumSegments: Int64;
54 >    FBlobSize: Int64;
55 >    FBlobType: Short;  { 0 = segmented, 1 = streamed }
56 >    FBuffer: PChar;
57 >    FBlobInitialized: Boolean;
58 >    FHandle: TISC_BLOB_HANDLE;
59 >    FMode: TBlobStreamMode;
60 >    FModified: Boolean;
61 >    FPosition: Int64;
62 >  protected
63 >    procedure CloseBlob;
64 >    procedure CreateBlob;
65 >    procedure EnsureBlobInitialized;
66 >    procedure GetBlobInfo;
67 >    function GetDatabase: TIBDatabase;
68 >    function GetDBHandle: PISC_DB_HANDLE;
69 >    function GetTransaction: TIBTransaction;
70 >    function GetTRHandle: PISC_TR_HANDLE;
71 >    procedure OpenBlob;
72 >    procedure SetBlobID(Value: TISC_QUAD);
73 >    procedure SetDatabase(Value: TIBDatabase);
74 >    procedure SetMode(Value: TBlobStreamMode);
75 >    procedure SetTransaction(Value: TIBTransaction);
76 >  public
77 >    constructor Create;
78 >    destructor Destroy; override;
79 >    function Call(ErrCode: ISC_STATUS; RaiseError: Boolean): ISC_STATUS;
80 >    procedure CheckReadable;
81 >    procedure CheckWritable;
82 >    procedure Finalize;
83 >    procedure LoadFromFile(Filename: string);
84 >    procedure LoadFromStream(Stream: TStream);
85 >    function Read(var Buffer; Count: Longint): Longint; override;
86 >    procedure SaveToFile(Filename: string);
87 >    procedure SaveToStream(Stream: TStream);
88 >    function Seek(const Offset: Int64; Origin: TSeekOrigin): Int64; override;
89 >    procedure SetSize(const NewSize: Int64); override;
90 >    procedure SetSize(NewSize: Longint); override;
91 >    procedure Truncate;
92 >    function Write(const Buffer; Count: Longint): Longint; override;
93 >    property Handle: TISC_BLOB_HANDLE read FHandle;
94 >    property BlobID: TISC_QUAD read FBlobID write SetBlobID;
95 >    property BlobMaxSegmentSize: Int64 read FBlobMaxSegmentSize;
96 >    property BlobNumSegments: Int64 read FBlobNumSegments;
97 >    property BlobSize: Int64 read FBlobSize;
98 >    property BlobType: Short read FBlobType;
99 >    property Database: TIBDatabase read GetDatabase write SetDatabase;
100 >    property DBHandle: PISC_DB_HANDLE read GetDBHandle;
101 >    property Mode: TBlobStreamMode read FMode write SetMode;
102 >    property Modified: Boolean read FModified;
103 >    property Transaction: TIBTransaction read GetTransaction write SetTransaction;
104 >    property TRHandle: PISC_TR_HANDLE read GetTRHandle;
105 >  end;
106 >
107 >  procedure GetBlobInfo(hBlobHandle: PISC_BLOB_HANDLE; var NumSegments: Int64; var MaxSegmentSize,
108 >                      TotalSize: Int64; var BlobType: Short);
109 >  procedure ReadBlob(hBlobHandle: PISC_BLOB_HANDLE; Buffer: PChar; BlobSize: Int64);
110 >  procedure WriteBlob(hBlobHandle: PISC_BLOB_HANDLE; Buffer: PChar; BlobSize: Int64);
111 >
112 > implementation
113 >
114 > uses IBIntf;
115 >
116 > procedure GetBlobInfo(hBlobHandle: PISC_BLOB_HANDLE; var NumSegments: Int64; var MaxSegmentSize,
117 >                      TotalSize: Int64; var BlobType: Short);
118 > var
119 >  items: array[0..3] of Char;
120 >  results: array[0..99] of Char;
121 >  i, item_length: Integer;
122 >  item: Integer;
123 > begin
124 >  items[0] := Char(isc_info_blob_num_segments);
125 >  items[1] := Char(isc_info_blob_max_segment);
126 >  items[2] := Char(isc_info_blob_total_length);
127 >  items[3] := Char(isc_info_blob_type);
128 >
129 >  if isc_blob_info(StatusVector, hBlobHandle, 4, @items[0], SizeOf(results),
130 >                    @results[0]) > 0 then
131 >    IBDatabaseError;
132 >
133 >  i := 0;
134 >  while (i < SizeOf(results)) and (results[i] <> Char(isc_info_end)) do
135 >  begin
136 >    item := Integer(results[i]); Inc(i);
137 >    item_length := isc_vax_integer(@results[i], 2); Inc(i, 2);
138 >    case item of
139 >      isc_info_blob_num_segments:
140 >        NumSegments := isc_portable_integer(@results[i], item_length);
141 >      isc_info_blob_max_segment:
142 >        MaxSegmentSize := isc_portable_integer(@results[i], item_length);
143 >      isc_info_blob_total_length:
144 >        TotalSize := isc_portable_integer(@results[i], item_length);
145 >      isc_info_blob_type:
146 >        BlobType := isc_portable_integer(@results[i], item_length);
147 >    end;
148 >    Inc(i, item_length);
149 >  end;
150 > end;
151 >
152 > procedure ReadBlob(hBlobHandle: PISC_BLOB_HANDLE; Buffer: PChar; BlobSize: Int64);
153 > var
154 >  CurPos: Int64;
155 >  BytesRead, SegLen: UShort;
156 >  LocalBuffer: PChar;
157 > begin
158 >  CurPos := 0;
159 >  LocalBuffer := Buffer;
160 >  SegLen := UShort(DefaultBlobSegmentSize);
161 >  while (CurPos < BlobSize) do
162 >  begin
163 >    if (CurPos + SegLen > BlobSize) then
164 >      SegLen := BlobSize - CurPos;
165 >    if not ((isc_get_segment(StatusVector, hBlobHandle, @BytesRead, SegLen,
166 >                             LocalBuffer) = 0) or
167 >            (StatusVectorArray[1] = isc_segment)) then
168 >      IBDatabaseError;
169 >    Inc(LocalBuffer, BytesRead);
170 >    Inc(CurPos, BytesRead);
171 >    BytesRead := 0;
172 >  end;
173 > end;
174 >
175 > procedure WriteBlob(hBlobHandle: PISC_BLOB_HANDLE; Buffer: PChar;
176 >  BlobSize: Int64);
177 > var
178 >  CurPos: Int64;
179 >  SegLen: Long;
180 > begin
181 >  CurPos := 0;
182 >  SegLen := DefaultBlobSegmentSize;
183 >  while (CurPos < BlobSize) do
184 >  begin
185 >    if (CurPos + SegLen > BlobSize) then
186 >      SegLen := BlobSize - CurPos;
187 >    if isc_put_segment(StatusVector, hBlobHandle, SegLen,
188 >         PChar(@Buffer[CurPos])) > 0 then
189 >      IBDatabaseError;
190 >    Inc(CurPos, SegLen);
191 >  end;
192 > end;
193 >
194 >
195 > { TIBBlobStream }
196 > constructor TIBBlobStream.Create;
197 > begin
198 >  inherited Create;
199 >  FBase := TIBBase.Create(Self);
200 >  FBuffer := nil;
201 >  FBlobSize := 0;
202 > end;
203 >
204 > destructor TIBBlobStream.Destroy;
205 > begin
206 >  if (FHandle <> nil) and
207 >     (Call(isc_close_blob(StatusVector, @FHandle), False) > 0) then
208 >    IBDataBaseError;
209 >  FBase.Free;
210 >  SetSize(0);
211 >  inherited Destroy;
212 > end;
213 >
214 > function TIBBlobStream.Call(ErrCode: ISC_STATUS; RaiseError: Boolean): ISC_STATUS;
215 > begin
216 >  result := 0;
217 >  if Transaction <> nil then
218 >    result := Transaction.Call(ErrCode, RaiseError)
219 >  else if RaiseError and (ErrCode > 0) then
220 >    IBDataBaseError;
221 > end;
222 >
223 > procedure TIBBlobStream.CheckReadable;
224 > begin
225 >  if FMode = bmWrite then IBError(ibxeBlobCannotBeRead, [nil]);
226 > end;
227 >
228 > procedure TIBBlobStream.CheckWritable;
229 > begin
230 >  if FMode = bmRead then IBError(ibxeBlobCannotBeWritten, [nil]);
231 > end;
232 >
233 > procedure TIBBlobStream.CloseBlob;
234 > begin
235 >  Finalize;
236 >  if (FHandle <> nil) and
237 >     (Call(isc_close_blob(StatusVector, @FHandle), False) > 0) then
238 >    IBDataBaseError;
239 > end;
240 >
241 > procedure TIBBlobStream.CreateBlob;
242 > begin
243 >  CheckWritable;
244 >  FBlobID.gds_quad_high := 0;
245 >  FBlobID.gds_quad_low := 0;
246 >  Truncate;
247 > end;
248 >
249 > procedure TIBBlobStream.EnsureBlobInitialized;
250 > begin
251 >  if not FBlobInitialized then
252 >    case FMode of
253 >      bmWrite:
254 >        CreateBlob;
255 >      bmReadWrite: begin
256 >        if (FBlobID.gds_quad_high = 0) and
257 >           (FBlobID.gds_quad_low = 0) then
258 >          CreateBlob
259 >        else
260 >          OpenBlob;
261 >      end;
262 >      else
263 >        OpenBlob;
264 >    end;
265 >  FBlobInitialized := True;
266 > end;
267 >
268 > procedure TIBBlobStream.Finalize;
269 > begin
270 >  if (not FBlobInitialized) or (FMode = bmRead) or (not FModified) then
271 >    exit;
272 >  { need to start writing to a blob, create one }
273 >  Call(isc_create_blob2(StatusVector, DBHandle, TRHandle, @FHandle, @FBlobID,
274 >                       0, nil), True);
275 >  IBBlob.WriteBlob(@FHandle, FBuffer, FBlobSize);
276 >  Call(isc_close_blob(StatusVector, @FHandle), True);
277 >  FModified := False;
278 > end;
279 >
280 > procedure TIBBlobStream.GetBlobInfo;
281 > var
282 >  iBlobSize: Int64;
283 > begin
284 >  IBBlob.GetBlobInfo(@FHandle, FBlobNumSegments, FBlobMaxSegmentSize,
285 >    iBlobSize, FBlobType);
286 >  SetSize(iBlobSize);
287 > end;
288 >
289 > function TIBBlobStream.GetDatabase: TIBDatabase;
290 > begin
291 >  result := FBase.Database;
292 > end;
293 >
294 > function TIBBlobStream.GetDBHandle: PISC_DB_HANDLE;
295 > begin
296 >  result := FBase.DBHandle;
297 > end;
298 >
299 > function TIBBlobStream.GetTransaction: TIBTransaction;
300 > begin
301 >  result := FBase.Transaction;
302 > end;
303 >
304 > function TIBBlobStream.GetTRHandle: PISC_TR_HANDLE;
305 > begin
306 >  result := FBase.TRHandle;
307 > end;
308 >
309 > procedure TIBBlobStream.LoadFromFile(Filename: string);
310 > var
311 >  Stream: TStream;
312 > begin
313 >  Stream := TFileStream.Create(FileName, fmOpenRead);
314 >  try
315 >    LoadFromStream(Stream);
316 >  finally
317 >    Stream.Free;
318 >  end;
319 > end;
320 >
321 > procedure TIBBlobStream.LoadFromStream(Stream: TStream);
322 > begin
323 >  CheckWritable;
324 >  EnsureBlobInitialized;
325 >  Stream.Position := 0;
326 >  SetSize(Stream.Size);
327 >  if FBlobSize <> 0 then
328 >    Stream.ReadBuffer(FBuffer^, FBlobSize);
329 >  FModified := True;
330 > end;
331 >
332 > procedure TIBBlobStream.OpenBlob;
333 > begin
334 >  CheckReadable;
335 >  Call(isc_open_blob2(StatusVector, DBHandle, TRHandle, @FHandle,
336 >                     @FBlobID, 0, nil), True);
337 >  try
338 >    GetBlobInfo;
339 >    SetSize(FBlobSize);
340 >    IBBlob.ReadBlob(@FHandle, FBuffer, FBlobSize);
341 >  except
342 >    Call(isc_close_blob(StatusVector, @FHandle), False);
343 >    raise;
344 >  end;
345 >  Call(isc_close_blob(StatusVector, @FHandle), True);
346 > end;
347 >
348 > function TIBBlobStream.Read(var Buffer; Count: Longint): Longint;
349 > begin
350 >  CheckReadable;
351 >  EnsureBlobInitialized;
352 >  if (Count <= 0) then
353 >  begin
354 >    result := 0;
355 >    exit;
356 >  end;
357 >  if (FPosition + Count > FBlobSize) then
358 >    result := FBlobSize - FPosition
359 >  else
360 >    result := Count;
361 >  Move(FBuffer[FPosition], Buffer, result);
362 >  Inc(FPosition, Result);
363 > end;
364 >
365 > procedure TIBBlobStream.SaveToFile(Filename: string);
366 > var
367 >  Stream: TStream;
368 > begin
369 >  Stream := TFileStream.Create(FileName, fmCreate);
370 >  try
371 >    SaveToStream(Stream);
372 >  finally
373 >    Stream.Free;
374 >  end;
375 > end;
376 >
377 > procedure TIBBlobStream.SaveToStream(Stream: TStream);
378 > begin
379 >  CheckReadable;
380 >  EnsureBlobInitialized;
381 >  if FBlobSize <> 0 then
382 >  begin
383 >    Seek(0, soFromBeginning);
384 >    Stream.WriteBuffer(FBuffer^, FBlobSize);
385 >  end;
386 > end;
387 >
388 > function TIBBlobStream.Seek(const Offset: Int64; Origin: TSeekOrigin): Int64;
389 > begin
390 >  EnsureBlobInitialized;
391 >  case Origin of
392 >    soBeginning     : FPosition := Offset;
393 >    soCurrent       : Inc(FPosition, Offset);
394 >    soEnd           : FPosition := FBlobSize + Offset;
395 >  end;
396 >  result := FPosition;
397 > end;
398 >
399 > procedure TIBBlobStream.SetBlobID(Value: TISC_QUAD);
400 > begin
401 >  System.Move(Value, FBlobID, SizeOf(TISC_QUAD));
402 >  FBlobInitialized := False;
403 > end;
404 >
405 > procedure TIBBlobStream.SetDatabase(Value: TIBDatabase);
406 > begin
407 >  FBase.Database := Value;
408 >  FBlobInitialized := False;
409 > end;
410 >
411 > procedure TIBBlobStream.SetMode(Value: TBlobStreamMode);
412 > begin
413 >  FMode := Value;
414 >  FBlobInitialized := False;
415 > end;
416 >
417 > procedure TIBBlobStream.SetSize(const NewSize: Int64);
418 > begin
419 >  if (NewSize <> FBlobSize) then
420 >  begin
421 >    ReallocMem(FBuffer, NewSize);
422 >    FBlobSize := NewSize;
423 >    if NewSize = 0 then
424 >      FBuffer := nil;
425 >  end;
426 > end;
427 >
428 > procedure TIBBlobStream.SetSize(NewSize: Longint);
429 > begin
430 >  SetSize(Int64(NewSize));
431 > end;
432 >
433 > procedure TIBBlobStream.SetTransaction(Value: TIBTransaction);
434 > begin
435 >  FBase.Transaction := Value;
436 >  FBlobInitialized := False;
437 > end;
438 >
439 > procedure TIBBlobStream.Truncate;
440 > begin
441 >  SetSize(0);
442 > end;
443 >
444 > function TIBBlobStream.Write(const Buffer; Count: Longint): Longint;
445 > begin
446 >  CheckWritable;
447 >  EnsureBlobInitialized;
448 >  result := Count;
449 >  if Count <= 0 then
450 >    exit;
451 >  if (FPosition + Count > FBlobSize) then
452 >    SetSize(FPosition + Count);
453 >  Move(Buffer, FBuffer[FPosition], Count);
454 >  Inc(FPosition, Count);
455 >  FModified := True;
456 > end;
457 >
458 > end.

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines