about summary refs log tree commit diff stats
path: root/src/loader/response.nim
diff options
context:
space:
mode:
authorbptato <nincsnevem662@gmail.com>2024-09-19 17:46:27 +0200
committerbptato <nincsnevem662@gmail.com>2024-09-22 22:44:53 +0200
commit080493c058f52a5c20638f1b975d032af45f4d3f (patch)
tree60e6ba6b3cb967d29d349018b3f315e7637b4b9e /src/loader/response.nim
parente23fa780cf2fff7146efcd64b2806ce428858b80 (diff)
downloadchawan-080493c058f52a5c20638f1b975d032af45f4d3f.tar.gz
loader: mmap intermediate image files, misc refactoring
* refactor parseHeader
* optimize response blob()
* add direct "to cache" mode for loader requests which sets stdout to a
  file, and use it for image processing
* move image resizing into a separate process
* mmap cache files in between processing steps when possible

At last, resize is no longer a part of image decoding. Also, it feels
much nicer to keep encoded image data in the same cache as everything
else.

The mmap operations *should* be more efficient than copying the whole
RGBA data through a pipe. In practice, it only makes a difference for
loading (well, now just mmapping) the encoded image into the pager,
where it singlehandedly speeds up image display by 10x on my test image.

For the other steps, the unfortunate fact that "tocache" must delay the
next fork/exec in the pipeline until the entire image is processed seems
to equal out any wins we might have gotten from skipping a single raw
RGBA copy.

I have tried moving the delay before the exec (it's possible with yet
another pipe), but it didn't help much and made the code much
uglier. (Not that tocache didn't, but I can live with this...)
Diffstat (limited to 'src/loader/response.nim')
-rw-r--r--src/loader/response.nim35
1 files changed, 19 insertions, 16 deletions
diff --git a/src/loader/response.nim b/src/loader/response.nim
index 1d7307cd..23c386ef 100644
--- a/src/loader/response.nim
+++ b/src/loader/response.nim
@@ -47,6 +47,8 @@ type
     opaque*: RootRef
     flags*: set[ResponseFlag]
 
+  FetchPromise* = Promise[JSResult[Response]]
+
 jsDestructor(Response)
 
 proc newResponse*(res: int; request: Request; stream: SocketStream;
@@ -59,6 +61,9 @@ proc newResponse*(res: int; request: Request; stream: SocketStream;
     status: status
   )
 
+proc newFetchPromise*(): FetchPromise =
+  return newPromise[JSResult[Response]]()
+
 func makeNetworkError*(): Response {.jsstfunc: "Response.error".} =
   #TODO use "create" function
   return Response(
@@ -115,6 +120,10 @@ func getReferrerPolicy*(this: Response): Option[ReferrerPolicy] =
   this.headers.table.withValue("Referrer-Policy", p):
     return strictParseEnum[ReferrerPolicy](p[][0])
 
+proc resume*(response: Response) =
+  response.resumeFun(response.outputId)
+  response.resumeFun = nil
+
 type TextOpaque = ref object of RootObj
   buf: string
   bodyRead: Promise[JSResult[string]]
@@ -145,15 +154,9 @@ proc onFinishText(response: Response; success: bool) =
     let err = newTypeError("NetworkError when attempting to fetch resource")
     bodyRead.resolve(JSResult[string].err(err))
 
-proc resume*(response: Response) =
-  response.resumeFun(response.outputId)
-  response.resumeFun = nil
-
 proc text*(response: Response): Promise[JSResult[string]] {.jsfunc.} =
   if response.body == nil:
-    let p = newPromise[JSResult[string]]()
-    p.resolve(JSResult[string].ok(""))
-    return p
+    return newResolvedPromise(JSResult[string].ok(""))
   if response.bodyUsed:
     let p = newPromise[JSResult[string]]()
     let err = JSResult[string]
@@ -179,12 +182,12 @@ proc onReadBlob(response: Response) =
   let opaque = BlobOpaque(response.opaque)
   while true:
     try:
-      let targetLen = opaque.len + BufferSize
-      if targetLen > opaque.size:
-        opaque.size = targetLen
-        opaque.p = realloc(opaque.p, targetLen)
+      if opaque.len + BufferSize > opaque.size:
+        opaque.size *= 2
+        opaque.p = realloc(opaque.p, opaque.size)
       let p = cast[ptr UncheckedArray[uint8]](opaque.p)
-      let n = response.body.recvData(addr p[opaque.len], BufferSize)
+      let diff = opaque.size - opaque.len
+      let n = response.body.recvData(addr p[opaque.len], diff)
       opaque.len += n
       if n == 0:
         break
@@ -211,13 +214,13 @@ proc onFinishBlob(response: Response; success: bool) =
 
 proc blob*(response: Response): Promise[JSResult[Blob]] {.jsfunc.} =
   if response.bodyUsed:
-    let p = newPromise[JSResult[Blob]]()
     let err = JSResult[Blob].err(newTypeError("Body has already been consumed"))
-    p.resolve(err)
-    return p
+    return newResolvedPromise(err)
   let opaque = BlobOpaque(
     bodyRead: newPromise[JSResult[Blob]](),
-    contentType: response.getContentType()
+    contentType: response.getContentType(),
+    p: alloc(BufferSize),
+    size: BufferSize
   )
   response.opaque = opaque
   response.onRead = onReadBlob