/*
* %CopyrightBegin%
*
* Copyright Ericsson AB 2010-2011. All Rights Reserved.
*
* The contents of this file are subject to the Erlang Public License,
* Version 1.1, (the "License"); you may not use this file except in
* compliance with the License. You should have received a copy of the
* Erlang Public License along with this software. If not, it can be
* retrieved online at http://www.erlang.org/.
*
* Software distributed under the License is distributed on an "AS IS"
* basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See
* the License for the specific language governing rights and limitations
* under the License.
*
* %CopyrightEnd%
*/
/*
* Purpose: Simple example of NIFs using resource objects to implement functions
* for matrix calculations.
*/
#include "erl_nif.h"
#include <stddef.h>
#include <assert.h>
typedef struct
{
unsigned nrows;
unsigned ncols;
double* data;
} Matrix;
/*
* Use a union for pointer type conversion to avoid compiler warnings
* about strict-aliasing violations with gcc-4.1. gcc >= 4.2 does not
* emit the warning.
* TODO: Reconsider use of union once gcc-4.1 is obsolete?
*/
typedef union
{
void* vp;
Matrix* p;
} mx_t;
#define POS(MX, ROW, COL) ((MX)->data[(ROW)* (MX)->ncols + (COL)])
static int get_number(ErlNifEnv* env, ERL_NIF_TERM term, double* dp);
static Matrix* alloc_matrix(ErlNifEnv* env, unsigned nrows, unsigned ncols);
static void matrix_dtor(ErlNifEnv* env, void* obj);
static ErlNifResourceType* resource_type = NULL;
static int load(ErlNifEnv* env, void** priv_data, ERL_NIF_TERM load_info)
{
ErlNifResourceType* rt = enif_open_resource_type(env, NULL,
"matrix_nif_example",
matrix_dtor,
ERL_NIF_RT_CREATE, NULL);
if (rt == NULL) {
return -1;
}
assert(resource_type == NULL);
resource_type = rt;
return 0;
}
static ERL_NIF_TERM create(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[])
{
/* create(Nrows, Ncolumns, [[first row],[second row],...,[last row]]) -> Matrix */
unsigned nrows, ncols;
unsigned i, j;
ERL_NIF_TERM list, row, ret;
Matrix* mx = NULL;
if (!enif_get_uint(env, argv[0], &nrows) || nrows < 1 ||
!enif_get_uint(env, argv[1], &ncols) || ncols < 1) {
goto badarg;
}
mx = alloc_matrix(env, nrows, ncols);
list = argv[2];
for (i = 0; i<nrows; i++) {
if (!enif_get_list_cell(env, list, &row, &list)) {
goto badarg;
}
for (j = 0; j<ncols; j++) {
ERL_NIF_TERM v;
if (!enif_get_list_cell(env, row, &v, &row) ||
!get_number(env, v, &POS(mx,i,j))) {
goto badarg;
}
}
if (!enif_is_empty_list(env, row)) {
goto badarg;
}
}
if (!enif_is_empty_list(env, list)) {
goto badarg;
}
ret = enif_make_resource(env, mx);
enif_release_resource(mx);
return ret;
badarg:
if (mx != NULL) {
enif_release_resource(mx);
}
return enif_make_badarg(env);
}
static ERL_NIF_TERM pos(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[])
{
/* pos(Matrix, Row, Column) -> float() */
mx_t mx;
unsigned i, j;
if (!enif_get_resource(env, argv[0], resource_type, &mx.vp) ||
!enif_get_uint(env, argv[1], &i) || (--i >= mx.p->nrows) ||
!enif_get_uint(env, argv[2], &j) || (--j >= mx.p->ncols)) {
return enif_make_badarg(env);
}
return enif_make_double(env, POS(mx.p, i,j));
}
static ERL_NIF_TERM add(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[])
{
/* add(Matrix_A, Matrix_B) -> Matrix_Sum */
unsigned i, j;
ERL_NIF_TERM ret;
mx_t mxA, mxB, mxS;
mxA.p = NULL;
mxB.p = NULL;
mxS.p = NULL;
if (!enif_get_resource(env, argv[0], resource_type, &mxA.vp) ||
!enif_get_resource(env, argv[1], resource_type, &mxB.vp) ||
mxA.p->nrows != mxB.p->nrows ||
mxB.p->ncols != mxB.p->ncols) {
return enif_make_badarg(env);
}
mxS.p = alloc_matrix(env, mxA.p->nrows, mxA.p->ncols);
for (i = 0; i < mxA.p->nrows; i++) {
for (j = 0; j < mxA.p->ncols; j++) {
POS(mxS.p, i, j) = POS(mxA.p, i, j) + POS(mxB.p, i, j);
}
}
ret = enif_make_resource(env, mxS.p);
enif_release_resource(mxS.p);
return ret;
}
static ERL_NIF_TERM size_of(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[])
{
/* size(Matrix) -> {Nrows, Ncols} */
mx_t mx;
if (!enif_get_resource(env, argv[0], resource_type, &mx.vp)) {
return enif_make_badarg(env);
}
return enif_make_tuple2(env, enif_make_uint(env, mx.p->nrows),
enif_make_uint(env, mx.p->ncols));
}
static ERL_NIF_TERM to_term(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[])
{
/* to_term(Matrix) -> [[first row], [second row], ...,[last row]] */
unsigned i, j;
ERL_NIF_TERM res;
mx_t mx;
mx.p = NULL;
if (!enif_get_resource(env, argv[0], resource_type, &mx.vp)) {
return enif_make_badarg(env);
}
res = enif_make_list(env, 0);
for (i = mx.p->nrows; i-- > 0; ) {
ERL_NIF_TERM row = enif_make_list(env, 0);
for (j = mx.p->ncols; j-- > 0; ) {
row = enif_make_list_cell(env, enif_make_double(env, POS(mx.p,i,j)),
row);
}
res = enif_make_list_cell(env, row, res);
}
return res;
}
static int get_number(ErlNifEnv* env, ERL_NIF_TERM term, double* dp)
{
long i;
return enif_get_double(env, term, dp) ||
(enif_get_long(env, term, &i) && (*dp=(double)i, 1));
}
static Matrix* alloc_matrix(ErlNifEnv* env, unsigned nrows, unsigned ncols)
{
Matrix* mx = enif_alloc_resource(resource_type, sizeof(Matrix));
mx->nrows = nrows;
mx->ncols = ncols;
mx->data = enif_alloc(nrows*ncols*sizeof(double));
return mx;
}
static void matrix_dtor(ErlNifEnv* env, void* obj)
{
Matrix* mx = (Matrix*) obj;
enif_free(mx->data);
mx->data = NULL;
}
static ErlNifFunc nif_funcs[] =
{
{"create", 3, create},
{"pos", 3, pos},
{"add", 2, add},
{"size_of", 1, size_of},
{"to_term", 1, to_term}
};
ERL_NIF_INIT(matrix_nif,nif_funcs,load,NULL,NULL,NULL);