about summary refs log tree commit diff stats
diff options
context:
space:
mode:
-rw-r--r--adapter/img/jebp.nim67
-rw-r--r--adapter/img/stbi.nim107
-rw-r--r--adapter/protocol/http.nim18
3 files changed, 131 insertions, 61 deletions
diff --git a/adapter/img/jebp.nim b/adapter/img/jebp.nim
index ec3c9073..03d72d59 100644
--- a/adapter/img/jebp.nim
+++ b/adapter/img/jebp.nim
@@ -1,5 +1,6 @@
 import std/options
 import std/os
+import std/posix
 import std/strutils
 
 import utils/sandbox
@@ -17,6 +18,9 @@ else:
 
 {.passc: "-I" & currentSourcePath().parentDir().}
 
+const STDIN_FILENO = 0
+const STDOUT_FILENO = 1
+
 {.push header: "jebp.h".}
 type
   jebp_io_callbacks {.importc.} = object
@@ -48,13 +52,36 @@ proc jebp_free_image(image: ptr jebp_image_t) {.importc.}
 {.pop.}
 
 proc myRead(data: pointer; size: csize_t; user: pointer): csize_t {.cdecl.} =
-  return csize_t(stdin.readBuffer(data, size))
+  var n = csize_t(0)
+  while n < size:
+    let i = read(STDIN_FILENO, addr cast[ptr UncheckedArray[char]](data)[n],
+      int(size - n))
+    if i == 0:
+      break
+    n += csize_t(i)
+  return n
 
 proc stbir_resize_uint8(input_pixels: ptr uint8;
   input_w, input_h, input_stride_in_bytes: cint; output_pixels: ptr uint8;
   output_w, output_h, output_stride_in_bytes, num_channels: cint): cint
   {.importc.}
 
+proc writeAll(data: pointer; size: int) =
+  var n = 0
+  while n < size:
+    let i = write(STDOUT_FILENO, addr cast[ptr UncheckedArray[uint8]](data)[n],
+      int(size) - n)
+    assert i >= 0
+    n += i
+
+proc puts(s: string) =
+  if s.len > 0:
+    writeAll(unsafeAddr s[0], s.len)
+
+proc die(s: string) {.noreturn.} =
+  puts(s)
+  quit(1)
+
 proc main() =
   enterNetworkSandbox()
   let scheme = getEnv("MAPPED_URI_SCHEME")
@@ -62,8 +89,7 @@ proc main() =
   case getEnv("MAPPED_URI_PATH")
   of "decode":
     if f != "webp":
-      stdout.write("Cha-Control: ConnectionError 1 unknown format " & f)
-      return
+      die("Cha-Control: ConnectionError 1 unknown format " & f)
     let headers = getEnv("REQUEST_HEADERS")
     var targetWidth = cint(-1)
     var targetHeight = cint(-1)
@@ -76,13 +102,11 @@ proc main() =
       of "Cha-Image-Target-Dimensions":
         let s = v.split('x')
         if s.len != 2:
-          stdout.write("Cha-Control: ConnectionError 1 wrong dimensions")
-          return
+          die("Cha-Control: ConnectionError 1 wrong dimensions\n")
         let w = parseUInt32(s[0], allowSign = false)
         let h = parseUInt32(s[1], allowSign = false)
         if w.isNone or w.isNone:
-          stdout.write("Cha-Control: ConnectionError 1 wrong dimensions")
-          return
+          die("Cha-Control: ConnectionError 1 wrong dimensions\n")
         targetWidth = cint(w.get)
         targetHeight = cint(h.get)
     var image = jebp_image_t()
@@ -90,32 +114,33 @@ proc main() =
     if infoOnly:
       let res = jebp_read_size_from_callbacks(addr image, addr cb, nil)
       if res == 0:
