Subversion Repositories Kolibri OS

Rev

Blame | Last modification | View Log | RSS feed

  1. //
  2. // Copyright 2012 Francisco Jerez
  3. //
  4. // Permission is hereby granted, free of charge, to any person obtaining a
  5. // copy of this software and associated documentation files (the "Software"),
  6. // to deal in the Software without restriction, including without limitation
  7. // the rights to use, copy, modify, merge, publish, distribute, sublicense,
  8. // and/or sell copies of the Software, and to permit persons to whom the
  9. // Software is furnished to do so, subject to the following conditions:
  10. //
  11. // The above copyright notice and this permission notice shall be included in
  12. // all copies or substantial portions of the Software.
  13. //
  14. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
  17. // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
  18. // OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
  19. // ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
  20. // OTHER DEALINGS IN THE SOFTWARE.
  21. //
  22.  
  23. #include "core/resource.hpp"
  24. #include "core/memory.hpp"
  25. #include "pipe/p_screen.h"
  26. #include "util/u_sampler.h"
  27. #include "util/u_format.h"
  28.  
  29. using namespace clover;
  30.  
  31. namespace {
  32.    class box {
  33.    public:
  34.       box(const resource::vector &origin, const resource::vector &size) :
  35.          pipe({ (int)origin[0], (int)origin[1],
  36.                 (int)origin[2], (int)size[0],
  37.                 (int)size[1], (int)size[2] }) {
  38.       }
  39.  
  40.       operator const pipe_box *() {
  41.          return &pipe;
  42.       }
  43.  
  44.    protected:
  45.       pipe_box pipe;
  46.    };
  47. }
  48.  
  49. resource::resource(clover::device &dev, memory_obj &obj) :
  50.    device(dev), obj(obj), pipe(NULL), offset() {
  51. }
  52.  
  53. resource::~resource() {
  54. }
  55.  
  56. void
  57. resource::copy(command_queue &q, const vector &origin, const vector &region,
  58.                resource &src_res, const vector &src_origin) {
  59.    auto p = offset + origin;
  60.  
  61.    q.pipe->resource_copy_region(q.pipe, pipe, 0, p[0], p[1], p[2],
  62.                                 src_res.pipe, 0,
  63.                                 box(src_res.offset + src_origin, region));
  64. }
  65.  
  66. void *
  67. resource::add_map(command_queue &q, cl_map_flags flags, bool blocking,
  68.                   const vector &origin, const vector &region) {
  69.    maps.emplace_back(q, *this, flags, blocking, origin, region);
  70.    return maps.back();
  71. }
  72.  
  73. void
  74. resource::del_map(void *p) {
  75.    erase_if([&](const mapping &m) {
  76.          return static_cast<void *>(m) == p;
  77.       }, maps);
  78. }
  79.  
  80. unsigned
  81. resource::map_count() const {
  82.    return maps.size();
  83. }
  84.  
  85. pipe_sampler_view *
  86. resource::bind_sampler_view(command_queue &q) {
  87.    pipe_sampler_view info;
  88.  
  89.    u_sampler_view_default_template(&info, pipe, pipe->format);
  90.    return q.pipe->create_sampler_view(q.pipe, pipe, &info);
  91. }
  92.  
  93. void
  94. resource::unbind_sampler_view(command_queue &q,
  95.                               pipe_sampler_view *st) {
  96.    q.pipe->sampler_view_destroy(q.pipe, st);
  97. }
  98.  
  99. pipe_surface *
  100. resource::bind_surface(command_queue &q, bool rw) {
  101.    pipe_surface info {};
  102.  
  103.    info.format = pipe->format;
  104.    info.writable = rw;
  105.  
  106.    if (pipe->target == PIPE_BUFFER)
  107.       info.u.buf.last_element = pipe->width0 - 1;
  108.  
  109.    return q.pipe->create_surface(q.pipe, pipe, &info);
  110. }
  111.  
  112. void
  113. resource::unbind_surface(command_queue &q, pipe_surface *st) {
  114.    q.pipe->surface_destroy(q.pipe, st);
  115. }
  116.  
  117. root_resource::root_resource(clover::device &dev, memory_obj &obj,
  118.                              command_queue &q, const std::string &data) :
  119.    resource(dev, obj) {
  120.    pipe_resource info {};
  121.  
  122.    if (image *img = dynamic_cast<image *>(&obj)) {
  123.       info.format = translate_format(img->format());
  124.       info.width0 = img->width();
  125.       info.height0 = img->height();
  126.       info.depth0 = img->depth();
  127.    } else {
  128.       info.width0 = obj.size();
  129.       info.height0 = 1;
  130.       info.depth0 = 1;
  131.    }
  132.  
  133.    info.target = translate_target(obj.type());
  134.    info.bind = (PIPE_BIND_SAMPLER_VIEW |
  135.                 PIPE_BIND_COMPUTE_RESOURCE |
  136.                 PIPE_BIND_GLOBAL |
  137.                 PIPE_BIND_TRANSFER_READ |
  138.                 PIPE_BIND_TRANSFER_WRITE);
  139.  
  140.    pipe = dev.pipe->resource_create(dev.pipe, &info);
  141.    if (!pipe)
  142.       throw error(CL_OUT_OF_RESOURCES);
  143.  
  144.    if (!data.empty()) {
  145.       box rect { {{ 0, 0, 0 }}, {{ info.width0, info.height0, info.depth0 }} };
  146.       unsigned cpp = util_format_get_blocksize(info.format);
  147.  
  148.       q.pipe->transfer_inline_write(q.pipe, pipe, 0, PIPE_TRANSFER_WRITE,
  149.                                     rect, data.data(), cpp * info.width0,
  150.                                     cpp * info.width0 * info.height0);
  151.    }
  152. }
  153.  
  154. root_resource::root_resource(clover::device &dev, memory_obj &obj,
  155.                              root_resource &r) :
  156.    resource(dev, obj) {
  157.    assert(0); // XXX -- resource shared among dev and r.dev
  158. }
  159.  
  160. root_resource::~root_resource() {
  161.    device().pipe->resource_destroy(device().pipe, pipe);
  162. }
  163.  
  164. sub_resource::sub_resource(resource &r, const vector &offset) :
  165.    resource(r.device(), r.obj) {
  166.    this->pipe = r.pipe;
  167.    this->offset = r.offset + offset;
  168. }
  169.  
  170. mapping::mapping(command_queue &q, resource &r,
  171.                  cl_map_flags flags, bool blocking,
  172.                  const resource::vector &origin,
  173.                  const resource::vector &region) :
  174.    pctx(q.pipe) {
  175.    unsigned usage = ((flags & CL_MAP_WRITE ? PIPE_TRANSFER_WRITE : 0 ) |
  176.                      (flags & CL_MAP_READ ? PIPE_TRANSFER_READ : 0 ) |
  177.                      (flags & CL_MAP_WRITE_INVALIDATE_REGION ?
  178.                       PIPE_TRANSFER_DISCARD_RANGE : 0) |
  179.                      (!blocking ? PIPE_TRANSFER_UNSYNCHRONIZED : 0));
  180.  
  181.    p = pctx->transfer_map(pctx, r.pipe, 0, usage,
  182.                           box(origin + r.offset, region), &pxfer);
  183.    if (!p) {
  184.       pxfer = NULL;
  185.       throw error(CL_OUT_OF_RESOURCES);
  186.    }
  187. }
  188.  
  189. mapping::mapping(mapping &&m) :
  190.    pctx(m.pctx), pxfer(m.pxfer), p(m.p) {
  191.    m.pctx = NULL;
  192.    m.pxfer = NULL;
  193.    m.p = NULL;
  194. }
  195.  
  196. mapping::~mapping() {
  197.    if (pxfer) {
  198.       pctx->transfer_unmap(pctx, pxfer);
  199.    }
  200. }
  201.  
  202. mapping &
  203. mapping::operator=(mapping m) {
  204.    std::swap(pctx, m.pctx);
  205.    std::swap(pxfer, m.pxfer);
  206.    std::swap(p, m.p);
  207.    return *this;
  208. }
  209.