|
@@ -0,0 +1,396 @@
|
|
|
+// Copyright (C) 2010 Internet Systems Consortium, Inc. ("ISC")
|
|
|
+//
|
|
|
+// Permission to use, copy, modify, and/or distribute this software for any
|
|
|
+// purpose with or without fee is hereby granted, provided that the above
|
|
|
+// copyright notice and this permission notice appear in all copies.
|
|
|
+//
|
|
|
+// THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
|
|
|
+// REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
|
+// AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
|
+// INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
|
|
+// LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
|
|
|
+// OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
|
+// PERFORMANCE OF THIS SOFTWARE.
|
|
|
+
|
|
|
+// $Id$
|
|
|
+
|
|
|
+#include <dns/opcode.h>
|
|
|
+
|
|
|
+using namespace isc::dns;
|
|
|
+
|
|
|
+//
|
|
|
+// Declaration of the custom exceptions (None for this class)
|
|
|
+
|
|
|
+//
|
|
|
+// Definition of the classes
|
|
|
+//
|
|
|
+
|
|
|
+// For each class, we need a struct, a helper functions (init, destroy,
|
|
|
+// and static wrappers around the methods we export), a list of methods,
|
|
|
+// and a type description
|
|
|
+
|
|
|
+namespace {
|
|
|
+//
|
|
|
+// Opcode
|
|
|
+//
|
|
|
+class s_Opcode : public PyObject {
|
|
|
+public:
|
|
|
+ s_Opcode() : opcode(NULL), static_code(false) {}
|
|
|
+ const Opcode* opcode;
|
|
|
+ bool static_code;
|
|
|
+};
|
|
|
+
|
|
|
+int Opcode_init(s_Opcode* const self, PyObject* args);
|
|
|
+void Opcode_destroy(s_Opcode* const self);
|
|
|
+
|
|
|
+PyObject* Opcode_getCode(const s_Opcode* const self);
|
|
|
+PyObject* Opcode_toText(const s_Opcode* const self);
|
|
|
+PyObject* Opcode_str(PyObject* const self);
|
|
|
+PyObject* Opcode_QUERY(s_Opcode* self);
|
|
|
+PyObject* Opcode_IQUERY(s_Opcode* self);
|
|
|
+PyObject* Opcode_STATUS(s_Opcode* self);
|
|
|
+PyObject* Opcode_RESERVED3(s_Opcode* self);
|
|
|
+PyObject* Opcode_NOTIFY(s_Opcode* self);
|
|
|
+PyObject* Opcode_UPDATE(s_Opcode* self);
|
|
|
+PyObject* Opcode_RESERVED6(s_Opcode* self);
|
|
|
+PyObject* Opcode_RESERVED7(s_Opcode* self);
|
|
|
+PyObject* Opcode_RESERVED8(s_Opcode* self);
|
|
|
+PyObject* Opcode_RESERVED9(s_Opcode* self);
|
|
|
+PyObject* Opcode_RESERVED10(s_Opcode* self);
|
|
|
+PyObject* Opcode_RESERVED11(s_Opcode* self);
|
|
|
+PyObject* Opcode_RESERVED12(s_Opcode* self);
|
|
|
+PyObject* Opcode_RESERVED13(s_Opcode* self);
|
|
|
+PyObject* Opcode_RESERVED14(s_Opcode* self);
|
|
|
+PyObject* Opcode_RESERVED15(s_Opcode* self);
|
|
|
+PyObject* Opcode_richcmp(const s_Opcode* const self,
|
|
|
+ const s_Opcode* const other, int op);
|
|
|
+
|
|
|
+PyMethodDef Opcode_methods[] = {
|
|
|
+ { "get_code", reinterpret_cast<PyCFunction>(Opcode_getCode), METH_NOARGS,
|
|
|
+ "Returns the code value" },
|
|
|
+ { "to_text", reinterpret_cast<PyCFunction>(Opcode_toText), METH_NOARGS,
|
|
|
+ "Returns the text representation" },
|
|
|
+ { "QUERY", reinterpret_cast<PyCFunction>(Opcode_QUERY),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a QUERY Opcode" },
|
|
|
+ { "IQUERY", reinterpret_cast<PyCFunction>(Opcode_IQUERY),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a IQUERY Opcode" },
|
|
|
+ { "STATUS", reinterpret_cast<PyCFunction>(Opcode_STATUS),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a STATUS Opcode" },
|
|
|
+ { "RESERVED3", reinterpret_cast<PyCFunction>(Opcode_RESERVED3),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a RESERVED3 Opcode" },
|
|
|
+ { "NOTIFY", reinterpret_cast<PyCFunction>(Opcode_NOTIFY),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a NOTIFY Opcode" },
|
|
|
+ { "UPDATE", reinterpret_cast<PyCFunction>(Opcode_UPDATE),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a UPDATE Opcode" },
|
|
|
+ { "RESERVED6", reinterpret_cast<PyCFunction>(Opcode_RESERVED6),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a RESERVED6 Opcode" },
|
|
|
+ { "RESERVED7", reinterpret_cast<PyCFunction>(Opcode_RESERVED7),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a RESERVED7 Opcode" },
|
|
|
+ { "RESERVED8", reinterpret_cast<PyCFunction>(Opcode_RESERVED8),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a RESERVED8 Opcode" },
|
|
|
+ { "RESERVED9", reinterpret_cast<PyCFunction>(Opcode_RESERVED9),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a RESERVED9 Opcode" },
|
|
|
+ { "RESERVED10", reinterpret_cast<PyCFunction>(Opcode_RESERVED10),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a RESERVED10 Opcode" },
|
|
|
+ { "RESERVED11", reinterpret_cast<PyCFunction>(Opcode_RESERVED11),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a RESERVED11 Opcode" },
|
|
|
+ { "RESERVED12", reinterpret_cast<PyCFunction>(Opcode_RESERVED12),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a RESERVED12 Opcode" },
|
|
|
+ { "RESERVED13", reinterpret_cast<PyCFunction>(Opcode_RESERVED13),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a RESERVED13 Opcode" },
|
|
|
+ { "RESERVED14", reinterpret_cast<PyCFunction>(Opcode_RESERVED14),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a RESERVED14 Opcode" },
|
|
|
+ { "RESERVED15", reinterpret_cast<PyCFunction>(Opcode_RESERVED15),
|
|
|
+ METH_NOARGS | METH_STATIC, "Creates a RESERVED15 Opcode" },
|
|
|
+ { NULL, NULL, 0, NULL }
|
|
|
+};
|
|
|
+
|
|
|
+PyTypeObject opcode_type = {
|
|
|
+ PyVarObject_HEAD_INIT(NULL, 0)
|
|
|
+ "pydnspp.Opcode",
|
|
|
+ sizeof(s_Opcode), // tp_basicsize
|
|
|
+ 0, // tp_itemsize
|
|
|
+ (destructor)Opcode_destroy, // tp_dealloc
|
|
|
+ NULL, // tp_print
|
|
|
+ NULL, // tp_getattr
|
|
|
+ NULL, // tp_setattr
|
|
|
+ NULL, // tp_reserved
|
|
|
+ NULL, // tp_repr
|
|
|
+ NULL, // tp_as_number
|
|
|
+ NULL, // tp_as_sequence
|
|
|
+ NULL, // tp_as_mapping
|
|
|
+ NULL, // tp_hash
|
|
|
+ NULL, // tp_call
|
|
|
+ Opcode_str, // tp_str
|
|
|
+ NULL, // tp_getattro
|
|
|
+ NULL, // tp_setattro
|
|
|
+ NULL, // tp_as_buffer
|
|
|
+ Py_TPFLAGS_DEFAULT, // tp_flags
|
|
|
+ "The Opcode class objects represent standard OPCODEs "
|
|
|
+ "of the header section of DNS messages.",
|
|
|
+ NULL, // tp_traverse
|
|
|
+ NULL, // tp_clear
|
|
|
+ (richcmpfunc)Opcode_richcmp, // tp_richcompare
|
|
|
+ 0, // tp_weaklistoffset
|
|
|
+ NULL, // tp_iter
|
|
|
+ NULL, // tp_iternext
|
|
|
+ Opcode_methods, // tp_methods
|
|
|
+ NULL, // tp_members
|
|
|
+ NULL, // tp_getset
|
|
|
+ NULL, // tp_base
|
|
|
+ NULL, // tp_dict
|
|
|
+ NULL, // tp_descr_get
|
|
|
+ NULL, // tp_descr_set
|
|
|
+ 0, // tp_dictoffset
|
|
|
+ (initproc)Opcode_init, // tp_init
|
|
|
+ NULL, // tp_alloc
|
|
|
+ PyType_GenericNew, // tp_new
|
|
|
+ NULL, // tp_free
|
|
|
+ NULL, // tp_is_gc
|
|
|
+ NULL, // tp_bases
|
|
|
+ NULL, // tp_mro
|
|
|
+ NULL, // tp_cache
|
|
|
+ NULL, // tp_subclasses
|
|
|
+ NULL, // tp_weaklist
|
|
|
+ NULL, // tp_del
|
|
|
+ 0 // tp_version_tag
|
|
|
+};
|
|
|
+
|
|
|
+
|
|
|
+int
|
|
|
+Opcode_init(s_Opcode* const self, PyObject* args) {
|
|
|
+ uint16_t code = 0;
|
|
|
+ if (PyArg_ParseTuple(args, "h", &code)) {
|
|
|
+ try {
|
|
|
+ self->opcode = new Opcode(code);
|
|
|
+ self->static_code = false;
|
|
|
+ } catch (const isc::OutOfRange& ex) {
|
|
|
+ PyErr_SetString(PyExc_OverflowError, ex.what());
|
|
|
+ return (-1);
|
|
|
+ } catch (...) {
|
|
|
+ PyErr_SetString(po_IscException, "Unexpected exception");
|
|
|
+ return (-1);
|
|
|
+ }
|
|
|
+ return (0);
|
|
|
+ }
|
|
|
+
|
|
|
+ PyErr_Clear();
|
|
|
+ PyErr_SetString(PyExc_TypeError, "Invalid arguments to Opcode constructor");
|
|
|
+
|
|
|
+ return (-1);
|
|
|
+}
|
|
|
+
|
|
|
+void
|
|
|
+Opcode_destroy(s_Opcode* const self) {
|
|
|
+ // Depending on whether we created the rcode or are referring
|
|
|
+ // to a global static one, we do or do not delete self->opcode here
|
|
|
+ if (!self->static_code) {
|
|
|
+ delete self->opcode;
|
|
|
+ }
|
|
|
+ self->opcode = NULL;
|
|
|
+ Py_TYPE(self)->tp_free(self);
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_getCode(const s_Opcode* const self) {
|
|
|
+ return (Py_BuildValue("I", self->opcode->getCode()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_toText(const s_Opcode* const self) {
|
|
|
+ return (Py_BuildValue("s", self->opcode->toText().c_str()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_str(PyObject* const self) {
|
|
|
+ // Simply call the to_text method we already defined
|
|
|
+ return (PyObject_CallMethod(self,
|
|
|
+ const_cast<char*>("to_text"),
|
|
|
+ const_cast<char*>("")));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_createStatic(const Opcode& opcode) {
|
|
|
+ s_Opcode* ret = PyObject_New(s_Opcode, &opcode_type);
|
|
|
+ if (ret != NULL) {
|
|
|
+ ret->opcode = &opcode;
|
|
|
+ ret->static_code = true;
|
|
|
+ }
|
|
|
+ return (ret);
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_QUERY(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::QUERY()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_IQUERY(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::IQUERY()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_STATUS(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::STATUS()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_RESERVED3(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::RESERVED3()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_NOTIFY(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::NOTIFY()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_UPDATE(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::UPDATE()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_RESERVED6(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::RESERVED6()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_RESERVED7(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::RESERVED7()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_RESERVED8(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::RESERVED8()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_RESERVED9(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::RESERVED9()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_RESERVED10(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::RESERVED10()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_RESERVED11(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::RESERVED11()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_RESERVED12(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::RESERVED12()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_RESERVED13(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::RESERVED13()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_RESERVED14(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::RESERVED14()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_RESERVED15(s_Opcode* self UNUSED_PARAM) {
|
|
|
+ return (Opcode_createStatic(Opcode::RESERVED15()));
|
|
|
+}
|
|
|
+
|
|
|
+PyObject*
|
|
|
+Opcode_richcmp(const s_Opcode* const self, const s_Opcode* const other,
|
|
|
+ const int op)
|
|
|
+{
|
|
|
+ bool c = false;
|
|
|
+
|
|
|
+ // Check for null and if the types match. If different type,
|
|
|
+ // simply return False
|
|
|
+ if (!other || (self->ob_type != other->ob_type)) {
|
|
|
+ Py_RETURN_FALSE;
|
|
|
+ }
|
|
|
+
|
|
|
+ // Only equals and not equals here, unorderable type
|
|
|
+ switch (op) {
|
|
|
+ case Py_LT:
|
|
|
+ PyErr_SetString(PyExc_TypeError, "Unorderable type; Opcode");
|
|
|
+ return (NULL);
|
|
|
+ case Py_LE:
|
|
|
+ PyErr_SetString(PyExc_TypeError, "Unorderable type; Opcode");
|
|
|
+ return (NULL);
|
|
|
+ case Py_EQ:
|
|
|
+ c = (*self->opcode == *other->opcode);
|
|
|
+ break;
|
|
|
+ case Py_NE:
|
|
|
+ c = (*self->opcode != *other->opcode);
|
|
|
+ break;
|
|
|
+ case Py_GT:
|
|
|
+ PyErr_SetString(PyExc_TypeError, "Unorderable type; Opcode");
|
|
|
+ return (NULL);
|
|
|
+ case Py_GE:
|
|
|
+ PyErr_SetString(PyExc_TypeError, "Unorderable type; Opcode");
|
|
|
+ return (NULL);
|
|
|
+ }
|
|
|
+ if (c)
|
|
|
+ Py_RETURN_TRUE;
|
|
|
+ else
|
|
|
+ Py_RETURN_FALSE;
|
|
|
+}
|
|
|
+
|
|
|
+} // end of unnamed namespace
|
|
|
+// end of Opcode
|
|
|
+
|
|
|
+// Module Initialization, all statics are initialized here
|
|
|
+bool
|
|
|
+initModulePart_Opcode(PyObject* mod) {
|
|
|
+ // We initialize the static description object with PyType_Ready(),
|
|
|
+ // then add it to the module. This is not just a check! (leaving
|
|
|
+ // this out results in segmentation faults)
|
|
|
+ if (PyType_Ready(&opcode_type) < 0) {
|
|
|
+ return (false);
|
|
|
+ }
|
|
|
+ Py_INCREF(&opcode_type);
|
|
|
+ void* p = &opcode_type;
|
|
|
+ if (PyModule_AddObject(mod, "Opcode", static_cast<PyObject*>(p)) != 0) {
|
|
|
+ Py_DECREF(&opcode_type);
|
|
|
+ return (false);
|
|
|
+ }
|
|
|
+
|
|
|
+ addClassVariable(opcode_type, "QUERY_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::QUERY_CODE));
|
|
|
+ addClassVariable(opcode_type, "IQUERY_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::IQUERY_CODE));
|
|
|
+ addClassVariable(opcode_type, "STATUS_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::STATUS_CODE));
|
|
|
+ addClassVariable(opcode_type, "RESERVED3_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::RESERVED3_CODE));
|
|
|
+ addClassVariable(opcode_type, "NOTIFY_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::NOTIFY_CODE));
|
|
|
+ addClassVariable(opcode_type, "UPDATE_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::UPDATE_CODE));
|
|
|
+ addClassVariable(opcode_type, "RESERVED6_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::RESERVED6_CODE));
|
|
|
+ addClassVariable(opcode_type, "RESERVED7_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::RESERVED7_CODE));
|
|
|
+ addClassVariable(opcode_type, "RESERVED8_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::RESERVED8_CODE));
|
|
|
+ addClassVariable(opcode_type, "RESERVED9_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::RESERVED9_CODE));
|
|
|
+ addClassVariable(opcode_type, "RESERVED10_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::RESERVED10_CODE));
|
|
|
+ addClassVariable(opcode_type, "RESERVED11_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::RESERVED11_CODE));
|
|
|
+ addClassVariable(opcode_type, "RESERVED12_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::RESERVED12_CODE));
|
|
|
+ addClassVariable(opcode_type, "RESERVED13_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::RESERVED13_CODE));
|
|
|
+ addClassVariable(opcode_type, "RESERVED14_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::RESERVED14_CODE));
|
|
|
+ addClassVariable(opcode_type, "RESERVED15_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::RESERVED15_CODE));
|
|
|
+ addClassVariable(opcode_type, "MAX_CODE",
|
|
|
+ Py_BuildValue("h", Opcode::MAX_CODE));
|
|
|
+
|
|
|
+ return (true);
|
|
|
+}
|