-        stdout.write("Cha-Image-Dimensions: " & $image.width & "x" &
+        puts("Cha-Image-Dimensions: " & $image.width & "x" &
           $image.height & "\n\n")
+        quit(0)
       else:
-        stdout.write("Cha-Control: ConnectionError 1 jepb error " &
+        die("Cha-Control: ConnectionError 1 jepb error " &
           $jebp_error_string(res))
-      return
     let res = jebp_read_from_callbacks(addr image, addr cb, nil)
     if res != 0:
-      stdout.write("Cha-Control: ConnectionError 1 jebp error " &
+      die("Cha-Control: ConnectionError 1 jebp error " &
         $jebp_error_string(res))
     elif targetWidth != -1 and targetHeight != -1:
-      let p2 = cast[ptr uint8](alloc(targetWidth * targetHeight * 4))
+      let hdr = "Cha-Image-Dimensions: " & $targetWidth & "x" & $targetHeight &
+        "\n\n"
+      let p2 = cast[ptr UncheckedArray[uint8]](alloc(hdr.len +
+        targetWidth * targetHeight * 4))
+      copyMem(addr p2[0], unsafeAddr hdr[0], hdr.len)
       doAssert stbir_resize_uint8(cast[ptr uint8](image.pixels), image.width,
-        image.height, 0, p2, targetWidth, targetHeight, 0, 4) == 1
-      stdout.write("Cha-Image-Dimensions: " & $targetWidth & "x" &
-        $targetHeight & "\n\n")
-      discard stdout.writeBuffer(p2, targetWidth * targetHeight * 4)
+        image.height, 0, addr p2[hdr.len], targetWidth, targetHeight, 0, 4) == 1
+      writeAll(p2, hdr.len + targetWidth * targetHeight * 4)
       dealloc(p2)
       jebp_free_image(addr image)
     else:
-      stdout.write("Cha-Image-Dimensions: " & $image.width & "x" &
-        $image.height & "\n\n")
-      discard stdout.writeBuffer(cast[ptr uint8](image.pixels), image.width *
-        image.height * 4)
+      puts("Cha-Image-Dimensions: " & $image.width & "x" & $image.height &
+        "\n\n")
+      writeAll(image.pixels, image.width * image.height * 4)
       jebp_free_image(addr image)
   of "encode":
-    stdout.write("Cha-Control: ConnectionError 1 not supported")
+    die("Cha-Control: ConnectionError 1 not supported")
 
 main()
diff --git a/adapter/img/stbi.nim b/adapter/img/stbi.nim
index f4a5faa2..8a1442f2 100644
--- a/adapter/img/stbi.nim
+++ b/adapter/img/stbi.nim
@@ -1,5 +1,6 @@
 import std/options
 import std/os
+import std/posix
 import std/strutils
 
 import utils/sandbox
@@ -30,18 +31,35 @@ proc stbi_image_free(retval_from_stbi_load: pointer) {.importc.}
 
 {.pop.}
 
-proc myRead(user: pointer; data: ptr char; size: cint): cint {.cdecl.} =
-  return cint(stdin.readBuffer(data, size))
+type StbiUser = object
+  atEof: bool
+
+const STDIN_FILENO = 0
+const STDOUT_FILENO = 1
 
-proc mySkip(user: pointer; n: cint) {.cdecl.} =
+proc myRead(user: pointer; data: ptr char; size: cint): cint {.cdecl.} =
+  var n = cint(0)
+  while n < size:
+    let i = read(STDIN_FILENO, addr cast[ptr UncheckedArray[char]](data)[n],
+      int(size - n))
+    if i == 0:
+      cast[ptr StbiUser](user)[].atEof = true
+      break
+    n += cint(i)
+  return n
+
+proc mySkip(user: pointer; size: cint) {.cdecl.} =
   var data: array[4096, uint8]
-  let n = int(n)
-  var i = 0
-  while i < n:
-    i += stdin.readBuffer(addr data[0], min(n - i, data.len))
+  var n = cint(0)
+  while n < size:
+    let i = read(STDIN_FILENO, addr data[0], min(int(size - n), data.len))
+    if i == 0:
+      cast[ptr StbiUser](user)[].atEof = true
+      break
+    n += cint(i)
 
 proc myEof(user: pointer): cint {.cdecl.} =
-  return cint(stdin.endOfFile())
+  return cint(cast[ptr StbiUser](user)[].atEof)
 
 type stbi_write_func = proc(context, data: pointer; size: cint) {.cdecl.}
 
@@ -54,8 +72,16 @@ proc stbi_write_jpg_to_func(fun: stbi_write_func; context: pointer;
   w, h, comp: cint; data: pointer; quality: cint) {.importc.}
 {.pop.}
 
+proc writeAll(data: pointer; size: int) =
+  var n = 0
+  while n < size:
+    let i = write(STDOUT_FILENO, addr cast[ptr UncheckedArray[uint8]](data)[n],
+      int(size) - n)
+    assert i >= 0
+    n += i
+
 proc myWriteFunc(context, data: pointer; size: cint) {.cdecl.} =
-  discard stdout.writeBuffer(data, size)
+  writeAll(data, int(size))
 
 {.push header: "stb_image_resize.h".}
 proc stbir_resize_uint8(input_pixels: ptr uint8;
@@ -64,6 +90,14 @@ proc stbir_resize_uint8(input_pixels: ptr uint8;
   {.importc.}
 {.pop.}
 
+proc puts(s: string) =
+  if s.len > 0:
+    writeAll(unsafeAddr s[0], s.len)
+
+proc die(s: string) {.noreturn.} =
+  puts(s)
+  quit(1)
+
 proc main() =
   enterNetworkSandbox()
   let scheme = getEnv("MAPPED_URI_SCHEME")
@@ -71,8 +105,8 @@ proc main() =
   case getEnv("MAPPED_URI_PATH")
   of "decode":
     if f notin ["jpeg", "gif", "bmp", "png", "x-unknown"]:
-      stdout.write("Cha-Control: ConnectionError 1 unknown format " & f)
-      return
+      die("Cha-Control: ConnectionError 1 unknown format " & f)
+    var user = StbiUser()
     var x: cint
     var y: cint
     var channels_in_file: cint
@@ -93,38 +127,40 @@ proc main() =
       of "Cha-Image-Target-Dimensions":
         let s = v.split('x')
         if s.len != 2:
-          stdout.write("Cha-Control: ConnectionError 1 wrong dimensions")
-          return
+          die("Cha-Control: ConnectionError 1 wrong dimensions\n")
         let w = parseUInt32(s[0], allowSign = false)
         let h = parseUInt32(s[1], allowSign = false)
         if w.isNone or w.isNone:
-          stdout.write("Cha-Control: ConnectionError 1 wrong dimensions")
-          return
+          die("Cha-Control: ConnectionError 1 wrong dimensions\n")
         targetWidth = cint(w.get)
         targetHeight = cint(h.get)
     if infoOnly:
-      if stbi_info_from_callbacks(addr clbk, nil, x, y, channels_in_file) == 1:
-        stdout.write("Cha-Image-Dimensions: " & $x & "x" & $y & "\n\n")
+      if stbi_info_from_callbacks(addr clbk, addr user, x, y,
+          channels_in_file) == 1:
+        puts("Cha-Image-Dimensions: " & $x & "x" & $y & "\n\n")
+        quit(0)
       else:
-        stdout.write("Cha-Control: ConnectionError 1 stbi error " &
+        die("Cha-Control: ConnectionError 1 stbi error " &
           $stbi_failure_reason())
-      return
-    let p = stbi_load_from_callbacks(addr clbk, nil, x, y, channels_in_file, 4)
+    let p = stbi_load_from_callbacks(addr clbk, addr user, x, y,
+      channels_in_file, 4)
     if p == nil:
-      stdout.write("Cha-Control: ConnectionError 1 stbi error " &
+      die("Cha-Control: ConnectionError 1 stbi error " &
         $stbi_failure_reason())
     elif targetWidth != -1 and targetHeight != -1:
-      let p2 = cast[ptr uint8](alloc(targetWidth * targetHeight * 4))
-      doAssert stbir_resize_uint8(p, x, y, 0, p2, targetWidth, targetHeight, 0,
-        4) == 1
-      stdout.write("Cha-Image-Dimensions: " & $targetWidth & "x" &
-        $targetHeight & "\n\n")
-      discard stdout.writeBuffer(p2, targetWidth * targetHeight * 4)
+      let hdr = "Cha-Image-Dimensions: " & $targetWidth & "x" &
+        $targetHeight & "\n\n"
+      let p2 = cast[ptr UncheckedArray[uint8]](alloc(hdr.len +
+        targetWidth * targetHeight * 4))
+      copyMem(addr p2[0], unsafeAddr hdr[0], hdr.len)
+      doAssert stbir_resize_uint8(p, x, y, 0, addr p2[hdr.len], targetWidth,
+        targetHeight, 0, 4) == 1
+      writeAll(p2, hdr.len + targetWidth * targetHeight * 4)
       dealloc(p2)
       stbi_image_free(p)
     else:
-      stdout.write("Cha-Image-Dimensions: " & $x & "x" & $y & "\n\n")
-      discard stdout.writeBuffer(p, x * y * 4)
+      puts("Cha-Image-Dimensions: " & $x & "x" & $y & "\n\n")
+      writeAll(p, x * y * 4)
       stbi_image_free(p)
   of "encode":
     let headers = getEnv("REQUEST_HEADERS")
@@ -138,22 +174,19 @@ proc main() =
         let w = parseUInt32(s[0], allowSign = false)
         let h = parseUInt32(s[1], allowSign = false)
         if w.isNone or w.isNone:
-          stdout.write("Cha-Control: ConnectionError 1 wrong dimensions")
-          return
+          die("Cha-Control: ConnectionError 1 wrong dimensions")
         width = cint(w.get)
         height = cint(h.get)
       of "Cha-Image-Quality":
         let s = hdr.after(':').strip()
         let q = parseUInt32(s, allowSign = false).get(101)
         if q < 1 or 100 < q:
-          stdout.write("Cha-Control: ConnectionError 1 wrong quality")
-          return
+          die("Cha-Control: ConnectionError 1 wrong quality")
         quality = cint(q)
     let s = stdin.readAll()
     if s.len != width * height * 4:
-      stdout.write("Cha-Control: ConnectionError 1 wrong size")
-      return
-    stdout.write("Cha-Image-Dimensions: " & $width & 'x' & $height & "\n\n")
+      die("Cha-Control: ConnectionError 1 wrong size")
+    puts("Cha-Image-Dimensions: " & $width & 'x' & $height & "\n\n")
     let p = unsafeAddr s[0]
     case f
     of "png":
@@ -165,6 +198,6 @@ proc main() =
       stbi_write_jpg_to_func(myWriteFunc, nil, cint(width), cint(height), 4, p,
         quality)
     else:
-      stdout.write("Cha-Control: ConnectionError 1 unknown format " & f)
+      die("Cha-Control: ConnectionError 1 unknown format " & f)
 
 main()
diff --git a/adapter/protocol/http.nim b/adapter/protocol/http.nim
index d00479e4..69542288 100644
--- a/adapter/protocol/http.nim
+++ b/adapter/protocol/http.nim
@@ -23,8 +23,20 @@ type
     earlyhint: EarlyHintState
     slist: curl_slist
 
+const STDIN_FILENO = 0
+const STDOUT_FILENO = 1
+
+proc writeAll(data: pointer; size: int) =
+  var n = 0
+  while n < size:
+    let i = write(STDOUT_FILENO, addr cast[ptr UncheckedArray[uint8]](data)[n],
+      int(size) - n)
+    assert i >= 0
+    n += i
+
 proc puts(s: string) =
-  discard write(1, unsafeAddr s[0], s.len)
+  if s.len > 0:
+    writeAll(unsafeAddr s[0], s.len)
 
 proc curlWriteHeader(p: cstring; size, nitems: csize_t; userdata: pointer):
     csize_t {.cdecl.} =
@@ -63,12 +75,12 @@ proc curlWriteHeader(p: cstring; size, nitems: csize_t; userdata: pointer):
 # From the documentation: size is always 1.
 proc curlWriteBody(p: cstring; size, nmemb: csize_t; userdata: pointer):
     csize_t {.cdecl.} =
-  return csize_t(write(stdout.getFileHandle(), p, int(nmemb)))
+  return csize_t(write(STDOUT_FILENO, p, int(nmemb)))
 
 # From the documentation: size is always 1.
 proc readFromStdin(p: pointer; size, nitems: csize_t; userdata: pointer):
     csize_t {.cdecl.} =
-  return csize_t(read(0, p, int(nitems)))
+  return csize_t(read(STDIN_FILENO, p, int(nitems)))
 
 proc curlPreRequest(clientp: pointer; conn_primary_ip, conn_local_ip: cstring;
     conn_primary_port, conn_local_port: cint): cint {.cdecl.} =