From 3b0ddb0509f4226f44948d6a798808529595e035 Mon Sep 17 00:00:00 2001 From: Vadim Kuznetsov Date: Fri, 26 Aug 2022 15:55:54 +0300 Subject: [PATCH] Import transcalc utility from Qucs Qt4 package --- CMakeLists.txt | 1 + qucs-transcalc/CMakeLists.txt | 195 +++ qucs-transcalc/ChangeLog | 91 ++ qucs-transcalc/bitmaps/CMakeLists.txt | 15 + qucs-transcalc/bitmaps/Makefile.am | 40 + qucs-transcalc/bitmaps/big.qucs.xpm | 225 +++ qucs-transcalc/bitmaps/c_microstrip.png | Bin 0 -> 2053 bytes qucs-transcalc/bitmaps/coax.png | Bin 0 -> 2899 bytes qucs-transcalc/bitmaps/cpw.png | Bin 0 -> 2005 bytes qucs-transcalc/bitmaps/cpw_back.png | Bin 0 -> 2059 bytes qucs-transcalc/bitmaps/microstrip.png | Bin 0 -> 1814 bytes qucs-transcalc/bitmaps/rectwaveguide.png | Bin 0 -> 1803 bytes qucs-transcalc/bitmaps/stripline.png | Bin 0 -> 1563 bytes qucs-transcalc/c_microstrip.cpp | 919 +++++++++++++ qucs-transcalc/c_microstrip.h | 102 ++ qucs-transcalc/coax.cpp | 223 +++ qucs-transcalc/coax.h | 61 + qucs-transcalc/coplanar.cpp | 302 +++++ qucs-transcalc/coplanar.h | 67 + qucs-transcalc/examples/CMakeLists.txt | 11 + qucs-transcalc/examples/Makefile.am | 35 + qucs-transcalc/examples/cstrip_kj_1.trc | 21 + qucs-transcalc/examples/cstrip_kj_2.trc | 21 + qucs-transcalc/examples/hp.trc | 21 + qucs-transcalc/examples/ustrip50fr4.trc | 19 + qucs-transcalc/examples/ustrip_jansen.trc | 19 + qucs-transcalc/helpdialog.cpp | 72 + qucs-transcalc/helpdialog.h | 45 + qucs-transcalc/main.cpp | 166 +++ qucs-transcalc/microstrip.cpp | 578 ++++++++ qucs-transcalc/microstrip.h | 93 ++ qucs-transcalc/optionsdialog.cpp | 109 ++ qucs-transcalc/optionsdialog.h | 46 + qucs-transcalc/qucstrans.1.cmake.in | 52 + qucs-transcalc/qucstrans.1.in | 52 + qucs-transcalc/qucstrans.cpp | 1510 +++++++++++++++++++++ qucs-transcalc/qucstrans.h | 200 +++ qucs-transcalc/qucstrans_.qrc | 12 + qucs-transcalc/rectwaveguide.cpp | 352 +++++ qucs-transcalc/rectwaveguide.h | 65 + qucs-transcalc/stripline.cpp | 222 +++ qucs-transcalc/stripline.h | 70 + qucs-transcalc/transline.cpp | 187 +++ qucs-transcalc/transline.h | 60 + qucs-transcalc/units.h | 75 + qucs/qucs_actions.cpp | 6 +- 46 files changed, 6357 insertions(+), 3 deletions(-) create mode 100644 qucs-transcalc/CMakeLists.txt create mode 100644 qucs-transcalc/ChangeLog create mode 100644 qucs-transcalc/bitmaps/CMakeLists.txt create mode 100644 qucs-transcalc/bitmaps/Makefile.am create mode 100644 qucs-transcalc/bitmaps/big.qucs.xpm create mode 100644 qucs-transcalc/bitmaps/c_microstrip.png create mode 100644 qucs-transcalc/bitmaps/coax.png create mode 100644 qucs-transcalc/bitmaps/cpw.png create mode 100644 qucs-transcalc/bitmaps/cpw_back.png create mode 100644 qucs-transcalc/bitmaps/microstrip.png create mode 100644 qucs-transcalc/bitmaps/rectwaveguide.png create mode 100644 qucs-transcalc/bitmaps/stripline.png create mode 100644 qucs-transcalc/c_microstrip.cpp create mode 100644 qucs-transcalc/c_microstrip.h create mode 100644 qucs-transcalc/coax.cpp create mode 100644 qucs-transcalc/coax.h create mode 100644 qucs-transcalc/coplanar.cpp create mode 100644 qucs-transcalc/coplanar.h create mode 100644 qucs-transcalc/examples/CMakeLists.txt create mode 100644 qucs-transcalc/examples/Makefile.am create mode 100644 qucs-transcalc/examples/cstrip_kj_1.trc create mode 100644 qucs-transcalc/examples/cstrip_kj_2.trc create mode 100644 qucs-transcalc/examples/hp.trc create mode 100644 qucs-transcalc/examples/ustrip50fr4.trc create mode 100644 qucs-transcalc/examples/ustrip_jansen.trc create mode 100644 qucs-transcalc/helpdialog.cpp create mode 100644 qucs-transcalc/helpdialog.h create mode 100644 qucs-transcalc/main.cpp create mode 100644 qucs-transcalc/microstrip.cpp create mode 100644 qucs-transcalc/microstrip.h create mode 100644 qucs-transcalc/optionsdialog.cpp create mode 100644 qucs-transcalc/optionsdialog.h create mode 100644 qucs-transcalc/qucstrans.1.cmake.in create mode 100644 qucs-transcalc/qucstrans.1.in create mode 100644 qucs-transcalc/qucstrans.cpp create mode 100644 qucs-transcalc/qucstrans.h create mode 100644 qucs-transcalc/qucstrans_.qrc create mode 100644 qucs-transcalc/rectwaveguide.cpp create mode 100644 qucs-transcalc/rectwaveguide.h create mode 100644 qucs-transcalc/stripline.cpp create mode 100644 qucs-transcalc/stripline.h create mode 100644 qucs-transcalc/transline.cpp create mode 100644 qucs-transcalc/transline.h create mode 100644 qucs-transcalc/units.h diff --git a/CMakeLists.txt b/CMakeLists.txt index 638bdf7e..27adae47 100755 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -37,6 +37,7 @@ ADD_SUBDIRECTORY( qucs-attenuator ) #ADD_SUBDIRECTORY( qucs-doc ) ADD_SUBDIRECTORY( qucs-filter ) ADD_SUBDIRECTORY( qucs-lib ) +ADD_SUBDIRECTORY( qucs-transcalc ) #ADD_SUBDIRECTORY( examples ) ADD_SUBDIRECTORY( translations ) diff --git a/qucs-transcalc/CMakeLists.txt b/qucs-transcalc/CMakeLists.txt new file mode 100644 index 00000000..bcb9cd7e --- /dev/null +++ b/qucs-transcalc/CMakeLists.txt @@ -0,0 +1,195 @@ +PROJECT(qucstrans CXX C) +CMAKE_MINIMUM_REQUIRED(VERSION 3.0) +cmake_policy(VERSION 3.0) + +SET(QUCS_NAME "qucs-s") + +# use top VERSION file +file (STRINGS ${PROJECT_SOURCE_DIR}/../VERSION QUCS_VERSION) +message(STATUS "Configuring ${PROJECT_NAME} (GUI): VERSION ${QUCS_VERSION}") + +set(PROJECT_VERSION "${QUCS_VERSION}") + +set(PROJECT_VENDOR "Qucs team. This program is licensed under the GNU GPL") +set(PROJECT_COPYRIGHT_YEAR "2014") +set(PROJECT_DOMAIN_FIRST "qucs") +set(PROJECT_DOMAIN_SECOND "org") + +SET(CMAKE_BUILD_TYPE Debug) + +ADD_DEFINITIONS( -DHAVE_CONFIG_H ) + +# configure the header config.h +CONFIGURE_FILE ( + "${PROJECT_SOURCE_DIR}/../config.h.cmake" + "${PROJECT_BINARY_DIR}/config.h" +) + +INCLUDE_DIRECTORIES("${PROJECT_BINARY_DIR}") + +SET(CMAKE_POSITION_INDEPENDENT_CODE ON) +SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wall ") # enable warning level +SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++0x ") # enable C++11 + +# flag not available in mingw 4.8.2, MSVC10 +IF(NOT WIN32) + SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wno-deprecated-register ") +ENDIF() + +FIND_PACKAGE( Qt5 COMPONENTS Core Gui Widgets REQUIRED) + + +INCLUDE_DIRECTORIES( + ${Qt5Core_INCLUDE_DIRS} + ${Qt5Widgets_INCLUDE_DIRS} + ) + + # bug, the find package does not seem to set the QT_LIBRARIES, do it manually +SET(QT_LIBRARIES ${Qt5Core_LIBRARIES} ${Qt5Widgets_LIBRARIES} ) + +ADD_DEFINITIONS(${QT_DEFINITIONS}) + +SET(QUCSTRANS_SRCS + helpdialog.cpp + main.cpp + optionsdialog.cpp + qucstrans.cpp +) + +SET(QUCSTRANS_HDRS + c_microstrip.h + coax.h + coplanar.h + microstrip.h + rectwaveguide.h + transline.h + stripline.h + units.h +) + +SET(QUCSTRANS_MOC_HDRS + helpdialog.h + optionsdialog.h + qucstrans.h +) + +QT5_WRAP_CPP( QUCSTRANS_MOC_SRCS ${QUCSTRANS_MOC_HDRS} ) + +SET( LIB_SRC + c_microstrip.cpp + coax.cpp + coplanar.cpp + microstrip.cpp + rectwaveguide.cpp + transline.cpp + stripline.cpp +) + +SET(RESOURCES qucstrans_.qrc) + +QT5_ADD_RESOURCES(RESOURCES_SRCS ${RESOURCES}) + +ADD_LIBRARY(transcalc ${LIB_SRC} ) + +IF(APPLE) + # set information on Info.plist file + SET(MACOSX_BUNDLE_INFO_STRING "${PROJECT_NAME} ${PROJECT_VERSION}") + SET(MACOSX_BUNDLE_BUNDLE_VERSION "${PROJECT_NAME} ${PROJECT_VERSION}") + SET(MACOSX_BUNDLE_LONG_VERSION_STRING "${PROJECT_NAME} ${PROJECT_VERSION}") + SET(MACOSX_BUNDLE_SHORT_VERSION_STRING "${PROJECT_VERSION}") + SET(MACOSX_BUNDLE_COPYRIGHT "${PROJECT_COPYRIGHT_YEAR} ${PROJECT_VENDOR}") + SET(MACOSX_BUNDLE_GUI_IDENTIFIER "${PROJECT_DOMAIN_SECOND}.${PROJECT_DOMAIN_FIRST}") + SET(MACOSX_BUNDLE_BUNDLE_NAME "${PROJECT_NAME}") + SET(MACOSX_BUNDLE_ICON_FILE qucstrans.icns) + + # set where in the bundle to put the icns file + SET_SOURCE_FILES_PROPERTIES(${CMAKE_CURRENT_SOURCE_DIR}/../qucs/bitmaps/qucstrans.icns PROPERTIES MACOSX_PACKAGE_LOCATION Resources) + # include the icns file in the target + SET(QUCSTRANS_SRCS ${QUCSTRANS_SRCS} ${CMAKE_CURRENT_SOURCE_DIR}/../qucs/bitmaps/qucstrans.icns) + +ENDIF(APPLE) + + +ADD_EXECUTABLE(${QUCS_NAME}trans MACOSX_BUNDLE WIN32 + ${QUCSTRANS_SRCS} + ${QUCSTRANS_HDRS} + ${QUCSTRANS_MOC_SRCS} + ${RESOURCES_SRCS} ) + +TARGET_LINK_LIBRARIES( ${QUCS_NAME}trans ${QT_LIBRARIES} transcalc ) + +#INSTALL(TARGETS ${QUCS_NAME}trans DESTINATION bin) + +#ADD_SUBDIRECTORY( bitmaps ) -> added as resources +ADD_SUBDIRECTORY( examples ) + +#INSTALL( FILES qucstrans.1 DESTINATION share/man/man1 ) + +# +# Prepare the installation +# +SET(plugin_dest_dir bin) +SET(qtconf_dest_dir bin) +SET(APPS "${CMAKE_INSTALL_PREFIX}/bin/${PROJECT_NAME}") +IF(APPLE) + SET(plugin_dest_dir ${PROJECT_NAME}.app/Contents/MacOS) + SET(qtconf_dest_dir ${PROJECT_NAME}.app/Contents/Resources) + SET(APPS "${CMAKE_INSTALL_PREFIX}/bin/${PROJECT_NAME}.app") +ENDIF(APPLE) + +IF(WIN32) + SET(APPS "${CMAKE_INSTALL_PREFIX}/bin/${PROJECT_NAME}.exe") +ENDIF(WIN32) + +# +# Install the Qucs application, on Apple, the bundle is at the root of the +# install tree, and on other platforms it'll go into the bin directory. +# +INSTALL(TARGETS ${QUCS_NAME}trans + BUNDLE DESTINATION bin COMPONENT Runtime + RUNTIME DESTINATION bin COMPONENT Runtime + ) + + + +# +# Install needed Qt plugins by copying directories from the qt installation +# One can cull what gets copied by using 'REGEX "..." EXCLUDE' +# +IF(APPLE) + INSTALL(DIRECTORY "${QT_PLUGINS_DIR}/imageformats" DESTINATION bin/${plugin_dest_dir}/plugins COMPONENT Runtime) +ENDIF() +# +# install a qt.conf file +# this inserts some cmake code into the install script to write the file +# +IF(APPLE) +INSTALL(CODE " + file(WRITE \"\${CMAKE_INSTALL_PREFIX}/bin/${qtconf_dest_dir}/qt.conf\" \"\") + " COMPONENT Runtime) +ENDIF() + +#-------------------------------------------------------------------------------- +# Use BundleUtilities to get all other dependencies for the application to work. +# It takes a bundle or executable along with possible plugins and inspects it +# for dependencies. If they are not system dependencies, they are copied. + +# directories to look for dependencies +IF(APPLE) + SET(DIRS ${QT_LIBRARY_DIRS}) +ENDIF() + +# Now the work of copying dependencies into the bundle/package +# The quotes are escaped and variables to use at install time have their $ escaped +# An alternative is the do a configure_file() on a script and use install(SCRIPT ...). +# Note that the image plugins depend on QtSvg and QtXml, and it got those copied +# over. +IF(APPLE) +INSTALL(CODE " + file(GLOB_RECURSE QTPLUGINS + \"\${CMAKE_INSTALL_PREFIX}/bin/${plugin_dest_dir}/plugins/*${CMAKE_SHARED_LIBRARY_SUFFIX}\") + include(BundleUtilities) + fixup_bundle(\"${APPS}\" \"\${QTPLUGINS}\" \"${DIRS}\") + " COMPONENT Runtime) +ENDIF() + diff --git a/qucs-transcalc/ChangeLog b/qucs-transcalc/ChangeLog new file mode 100644 index 00000000..07b9a4c9 --- /dev/null +++ b/qucs-transcalc/ChangeLog @@ -0,0 +1,91 @@ +2011-04-03 Stefan Jahn + + * rectwaveguide.cpp (analyze): Fixed calculation of Z0 in + rectangular waveguide. + +2011-03-02 Stefan Jahn + + * coax.cpp (alphac_coax): Fixed factor of 2 in conductor loss + calculation. + +2009-05-14 Stefan Jahn + + * coax.cpp (show_results): Fixed computation of cut-off + frequencies. + +2008-06-29 Stefan Jahn + + * qucstrans.cpp (slotCopyToClipBoard): Added clipboard copy for + rectangular waveguide schematic. + +2008-03-16 Stefan Jahn + + * qucstrans.cpp (slotCopyToClipBoard): Added clipboard copy for + new coplanar line types. + +2008-02-09 Michael Margraf + + * added new lines: coplanar with and without backside metal + +2006-10-29 Stefan Jahn + + * coax.cpp (synthesize): Fixed synthesis equations for coax + cable. + +2006-08-19 Stefan Jahn + + * units.h (ZF0): Added definition for wave resistance in vacuum. + + * coax.cpp (alphac_coax): Using 20/log(10) instead of 8.686. + + * rectwaveguide.cpp, microstrip.cpp, coax.cpp, c_microstrip.cpp: + Using ZF0 instead of 120*PI wherever possible and necessary. + +2006-01-30 Stefan Jahn + + * qucstrans.cpp (slotCopyToClipBoard): Added supportexport function + for coaxial line into clipboard. + +2005-06-06 Stefan Jahn + + * c_microstrip.cpp (show_results): Added computation of + electrical and physical length. + +2005-04-09 Stefan Jahn + + * qucstrans.cpp (slotCopyToClipBoard): Added schematic creation + for coupled microstrip. + +2005-04-04 Stefan Jahn + + * qucstrans.cpp (slotCopyToClipBoard): Started to implement the + clipboard functionality used to created a schematic part for + Qucs. Added a regex validator for the editable values. + +2005-04-03 Stefan Jahn + + * optionsdialog.h: Added configuration option dialog. + + * helpdialog.h: Implemented help dialog. + + * qucstrans.cpp: Applied reasonable default values for line + properties. Implemented status bar. Improved radio button + behaviour. Implemented config file operations as well as loading + and saving of transmission line files. + + * microstrip.cpp (synthesize): Computing length of microstrip + line based on electrical angle during synthesis. + + * main.cpp: Saving and storing current property values in resource + file on application startup and exit. + + * c_microstrip.cpp (er_eff_freq): Fixed P_9 formula. Use '0.7913' + instead of '0.7193'. + (Z0_dispersion): Fixed Q_19 formula. Use '4.9' instead of '4.19'. + +2005-03-03 Stefan Jahn + + * qucstrans.cpp: Initial attempt to create a 'transcalc' like + GUI for Qt. Thus the program can be ported for use with Qucs. + + diff --git a/qucs-transcalc/bitmaps/CMakeLists.txt b/qucs-transcalc/bitmaps/CMakeLists.txt new file mode 100644 index 00000000..718c3c96 --- /dev/null +++ b/qucs-transcalc/bitmaps/CMakeLists.txt @@ -0,0 +1,15 @@ + +SET(XPMS ) + +SET(PNGS +c_microstrip.png +coax.png +microstrip.png +stripline.png +rectwaveguide.png +cpw.png +cpw_back.png) + +SET(ICON ) + +INSTALL( FILES ${PNGS} DESTINATION share/qucs/bitmaps ) diff --git a/qucs-transcalc/bitmaps/Makefile.am b/qucs-transcalc/bitmaps/Makefile.am new file mode 100644 index 00000000..206f0711 --- /dev/null +++ b/qucs-transcalc/bitmaps/Makefile.am @@ -0,0 +1,40 @@ +## Process this file with automake to produce Makefile.in +# +# qucs-transcalc/bitmaps/Makefile.am +# +# Automake input file. +# +# Copyright (C) 2005, 2008 Stefan Jahn +# +# This is free software; you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation; either version 2, or (at your option) +# any later version. +# +# This software is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this package; see the file COPYING. If not, write to +# the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, +# Boston, MA 02110-1301, USA. +# + +XPMS = + +PNGS = c_microstrip.png coax.png microstrip.png stripline.png rectwaveguide.png cpw.png \ + cpw_back.png \ + big.qucs.xpm + +ICONS = + +EXTRA_DIST = $(XPMS) $(PNGS) $(ICONS) + +# installation of pictures +picdatadir = ${prefix}/share/qucs/bitmaps +picdata_DATA = $(EXTRA_DIST) + +CLEANFILES = *~ +MAINTAINERCLEANFILES = Makefile.in diff --git a/qucs-transcalc/bitmaps/big.qucs.xpm b/qucs-transcalc/bitmaps/big.qucs.xpm new file mode 100644 index 00000000..b69901cf --- /dev/null +++ b/qucs-transcalc/bitmaps/big.qucs.xpm @@ -0,0 +1,225 @@ +/* XPM */ +static char *big_qucs_xpm[] = { +"32 32 190 2", +" c None", +". c #BCA2BC", +"+ c #B4B2D4", +"@ c #BC96BC", +"# c #7C6E9C", +"$ c #ACB2CC", +"% c #9C869C", +"& c #BCA2CC", +"* c #CCC6E4", +"= c #140A1C", +"- c #140A14", +"; c #0C0A14", +"> c #5C5A64", +", c #ACB6C4", +"' c #CCC6D4", +") c #0C0614", +"! c #0C0A1C", +"~ c #1C121C", +"{ c #1C1224", +"] c #0C060C", +"^ c #1C0A1C", +"/ c #2C2234", +"( c #0C0A24", +"_ c #140A24", +": c #1C0E2C", +"< c #240E24", +"[ c #3C323C", +"} c #B49EC4", +"| c #AC9ECC", +"1 c #CCCEE4", +"2 c #84768C", +"3 c #140E24", +"4 c #241634", +"5 c #4C324C", +"6 c #341A3C", +"7 c #645E7C", +"8 c #8472AC", +"9 c #9486BC", +"0 c #D4D6EC", +"a c #4C4664", +"b c #443A64", +"c c #DCEAF4", +"d c #CCBADC", +"e c #1C1634", +"f c #342644", +"g c #7C628C", +"h c #6C567C", +"i c #4C3654", +"j c #2C1634", +"k c #1C0E24", +"l c #645A7C", +"m c #8C7AB4", +"n c #9C92C4", +"o c #D4DEF4", +"p c #ACAECC", +"q c #544A6C", +"r c #A4A2C4", +"s c #E4EEF4", +"t c #CCCAE4", +"u c #DCD2E4", +"v c #540E1C", +"w c #840E14", +"x c #440E1C", +"y c #24264C", +"z c #44365C", +"A c #ACAAC4", +"B c #C4C6DC", +"C c #8C7AAC", +"D c #5C426C", +"E c #3C2A44", +"F c #2C162C", +"G c #A49EB4", +"H c #9492AC", +"I c #8C82A4", +"J c #340E1C", +"K c #7C0E14", +"L c #14122C", +"M c #3C3A64", +"N c #3C3E74", +"O c #ACA6CC", +"P c #D4DEEC", +"Q c #BCB2DC", +"R c #846694", +"S c #5C3E64", +"T c #2C1E3C", +"U c #341E34", +"V c #C4C2CC", +"W c #ECEAEC", +"X c #2C2A4C", +"Y c #242254", +"Z c #1C1A4C", +"` c #7466A4", +" . c #BCBADC", +".. c #CCDEEC", +"+. c #745A84", +"@. c #24162C", +"#. c #3C2A3C", +"$. c #241E2C", +"%. c #1C1E4C", +"&. c #242654", +"*. c #142E64", +"=. c #1C326C", +"-. c #1C265C", +";. c #34366C", +">. c #8476AC", +",. c #CCD2EC", +"'. c #C4CAE4", +"). c #4C4254", +"!. c #44325C", +"~. c #4C2E54", +"{. c #141234", +"]. c #142654", +"^. c #144274", +"/. c #1C3E74", +"(. c #1C2254", +"_. c #1C1E54", +":. c #4C4A7C", +"<. c #A492C4", +"[. c #6C6A7C", +"}. c #64628C", +"|. c #847A8C", +"1. c #644A74", +"2. c #14224C", +"3. c #1C3674", +"4. c #14427C", +"5. c #143E7C", +"6. c #1C3A7C", +"7. c #2C2644", +"8. c #D4E2F4", +"9. c #B4AACC", +"0. c #8C7AA4", +"a. c #140E2C", +"b. c #141634", +"c. c #14366C", +"d. c #1C427C", +"e. c #1C3E7C", +"f. c #141E44", +"g. c #C4CEEC", +"h. c #DCE6EC", +"i. c #C4C6C4", +"j. c #6C5E74", +"k. c #445274", +"l. c #A4B6DC", +"m. c #2C528C", +"n. c #6472AC", +"o. c #14163C", +"p. c #4C4684", +"q. c #BCB6DC", +"r. c #D4CEDC", +"s. c #9C9EA4", +"t. c #B4C2DC", +"u. c #2C365C", +"v. c #5C6A94", +"w. c #8CA2CC", +"x. c #BCBEE4", +"y. c #8C86AC", +"z. c #1C2A64", +"A. c #74769C", +"B. c #CCCADC", +"C. c #C4CEE4", +"D. c #A49EBC", +"E. c #1C162C", +"F. c #8C8AB4", +"G. c #445A8C", +"H. c #A49EC4", +"I. c #B4BAD4", +"J. c #B4B2DC", +"K. c #CCCEEC", +"L. c #CCDAF4", +"M. c #BCC6E4", +"N. c #746A8C", +"O. c #949AB4", +"P. c #9C8EB4", +"Q. c #544E74", +"R. c #342E4C", +"S. c #242A54", +"T. c #6C7AA4", +"U. c #4C5E94", +"V. c #7486B4", +"W. c #9CAAD4", +"X. c #C4D2EC", +"Y. c #D4E2EC", +"Z. c #DCEEF4", +"`. c #D4DAEC", +" + c #C4C6E4", +".+ c #B4AED4", +"++ c #7C7AA4", +"@+ c #9496B4", +"#+ c #9492B4", +"$+ c #8C92AC", +" ", +" ", +" . + ", +" @ # $ ", +" % & * $ ", +" = - ; = = - > , ' ", +" = ; ; = ) = ! ) = ! ", +" ~ ; ) ) ; = ! = ! = { ", +" = ; ] ) ; ) ) = ! = = ^ / ", +" = ; ) ! ) = ! ( ! ! _ : < _ [ } | 1 2 ", +" 3 = ) ! ) ( ( ( ( _ 3 4 5 6 { = 7 8 9 0 a b 0 c d ", +" = ( = = ( = ! _ ! e f g h i j k l m n o p q r s t u ", +" 3 _ v w x ) _ ( _ y z A B C D E j F n r C G H I ", +" _ J w K < _ _ L M b N O P Q R S T E U V V W ", +" _ _ x ^ _ 3 X M Y Z X ` ...| +.@.T #. ", +" $.= _ _ L %.&.*.=.-.Z ;.>.,.'.).!.~. ", +" = ! = 3 {.].^././.*.(._.:.<.[.}.|.1. ", +" > = ! ( L 2.3.4.5.6.*._.Y 7.}.8.9. ", +" 0.@.M 3 = a.b.].c.d.e.3.f.a.&.8 g.h. ", +" i.j.k.l.b ; ( 3 f.*.4.m.n.o.Z %.p.q. ", +" r.s.t.l. = ! 3 u.v.w.x.y.*.z._.A. ", +" B.C.D.t. ! E.x.Q 9.+ F.6.G. ", +" H.I.I. ! J. .K...'. ", +"r.J.A L.M.D.N.O. ", +"P.9. Q.R.%.S.T. ", +" N U.V.W.X. ", +" M.X.Y.c c ", +" c Z.c `. + ", +" B .+n ++@+ ", +" #+$+p A ", +" ", +" "}; diff --git a/qucs-transcalc/bitmaps/c_microstrip.png b/qucs-transcalc/bitmaps/c_microstrip.png new file mode 100644 index 0000000000000000000000000000000000000000..c7d65d5f1642fb37df34b5b1a6386846a40955e3 GIT binary patch literal 2053 zcmb7_d00|eAIHfSQ*koWT+1}4NwEOaXx!Q)MM)>rv`hpStWk%VT)Qq=jb^TiC8)WC zrMccl9IXt?Oj9AXj3|wiTteI$1S=JtOYi$ff4$H1KHqch@B91Q^E~ICbAHdcSAKDG zQdiYig~4FzIA=!!6gi5%UI}`-=YI-g-ojnh|qA32R&m4ri<*|MxC{i{K54*O0=jKttMQeEfZ8M zeNEmNT!ot&URbm5scPnE&%CBU#gKH%4P`cw$~DmPm4bued{CZHQ2OkEY*O+O7z>Fx zN8tYr^BLLj0EU92;IOuDQrQ0{-`TcrL;e@X6SOSzJHw)5N5UEd0~ee6R?}ji}_n@@YY^=p;SI=B>Y*}Rh7_fX`MmL9~z(Uvc+uWk;U%a1WA1O zDKpis2s4gG;11VyJ$XbS`UMCjEggp!92I1ze5Pq^OI&nO=r(-tKHRssV zu+8vA&x5nNav(~xKm1Txh+oxFmhB*cOEGrcJ64+6<#8uZ~A_+E}A(Bb_f^Kk`k>o3S0AyOw^ zdo8;7%UuKJ1Io8jn#>!I*%NXrrl*Ulk5f3Fdm%V}$Gq{czkDCO&DY?aU%cj9+mS(@ z=`kv7YAT7XY(26vSSL}|W(m0m^44^q9;0CJ)e1d^Z0}U_NU89Uu*E3qbd}yEYGjy4 zqwdFWt!glLbi|xYGrafkPGZPNZ+3Xq^CZ@`ya$%WAU_YJ)&4kWq%l3u=Rd>`o7P_M*8Y?mKMwKXpr&-m=EBZLD|>mu$20n`4q9 zaEkAkJHBCV@YSb)4ivl%warz@kQCp3_a$jILB61a_SoQCbdj6J+{v+AIXl!;`MH|8 z^Odm+bR3B0?x{XY2E*xBMy4O{QKhimj?mV}p%cqcaJHS;Ivz?u;%W?>{czGcsI3pN zhP^IS7g)=rr{0LYHoUO#pvv5J+V+hX>oX!JfJy(PAIw!Z#YrTR4;AHLHlM}X!`g+U zyJEIqSU$6ot%2oof^7=Bo6geEkO&6a}>0bLJh!XDv7;j*~^&-rVf$GkLck zC3ZDY&qmL`Ti!H*u{ns84$wD)AhtL6{PGl&-j9Z}KcsyX3O&HRbWV(TtOq4c+dXcy zUuq^4F5PoSI;Gt7&IrlE8V8&8SH2VALfB7RU0OxjAW=|0Utqzh@kaE8HURfKX%yMX z-5%03H)MK>o)KHxwA6<%VTb7*IqL|Ko8C$2kP`*Lm2qsY0mU8U^YwssYak!P6^pD( z%Dm%&d@FAB*&A*z5Dx4Ny>Ze^6i`s+9SP*4Gb-(Y6IO6`ABPF8Dv-;{w`Jez-I6NO zKErhqgbqE5aC^e9{u0vS(>YkUbe}OKz%2)-5j&a%G~yQ4t&taq-rH59pQ#%Uf}K)v z0Aq3VPVKKN0vD;OBUr||1>X0jIeXz~rQY;CPI6%JWl%AAl#?+Tb%VUMV!on+;3bjU z0#O!tJA3&&KRg?*97ZEiSPWt!xc!>7@N2tFGfiM|#j#Of)?g0v9(3v18e;?|J-^wJ~b||uO4#(3U9~XPIPdWhY4Kw|m z!%A$9X-W0ipJn`$d5-_7h7}Nyo=oaWhW5=5ss^KU|7c=SpI@F~6469+#7VSiPR>5% zmt04`Q_xb#l0@$e^xN+Txv{eaeGa(c3JbAe2EJqt50_*v&NS^T7P5)r+2z(5^CIYQFyLVtRH zx@8ul)F#z$^-2zJm36cZcJDW@6Y|@BqxR>i?am0QA8NtWTd*3=7s?85AeZd?HB(M9 zd<;T|z<`2@!;!!Q6OnRZ7h7VdlkHCP&Q-Ce3MhfNWOQlW+UZlPN`+3wlVfl178ZUi zEF1{G~S44Owsc-Rmc)yVW`= z7yQ^tHuTZ7popLLiHr;Yw1S}EQ{rSY$s1(AIT1mr?obS6t=s))IsJD(OPk-Q9N8$*PD`bVJWK= mBIzphE&0y=pS+dyb;Is!cAv+VA8JBBY#0vf=2&-#bn#yxRpGJ# literal 0 HcmV?d00001 diff --git a/qucs-transcalc/bitmaps/coax.png b/qucs-transcalc/bitmaps/coax.png new file mode 100644 index 0000000000000000000000000000000000000000..a0f6c5e4c32a047a8ee548644f0257d203861373 GIT binary patch literal 2899 zcmZ9Oc{J4D8^;mDFolW{es;p7ks>N zV`*$zTNwK?)-aYKS)YykyrR5pY-~qS*AW)1 zvFlIg;b2*_6sA3E06j25BG`KVv;_Mze5@I;$8~EjHa3XKpU(cRx5Sh+$?c6YG2|ZN z5I7DM-b(asVPoT^qY(PHfRUASxYx`l;Z{GJ^j){YhQcoaTW4AFJ-;kuE?sVTHCR; zw&t&MIrS$tk2G-b5cH)WA$lU5R4D-vOLE`@ctTk+Se56mWUbT}O0EmNx3O_4@!66V z1WQ+Z7BraM>M!7#X|_RI3KtKub!z7c_Ul=}K0dh7|Ff{4FC*`l@{f^A=0|WUUshID z9+YcBi?ic0xT9?zrUjP?TO}o^y_gA)?d?5TXQQ2#Npxk$qLsn{!B^E*`HV!1=MnD` zEYN@zY5h2qFEG&XsYF@IIS-~Pea@<_J4>;1;LRW1+MQtG)~5!DFh^3X#eYlZcr)y! zAb_CY_+cUf)6tA9N$})5#lyIa-P79L{a|1IcG24w&nqo%@a#(QAYh^+k1{}PZmAsn z{;Zi4uZ24zF>sTz>2>_>3bVHH^LKm1>^n0M)qe>)Eh7YDlf#0m(m%1N3!yHls9a!}IO@3uc|iTcuqoLdlrBT)1?D4ccU zUw^~8N6`~uZbV_42{qL}piEv-685nOvydI0L@*R&a$x`D5{`W$2LWi2m^D_G)MqTy zRh5*U`|8GE!kS>ll>CuIchjXr8J$~79n+wsrlKyua$U6w7K(8b1K3Fv$@I%rAxD>) zhUV|boisX|oVBk(0j`*&B;!VqV-UucxnW?XDR=@837Bf-4u60%Z76%WJKYV|eeNQQ77jo1Bzd zQ>$Ze>;~G!M6MDd_|;WoCj7^4K!%Xho9t4I!{Ng0O#_6P*#%K{l8m~R>|lqbY$(*d z5;F%V%i?<}lp&g?8lLecSBg)ZG;pE~Fv;tqIk&nLfTf=`XZAckn`ERw4yw!uG|~|B z-zQ#^_1_s9+e`X5Lt8b@SbshPlv@eBvT)W&wVpOaG*5YGq?Rk8m9*yZKKp~hr2WTq z+FgD-3N|F6f}279((&lL?NwHJdZPn>iPd2K)$XRt6M%pK?}$Oqnr@x69(-(UtO1Xb z{3P_M?&-xO5x|E+ywET`P)qER&!`z2^=y^OIcd;e*=g|)(>W&`r9>VePdP~4L_3x6 z_o>lY>C)63h)ybCdmUCB1CXOtec}605nL3}@ZubR9I0yd@l|E*hP#VYQNSbvJ4F+v zR&y;Cz#h3lA8Lx@i;U=&#rQp3Jh)`={qPrf{8}XljEpXay3uJOxQiEeV>qZ| z!m(X*YJ=6f4D!r+HAsvBVfia?9tDlY#DnIUw)1TShnuo>eaO4R@)Xfh`AJFvt*LXn zzO#|YK!=xC*a7OOp*i`3&6`TpkXG2^v?iD))y-|;@X#B)*wRFqZXHg3@^)DcYjRwn zWl@P}+TbEa$-mP+sdK|u2wqq-^IB{aOVNFv^^rbxu)ou)eCYD#e8z8>oiuW5Z7q#7 z$dE%-=v9}Psnn7F(-sv{e^D&yN|k79NieHTfpq19#)_U0EPFqG z-q6+Q*#Z(G+4MnN;xdc@WLr zT)CLe&&m!0{dFculGZ?_2VdvJ@{TH92qEG7FS7CX>bYU}!VB!!Oo1wF-}YKweCq9h zNCof_ERGg8*^sUXTD(8prFvD4^?&h3sFpa9M;z?9GV_Eq+ScEvyJ8+^9EJt zde^x_*_pgW#DHx#1}>ES$-$np%hdus=tPHx(zNEh%l9Tm{_Dl+jkwFHJ&WXjyY;@a zNzXtaK$wWUq{IRPv-fkbud8o;tWCADe7r|I2$M{Bm{!$-u=Q=+j(S^G@eZo<;!A~s2E|2FtI3=ON`cwBUPdZJVYW%HF%Dyg< zSKyefzSX+ZrPt~gd8|)SUM_wZZT5UgOWFDS-K!d-PWFs0Iz>T~y&v2w63=w=w<@g_ zk8MBfcGby7`jWb7uvU7qvykcLx({8f5GZ#JXyzj?8Dk&xy2DRXK0iAGGQiK9Ltvf5jKSjyEQ)kaE+_M-&(BkzQ2XG+KHT<`w_mJO z&;8$B+OvZj@zOA#`8}vP`hDGP#TH+)t<0L1lqdMr%c}gBH^(m&NM70`85k@_D`QNZ z>x~KtboB@v{L^MSk)wv+d29CR&ylA*DLR}~QA>?FWSu0T2CiW;o#WGx>ijfp7D9eU zduv^KwleIUl1k5^t4d1AwM=KKs|JtxaTXu`rN;k3MV2J!vjFlhPyR*A|9AR}mia4t Z+}8?b8!zI|Xt3~?4P|JGC^fhr@?U$8ayI|~ literal 0 HcmV?d00001 diff --git a/qucs-transcalc/bitmaps/cpw.png b/qucs-transcalc/bitmaps/cpw.png new file mode 100644 index 0000000000000000000000000000000000000000..e3bce0b0859873ff72b889e3fde0bee2ea14b69b GIT binary patch literal 2005 zcmZXV2~g8X7RIA92t!~+ARNlT5-um8$s#ubngt{i4974gfb);AAR^)chyjE++?P58 zIpkIlB^=?D5)zOI1tb9!1mp;UQHCp$5QzpM*<`h9$K9^#{;FSnuU}VpSN}3RF(;H2 z_bY-xAZ3)Bv$s^4vaYaOI(ppY1xaO3Ao_$eNCIjG*{~(=)ufRhF1z_gf>hxw1e+{ z!;8NU0iGWllK)m>dM@?5S>%J3DSt{uam(`^pTN+LW~f#9LRQI3luo}`ocgDiyh7jB z1xe$N_8FD;4<--b95i<6XREj-BamA9*$M$N-A7jla8@u#-bYVYah0jepkVN=AsGRc z8onC88gQK?=lkdQaFicLpZVCSFgQ%t!U8q}X+4~Sta2L@wTIdfgC4%T)W-Tkpa$_1&pe(^!vqzTvHBSO-FEY3=M-U&ZRW)wf0 z+J+AQ7Xy_S_w1dr$2$p53lfLw&+E5l9PiKT$gbIr6+Nw8I9@pCrO%2#-**jP0|__} z%|r{55Zp>+A-=dZg7z3!`p%f#Tn~FV@!pfhd8D$q-q6B^!vdHj#vxBlsJOiG7_Rac zr9;snwe}3gWAgQ?1&g-Z42)L~TXHHBx@p$A%uzPMiP&cTB{`SOPjV~YkaJpIipUH( z?alfvul|<6Joy7<&qp0$$xn_>k|(A%AjjIEGo^Dr-ECW)#k%cBU}*dAg`{4qZ;rlu z`oqenqJpQFw!Cn;&FWZv(@Dk8JEFX<1R~Q2xk-d|zZX@lU3+TfE~Yx;gdbXWN{_bO zWt~-Z8C_>m@M276c%kMDVl&h9kU!YTq_6f`35^%FI@hvrn_i?&nbR*rt4gnGaan{g zPC?-Qp8X2ltT1rl{V?*uPQvRnBi`cBg9cmG7s5nQZmvg%>(z*j!NEiaJG+QzM>A%# zA!^j(=HB(wy1&!1a}B+SXT_`>lS&4#VV3)W^N3^i0kEa`^dIPH6+}3aKb8v&Lq!_Z z3|SbiY^=nKupzIm$3*2wwtrfmLtVF7aRkfr60uMGsa^R~7sOnmB)gS`+zar$*{|V* zwpZv&MJc+}`3D^xMZ_2D>r`Uh-T%y^b5PJwSY^)a+)7ExcZ8nWw-CKsZ@MiR~wirb3dYKP4C_O6dUu5Wr3*wba%hk~FCk8>0qF z0B^b%orI8XBNdAc=-glc5riqdU-NA_<0xEKeL;rx`<}3{<^CK~fSb}3sdUi?Lugaq z11DI%9e35D&X1QAtZhvz66E--rtR3x(IA~knw+;ig`M1ETA`-a*PbJ&pS&1{B3zlP zm7`SC=jABT)cvyZeuC-4(()_xuq;`514_rAq}UiE$>LbeQHMeN*tRe_t#22lA|`w` zHS~3CUlZ%>#`9STKvt+No3nzyTT<_d98Uyz|!5Cm`6{{c>|TS%Wb2ku)(JH%@HU zrLJ^#=JIV208|y?a^?Ej2X?2pq##W^(rTw#>s8y5iRa#zd?>lS>o;XNZ zxNOr676=59d~Q-?z>7W~W~R?_k-^Q%h_({zp~U6-$*mSh*Z8f;Rh{u$C$3r%MAFl` zC)n`Qq6%*mbkpbC^r%N6dnmMpca;bNR}H?N+!HoJh%sS}K(y`P`>yn%U)t`JI<}W0 zS&s0l85a^TcxPQ*U7Ygg>B}}zbTqrSHRReU`MPx)J}L{meZ@?Sx4hp-<~74Em}{pG za14r@Xun0(QtK3pF)n_z){;bAR9K8Jq@@4_msXJhoUSsSC4$k zaYQrxE2F+3=>Hh?1!gifeZ#0PFq1*-e`$BHLM$JfYk2zn_vUU=0E19280QA$nZ$ns D6M5P- literal 0 HcmV?d00001 diff --git a/qucs-transcalc/bitmaps/cpw_back.png b/qucs-transcalc/bitmaps/cpw_back.png new file mode 100644 index 0000000000000000000000000000000000000000..b3d64f63fadd7127a4de991db2565278ad2cae97 GIT binary patch literal 2059 zcmZWq4LH;L9+!JbB2=1}=pvoGZl~hldMSBrQC=6#YD!Lcn)fsB9X9g1gBb3Sp2)Dv zSg~qe;E$RLq$=xpQ{#Jm4R|N)YDC2TfXjg6Ut?oU9nyv2?`2oKJwVX9B%cM zC3hxz`?&A?sHnMjhl-}+R;(;j$7N}M%T|;GL-xzjXy_kQ0{$;_(^t^rc&E<9_8Cwo)lEcZP@AEhN z=ypd+3hrqdUs1oHgn_u~Xwg)V7>F@Oj&3R@7%ihBO%D|lR~XUP!{qC(UWZ|jkda~|=Pw%H6tzVxlJ+RwPe7VNjWpc%(xTGM2Nzb@IedpW= z546{BTSA*}(`ijrrUo!W_R*aN-|@^FMKhy+A?5Bwrg~x_tIg`{jQU_qGxn(UD-GI1 zoGLPt&wR;eg(p-}n_l*`z(c90B_;K)RlNa`R8$}>*t8<;uC?127vzua9t54xvylt? zA|fsYIWuFcS)LJuCeEF|MZ1@-TT$m{)Ik@dla?3nl%mLI`SsI5CXvnJpbF5{cLW~P zV5A_(Fb3@k-DuLB6TykBYebSli~88o=f?Fm$@0X?qgua^8qF%x{EYt$^Vg|ocJg^^ zHBb&E!|b2#Aq6la3zGJpyh7M|ZodIot7SVtnmhuE=rvhEinvfr1P({tE{k-=OX2st za!*SoLu~gB4RQFP#*YLMD~NRok!ayjWY3vWzWZWD@IzXBHK^u zd&}^*YZ+Fg;M+7TFLNNFbn+ig}+ux0Y+C#m7%CreCOx@Gfh(u6Kz2 z#8EM==WLnVW|dT2U2m-JKjAKaQozYJLa`g`ynkjqY0z ziI}$`Xep%SUyiupggn4G^UL>cI>5)YRuR)u4IyX|FwJBT%DwNS18_P^@rFo2_6J*p z9!A!O4e&&D91mOhxqoP@iOkR?`O-ocyU%`^G6a$mX>Q#PEA-wsAeHEL0^1#O)3hM48L3 zkD?Px zr{f6FCE0=^wr)>Oo|v*@7(TnO_ssssco<5Jn6 z{Ro?GDIgdP&rGC6n=xql`J(e`2);*>Bttlde==G7T9Gthd7TfyHvhiRN9-BD_!v+t z!ryXp_XE4CK4%Ci3jPa`Y`gz;li@OMyQw3bAdA6&AfJMQqvKYduH2 z!M#5fJd)=_s9#t(bhiWGF={o&ld>LF^To(_%lr0aEvD8LPh2YF1R*SLLXN$#D7iA9 zS+_i&7??o7*&WyeMB#q&FpkgKzCr>kJ%i@7%Va)$=ppToA1DkdV~AXJs%No~V?(=< zS1a+kCK*hLegrthh*20d9E=(MKMIKU!V>(d7b+@fcy4noE24%o;9T^(%1~E(S5CwE zl(7TJ$_>%7<0SvZRwM4L5%H_$8S~wqVhm=$-ZL9bBYMGY_p z`vyS)trUQ0v?eJZn=ylJhxq?+acK=t@Rjk}Pj|@aRXh|JI1U3at8ZO&?%qIKVWqJ+ z__5C7kw|?CNeRIy%ln4`LL~<;EY#9e24no%fv@o5E0mVNg?qKk3bNH-Z!k|-9n>}| zsxsa6tGWH=UH@Zle{{kBn%kdT@OPVRqHJm;x*NaPa@4scvuy=$cVD+Q*UQ=e2EtYa AcmMzZ literal 0 HcmV?d00001 diff --git a/qucs-transcalc/bitmaps/microstrip.png b/qucs-transcalc/bitmaps/microstrip.png new file mode 100644 index 0000000000000000000000000000000000000000..423a63ed58d70e3f6f4e707828ba2ef6645d5e3f GIT binary patch literal 1814 zcmZ{l2~d++630i)m#>i4R;U%g-7 zI~8u-$bdCG|PH9yP^HMQM= zD%;WDTN$JbYTon@^wAvJX#iC>u>0Ywr^lo?xxM+IW|7D1oEkl`kqu>h9N?et-@JhoSBpy!QQdbhG@Ilo6tQ&&{X}xR z$$hsMUR()B&<6KEykp?2W$6V+WE=RBQ7YBKkx>?82vZB^t!1gyOjQCI^4|$wa8-gT zvFC{!)rLYNDmrOqAQUGF!qjtDcXGmm6BO<>7FB<_n7HL@VZnZTeZ8BX=hq*;AbU14)JC=R2;sDyq9w?{G z?Vj2MaE!{R=YHfvOY<(Fz}oF#)8sC-1eh}M}V>yn3|`NH^{2TvvJxL08HiLqsz9C>o<vt`gk4>qA5TcA zHNw3}NT)sCbWIZMj}k_(~ODli$$KQp@i^Etffy~(MR_<8ddcgMn* z`h>3RRapV5Ct{2B9p2R086#o7JAdPZs84k6_0tvW_ltY=3>6`m_4wXPVK|wtA?GcU zPUP3+p~6JiKQ}LGB7?!srccNE1>A`CC!Y!zPDrdv*sC2cZh^^Vi%q`gnA3lcRY*p* zo$XewH!4kk4;7D`9>Rijqr1ETISL$}4&`+YW1wzGlb`fcsmvK}1VSi1ehsUB4o)`$ zi;Jz~J8DIjvV-8ykkP zzxQXnGMww7h)vo2_J&5v>guj#%GE&9BJaxcM2?VKdv!Fy6}WnQ@5l600B1nE`ZHJO zaXkPXb1nX6$D2vfEXNC66Y7A6FEfJP^~D$vd^8A~wiV^x1H2}JPrGXG`eH>C<`m}v zR$qi+OLTPO4rBO&nYxjfT6u)ywCy2jKlPKtU-~t;;f!$bE#wC5k|`-y)^E@ z!!4j6>RY|CJSO@!xyZFTrOy}_ofVcJQ%cn2thtypa55NOc3^m-8D2cM!xCKtL0-YV z;*Q5oZGC+(m0n=gc@#;`jlgPIHqi}y6A1XFSx$%bhXVvecb4z>w$r^?7|Q&L7kqds z4MpqH#YIzL%#Nc->SHTs3+HBtuP}Slvk%nUI(f4`R8bK2Ns*6i9 zX%H>VgNW|7s-aR45m0w1QC?hNi)F4!V5y^e)z-MOMH^thbn!}x!^4v^1y-=4J<7k| zCEMJk1OfA6{XbaH_lTe(gQa1^gq1~q{gF5iv9V8|%V96L9RU4Lxc`%jDyaV(?q8;W kivC}~`9H?Q=^DA_RGH%_+M}XUIlB#O0*YA0JU@(pMFFj#X(F;n!WMQ# z0rgP`q|IE4nbLY-IGryB`)YDa6 zRYz4pK|!71=Ikw3Xg%PH^4%*}7$6trbMCIr3Tx}@hsM&|@{G!NZoc6P3J8z&*wEcs z;VDnTt`a;B!3GsIwJ_MD^@Y#mg*gQ0{XUf8g;zlMh#MwGbkat-@?mvMifBvPc`=$UaH9hl!xT*yI&NLzBsvaDM>fMr&Tu2SCaxRk$p1GS zz^ofS8^17odDB0J_VLp%sb#Gge*{p!#3;gOBwW$6xsI@EH70nY;v}s8TszP=yHQ*^ za#4e2?J+KO3z(Y%s4`pK)E}3D?iv!$&zK;N*M(m2;?kvuu&vOGR1!Az~rY_2WlpIxQk zGV1hYZ4i;PC@E{L`q|k$pw?hzVsaYvR%01wTHmSD&wZcE;@ay8dPQYq<}DJ-(^3Z3D7MBOE4xj`I7CZI=hgqtaXE8VgQ0G?BC9K%9^or?)BY;n+2q?W1o@x%DD)B;XWx!PctI3={3i1_*PsB|bZ{NN+~GT&i)DV(jBL)=42wke!7F~O5ZHI&CUSZCF$uoZXNWdWZo?rkKX{d zj|pAF?I&J^xCF`zFwJ zP*$Ix-(Kn!7Io<+T3g#uq-8O11Ko1t#LdXc_BbY!`@Da1z1Ozox?dGRP%Qs*^#tz< zn4{iey9LBHW1PaaIDzu6?#;&KI}NJL{Iat9Ou@o-hcuy9^1FmBI>^>{8~&=H`z@Rgk4fHOtJnxztI>U{};Hf>XGu>hV#`S z%wH^PL`g`!G5C%gnUZ@1SkarYW#n+ng=tq^K}x2G5=HI)AX@InWqpc(ht5m{geE7s zu}lT`>!@v2{m5u(0Cv-+;M&WDFLT-le+zkX!@l5fY4s<|s&^0UU9NDYp?9;7`NWg9 z9@-un^0=b=sv!Q_{cN=t=;Q}Id-PWv3p>8iY<_^p8@Qy9V3ns~R8h;YGQ7G2%Ha@c z_%qF5_xZj=&PUHZaQa4AGDl~18U4ik;ye~fjv?7o%2GAY=MKy{Sl2sQw2`E76_+-q z2`B)ysW?g*NB6DNsoY{k>MPo`zbRiSf^ZJ;FS36I_rJ*g8Qkkc|B~!~kAQXb|3mh$ cnzap&UppYjA}8kMV@rW>$kVyTyI~Rzntr zLdoNuab5_vN<#VzYcy0i*GnSgP_ zxTR&8Wg#?27onx|KGJD;g#K9T9ViS>A^Q#nnYNyL7jWa3t_|{wrvskzs~4i8HY$Ag z#%9!6X8jO(v2blQ2$CYC>4{RUuzLdk5hGEUdpyp&bdoDzFYLAt8}^?&65qAeH?{GWAvLmxJF&15 zXVborD1Ohul1kb)O(qtuZj!xbHecR*Jsq=d!*PYy-UaVtP?~k~CkiG>mcJ2*j)N;X zjuznhs;XV0bl$STT~hgy&=&r=_^XZ%u5f5q()#jE4PuoIMW9Ne(>-DI{n|%xp9%$7 zC%<18owiC`Sg*|jTsb0_Gt?4~!=)K{d7@dr2Umczf&p8iUK6YE@S1_&=vy9OEV4NX zMzmccYVU^@cHeK8<-YxuubEdf0Ml)>v$0 zPZ4x4XH}s=y>vl3zZ7nfMmlYf>Q}7fAYIV90^ML|Ze0{^Yr3}iqc z>P5d>92HE^;|Cb*?clCTT@W94UMZxqiI_gV!OhOe5zfuo45g$tenW|3CduzeWwBUG zB$^ZRw&7s9;@sRERhZPPLN?V5xVS%|JfQqlqgHoS;}tXbW%1>`91+H?S&U4TlfNG; zN#eUyA+`;+hz8A4(0n_nzrSC~*YW!xp9`_EukXv-XwZtSTytrVCAl{|QG3=xi{@mS z6jf7G%em-k-h*aQ20d(OLp$?-O+)}PlvK;h=m|kd~ zP71ZOw5L}W?h?5$fTR<~B5EGsnI=#*g)BmvsaUyrmtsI1s1D{XoC}n@biNH7y-ESv z;EALV9XIAB(`VwEME1C<9&+z%TE{bYwKGTuW?=n1d|cog<({QQb{cUu*7@j}ajXuQ zi=Ni3Pv9CitqfAz8&-V+JACplG;DsI$&jMA*N;>LZx#-#j+#7+51{aNu=8S={F#H~ zm^^=5IAysul2Bs+^#O#{CgaY1CNf}IA8B=~AXqN;?2rjEv~z`O1fqFs>-m#<;M&bm zWTAO%7bd@SvR0u8V2`Yx)HJRbi0H6h+14{o_eT+S8SQ8*Ao1hfco6p35MzM#bZf>R`KA zb{fP!)Fr0lAiB2^T&&IQQSO}%nOwZ)k_kA^CE)KqZ_RyQCi&Ear0e8Y!n%<~-L}xN zrCRr8x3kG~A8~Y81|r}6|8_yq!H~*Pm?D!()wBzg^egXtu)zQL;^%hx!O=dq%MZ1m d?6Rx>{npnoj}ZQ(^B!XCD7>QwuEODT%HLOG11kUk literal 0 HcmV?d00001 diff --git a/qucs-transcalc/c_microstrip.cpp b/qucs-transcalc/c_microstrip.cpp new file mode 100644 index 00000000..c39eab46 --- /dev/null +++ b/qucs-transcalc/c_microstrip.cpp @@ -0,0 +1,919 @@ +/* + * c_microstrip.cpp - coupled microstrip class implementation + * + * Copyright (C) 2002 Claudio Girardi + * Copyright (C) 2005, 2006 Stefan Jahn + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or (at + * your option) any later version. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + +/* c_microstrip.c - Puts up window for coupled microstrips and + * performs the associated calculations + * Based on the original microstrip.c by Gopal Narayanan + */ + +#include +#include +#include +#include + +#include "units.h" +#include "transline.h" +#include "microstrip.h" +#include "c_microstrip.h" + +c_microstrip::c_microstrip() : transline() +{ + aux_ms = NULL; +} + +c_microstrip::~c_microstrip() +{ + if (aux_ms) delete aux_ms; +} + +/* + * delta_u_thickness_single() computes the thickness effect on + * normalized width for a single microstrip line + * + * References: H. A. Atwater, "Simplified Design Equations for + * Microstrip Line Parameters", Microwave Journal, pp. 109-115, + * November 1989. + */ +double c_microstrip::delta_u_thickness_single(double u, double t_h) +{ + double delta_u; + + if (t_h > 0.0) { + delta_u = (1.25 * t_h / pi) * (1.0 + log((2.0 + (4.0 * pi * u - 2.0) / (1.0 + exp(-100.0 * (u - 1.0 / (2.0 * pi))))) / t_h)); + } else { + delta_u = 0.0; + } + return delta_u; +} + +/* + * delta_u_thickness() - compute the thickness effect on normalized + * width for coupled microstrips + * + * References: Rolf Jansen, "High-Speed Cmputation of Single and + * Coupled Microstrip Parameters Including Dispersion, High-Order + * Modes, Loss and Finite Strip Thickness", IEEE Trans. MTT, vol. 26, + * no. 2, pp. 75-82, Feb. 1978 + */ +void c_microstrip::delta_u_thickness() +{ + double e_r, u, g, t_h; + double delta_u, delta_t, delta_u_e, delta_u_o; + + e_r = er; + u = w / h; /* normalized line width */ + g = s / h; /* normalized line spacing */ + t_h = t / h; /* normalized strip thickness */ + + if (t_h > 0.0) { + /* single microstrip correction for finite strip thickness */ + delta_u = delta_u_thickness_single(u, t_h); + delta_t = t_h / (g * e_r); + /* thickness correction for the even- and odd-mode */ + delta_u_e = delta_u * (1.0 - 0.5 * exp(-0.69 * delta_u / delta_t)); + delta_u_o = delta_u_e + delta_t; + } else { + delta_u_e = delta_u_o = 0.0; + } + + w_t_e = w + delta_u_e * h; + w_t_o = w + delta_u_o * h; +} + +/* + * compute various parameters for a single line + */ +void c_microstrip::compute_single_line() +{ + if (aux_ms == NULL) + aux_ms = new microstrip (); + + /* prepare parameters for single microstrip computations */ + aux_ms->er = er; + aux_ms->w = w; + aux_ms->h = h; + aux_ms->t = 0.0; + //aux_ms->t = t; + aux_ms->ht = 1e12; /* arbitrarily high */ + aux_ms->f = f; + aux_ms->mur = mur; + aux_ms->microstrip_Z0(); + aux_ms->dispersion(); +} + + +/* + * filling_factor_even() - compute the filling factor for the coupled + * microstrips even-mode without cover and zero conductor thickness + */ +double c_microstrip::filling_factor_even(double u, double g, double e_r) +{ + double v, v3, v4, a_e, b_e, q_inf; + + v = u * (20.0 + g * g) / (10.0 + g * g) + g * exp(-g); + v3 = v * v * v; + v4 = v3 * v; + a_e = 1.0 + log((v4 + v * v / 2704.0) / (v4 + 0.432)) / 49.0 + log(1.0 + v3 / 5929.741) + / 18.7; + b_e = 0.564 * pow(((e_r - 0.9) / (e_r + 3.0)), 0.053); + + /* filling factor, with width corrected for thickness */ + q_inf = pow((1.0 + 10.0 / v), -a_e * b_e); + + return q_inf; +} + +/** + * filling_factor_odd() - compute the filling factor for the coupled + * microstrips odd-mode without cover and zero conductor thickness + */ +double c_microstrip::filling_factor_odd(double u, double g, double e_r) +{ + double b_o, c_o, d_o, q_inf; + + b_o = 0.747 * e_r / (0.15 + e_r); + c_o = b_o - (b_o - 0.207) * exp(-0.414 * u); + d_o = 0.593 + 0.694 * exp(-0.562 * u); + + /* filling factor, with width corrected for thickness */ + q_inf = exp(-c_o * pow(g, d_o)); + + return q_inf; +} + + +/* + * delta_q_cover_even() - compute the cover effect on filling factor + * for the even-mode + */ +double c_microstrip::delta_q_cover_even(double h2h) +{ + double q_c; + + if (h2h <= 39) { + q_c = tanh(1.626 + 0.107 * h2h - 1.733 / sqrt(h2h)); + } else { + q_c = 1.0; + } + + return q_c; +} + +/* + * delta_q_cover_odd() - compute the cover effect on filling factor + * for the odd-mode + */ +double c_microstrip::delta_q_cover_odd(double h2h) +{ + double q_c; + + if (h2h <= 7) { + q_c = tanh(9.575 / (7.0 - h2h) - 2.965 + 1.68 * h2h - 0.311 * h2h * h2h); + } else { + q_c = 1.0; + } + + return q_c; +} + +/** + * er_eff_static() - compute the static effective dielectric constants + * + * References: Manfred Kirschning and Rolf Jansen, "Accurate + * Wide-Range Design Equations for the Frequency-Dependent + * Characteristic of Parallel Coupled Microstrip Lines", IEEE + * Trans. MTT, vol. 32, no. 1, Jan. 1984 + */ +void c_microstrip::er_eff_static() +{ + double u_t_e, u_t_o, g, h2, h2h; + double a_o, t_h, q, q_c, q_t, q_inf; + double er_eff_single; + + /* compute zero-thickness single line parameters */ + compute_single_line(); + er_eff_single = aux_ms->er_eff_0; + + h2 = ht; + u_t_e = w_t_e / h; /* normalized even_mode line width */ + u_t_o = w_t_o / h; /* normalized odd_mode line width */ + g = s / h; /* normalized line spacing */ + h2h = h2 / h; /* normalized cover height */ + t_h = t / h; /* normalized strip thickness */ + + /* filling factor, computed with thickness corrected width */ + q_inf = filling_factor_even(u_t_e, g, er); + /* cover effect */ + q_c = delta_q_cover_even(h2h); + /* thickness effect */ + q_t = aux_ms->delta_q_thickness(u_t_e, t_h); + /* resultant filling factor */ + q = (q_inf - q_t) * q_c; + /* static even-mode effective dielectric constant */ + er_eff_e_0 = 0.5 * (er + 1.0) + 0.5 * (er - 1.0) * q; + + /* filling factor, with width corrected for thickness */ + q_inf = filling_factor_odd(u_t_o, g, er); + /* cover effect */ + q_c = delta_q_cover_odd(h2h); + /* thickness effect */ + q_t = aux_ms->delta_q_thickness(u_t_o, t_h); + /* resultant filling factor */ + q = (q_inf - q_t) * q_c; + + a_o = 0.7287 * (er_eff_single - 0.5 * (er + 1.0)) * (1.0 - exp(-0.179 * u_t_o)); + + /* static odd-mode effective dielectric constant */ + er_eff_o_0 = (0.5 * (er + 1.0) + a_o - er_eff_single) * q + er_eff_single; +} + + +/** + * delta_Z0_even_cover() - compute the even-mode impedance correction + * for a homogeneous microstrip due to the cover + * + * References: S. March, "Microstrip Packaging: Watch the Last Step", + * Microwaves, vol. 20, no. 13, pp. 83.94, Dec. 1981. + */ +double c_microstrip::delta_Z0_even_cover(double g, double u, double h2h) +{ + double f_e, g_e, delta_Z0_even; + double x, y, A, B, C, D, E, F; + + A = -4.351 / pow(1.0 + h2h, 1.842); + B = 6.639 / pow(1.0 + h2h, 1.861); + C = -2.291 / pow(1.0 + h2h, 1.90); + f_e = 1.0 - atanh(A + (B + C * u) * u); + + if (g < 4.46631063751) { + x = pow(10.0, 0.103 * g - 0.159); + y = pow(10.0, 0.0492 * g - 0.073); + D = 0.747 / sin(0.5 * pi * x); + E = 0.725 * sin(0.5 * pi * y); + F = pow(10.0, 0.11 - 0.0947 * g); + g_e = 270.0 * (1.0 - tanh(D + E * sqrt(1.0 + h2h) - F / (1.0 + h2h))); + } else + g_e = 0.0; + + delta_Z0_even = f_e * g_e; + + return delta_Z0_even; +} + + +/** + * delta_Z0_odd_cover() - compute the odd-mode impedance correction + * for a homogeneous microstrip due to the cover + * + * References: S. March, "Microstrip Packaging: Watch the Last Step", + * Microwaves, vol. 20, no. 13, pp. 83.94, Dec. 1981. + */ +double c_microstrip::delta_Z0_odd_cover(double g, double u, double h2h) +{ + double f_o, g_o, delta_Z0_odd; + double G, J, K, L; + + J = tanh(pow(1.0 + h2h, 1.585) / 6.0); + f_o = pow(u, J); + + G = 2.178 - 0.796 * g; + if (g > 0.858) { + K = log10(20.492 * pow(g, 0.174)); + } else { + K = 1.30; + } + if (g > 0.873) { + L = 2.51 * pow(g, -0.462); + } else { + L = 2.674; + } + g_o = 270.0 * (1.0 - tanh(G + K * sqrt(1.0 + h2h) - L / (1.0 + h2h))); + + delta_Z0_odd = f_o * g_o; + + return delta_Z0_odd; +} + +/** + * Z0_even_odd() - compute the static even- and odd-mode static + * impedances + * + * References: Manfred Kirschning and Rolf Jansen, "Accurate + * Wide-Range Design Equations for the Frequency-Dependent + * Characteristic of Parallel Coupled Microstrip Lines", IEEE + * Trans. MTT, vol. 32, no. 1, Jan. 1984 + */ +void c_microstrip::Z0_even_odd() +{ + double er_eff, h2, u_t_e, u_t_o, g, h2h; + double Q_1, Q_2, Q_3, Q_4, Q_5, Q_6, Q_7, Q_8, Q_9, Q_10; + double delta_Z0_e_0, delta_Z0_o_0, Z0_single, er_eff_single; + + h2 = ht; + u_t_e = w_t_e / h; /* normalized even-mode line width */ + u_t_o = w_t_o / h; /* normalized odd-mode line width */ + g = s / h; /* normalized line spacing */ + h2h = h2 / h; /* normalized cover height */ + + Z0_single = aux_ms->Z0_0; + er_eff_single = aux_ms->er_eff_0; + + /* even-mode */ + er_eff = er_eff_e_0; + Q_1 = 0.8695 * pow(u_t_e, 0.194); + Q_2 = 1.0 + 0.7519 * g + 0.189 * pow(g, 2.31); + Q_3 = 0.1975 + pow((16.6 + pow((8.4 / g), 6.0)), -0.387) + log(pow(g, 10.0) / (1.0 + pow(g / 3.4, 10.0))) / 241.0; + Q_4 = 2.0 * Q_1 / (Q_2 * (exp(-g) * pow(u_t_e, Q_3) + (2.0 - exp(-g)) * pow(u_t_e, -Q_3))); + /* static even-mode impedance */ + Z0_e_0 = Z0_single * sqrt(er_eff_single / er_eff) / (1.0 - sqrt(er_eff_single) * Q_4 * Z0_single / ZF0); + /* correction for cover */ + delta_Z0_e_0 = delta_Z0_even_cover(g, u_t_e, h2h) / sqrt(er_eff); + + Z0_e_0 = Z0_e_0 - delta_Z0_e_0; + + /* odd-mode */ + er_eff = er_eff_o_0; + Q_5 = 1.794 + 1.14 * log(1.0 + 0.638 / (g + 0.517 * pow(g, 2.43))); + Q_6 = 0.2305 + log(pow(g, 10.0) / (1.0 + pow(g / 5.8, 10.0))) / 281.3 + log(1.0 + 0.598 * pow(g, 1.154)) / 5.1; + Q_7 = (10.0 + 190.0 * g * g) / (1.0 + 82.3 * g * g * g); + Q_8 = exp(-6.5 - 0.95 * log(g) - pow(g / 0.15, 5.0)); + Q_9 = log(Q_7) * (Q_8 + 1.0 / 16.5); + Q_10 = (Q_2 * Q_4 - Q_5 * exp(log(u_t_o) * Q_6 * pow(u_t_o, -Q_9))) / Q_2; + + /* static odd-mode impedance */ + Z0_o_0 = Z0_single * sqrt(er_eff_single / er_eff) / (1.0 - sqrt(er_eff_single) * Q_10 * Z0_single / ZF0); + /* correction for cover */ + delta_Z0_o_0 = delta_Z0_odd_cover(g, u_t_o, h2h) / sqrt(er_eff); + + Z0_o_0 = Z0_o_0 - delta_Z0_o_0; +} + + +/* + * mur_eff() - returns effective magnetic permeability + */ +double c_microstrip::calc_mur_eff() +{ + double mureff; + mureff = mur; /* FIXME: ... */ + return mureff; +} + + +/* + * er_eff_freq() - compute er_eff as a function of frequency + */ +void c_microstrip::er_eff_freq() +{ + double P_1, P_2, P_3, P_4, P_5, P_6, P_7; + double P_8, P_9, P_10, P_11, P_12, P_13, P_14, P_15; + double F_e, F_o; + double er_eff, u, g, f_n; + + u = w / h; /* normalize line width */ + g = s / h; /* normalize line spacing */ + + /* normalized frequency [GHz * mm] */ + f_n = f * h / 1e06; + + er_eff = er_eff_e_0; + P_1 = 0.27488 + (0.6315 + 0.525 / pow(1.0 + 0.0157 * f_n, 20.0)) * u - 0.065683 * exp(-8.7513 * u); + P_2 = 0.33622 * (1.0 - exp(-0.03442 * er)); + P_3 = 0.0363 * exp(-4.6 * u) * (1.0 - exp(-pow(f_n / 38.7, 4.97))); + P_4 = 1.0 + 2.751 * (1.0 - exp(-pow(er / 15.916, 8.0))); + P_5 = 0.334 * exp(-3.3 * pow(er / 15.0, 3.0)) + 0.746; + P_6 = P_5 * exp(-pow(f_n / 18.0, 0.368)); + P_7 = 1.0 + 4.069 * P_6 * pow(g, 0.479) * exp(-1.347 * pow(g, 0.595) - 0.17 * pow(g, 2.5)); + + F_e = P_1 * P_2 * pow((P_3 * P_4 + 0.1844 * P_7) * f_n, 1.5763); + /* even-mode effective dielectric constant */ + er_eff_e = er - (er - er_eff) / (1.0 + F_e); + + er_eff = er_eff_o_0; + P_8 = 0.7168 * (1.0 + 1.076 / (1.0 + 0.0576 * (er - 1.0))); + P_9 = P_8 - 0.7913 * (1.0 - exp(-pow(f_n / 20.0, 1.424))) * atan(2.481 * pow(er / 8.0, 0.946)); + P_10 = 0.242 * pow(er - 1.0, 0.55); + P_11 = 0.6366 * (exp(-0.3401 * f_n) - 1.0) * atan(1.263 * pow(u / 3.0, 1.629)); + P_12 = P_9 + (1.0 - P_9) / (1.0 + 1.183 * pow(u, 1.376)); + P_13 = 1.695 * P_10 / (0.414 + 1.605 * P_10); + P_14 = 0.8928 + 0.1072 * (1.0 - exp(-0.42 * pow(f_n / 20.0, 3.215))); + P_15 = std::abs(1.0 - 0.8928 * (1.0 + P_11) * P_12 * exp(-P_13 * pow(g, 1.092)) / P_14); + + F_o = P_1 * P_2 * pow((P_3 * P_4 + 0.1844) * f_n * P_15, 1.5763); + /* odd-mode effective dielectric constant */ + er_eff_o = er - (er - er_eff) / (1.0 + F_o); +} + +/* + * conductor_losses() - compute microstrips conductor losses per unit + * length + */ +void c_microstrip::conductor_losses() +{ + double e_r_eff_e_0, e_r_eff_o_0, Z0_h_e, Z0_h_o, delta; + double K, R_s, Q_c_e, Q_c_o, alpha_c_e, alpha_c_o; + + e_r_eff_e_0 = er_eff_e_0; + e_r_eff_o_0 = er_eff_o_0; + Z0_h_e = Z0_e_0 * sqrt(e_r_eff_e_0); /* homogeneous stripline impedance */ + Z0_h_o = Z0_o_0 * sqrt(e_r_eff_o_0); /* homogeneous stripline impedance */ + delta = skindepth; + + if (f > 0.0) { + /* current distribution factor (same for the two modes) */ + K = exp(-1.2 * pow((Z0_h_e + Z0_h_o) / (2.0 * ZF0), 0.7)); + /* skin resistance */ + R_s = 1.0 / (sigma * delta); + /* correction for surface roughness */ + R_s *= 1.0 + ((2.0 / pi) * atan(1.40 * pow((rough / delta), 2.0))); + + /* even-mode strip inductive quality factor */ + Q_c_e = (pi * Z0_h_e * w * f) / (R_s * C0 * K); + /* even-mode losses per unith length */ + alpha_c_e = (20.0 * pi / log(10.0)) * f * sqrt(e_r_eff_e_0) / (C0 * Q_c_e); + + /* odd-mode strip inductive quality factor */ + Q_c_o = (pi * Z0_h_o * w * f) / (R_s * C0 * K); + /* odd-mode losses per unith length */ + alpha_c_o = (20.0 * pi / log(10.0)) * f * sqrt(e_r_eff_o_0) / (C0 * Q_c_o); + } else { + alpha_c_e = alpha_c_o = 0.0; + } + + atten_cond_e = alpha_c_e * l; + atten_cond_o = alpha_c_o * l; +} + + +/* + * dielectric_losses() - compute microstrips dielectric losses per + * unit length + */ +void c_microstrip::dielectric_losses() +{ + double e_r, e_r_eff_e_0, e_r_eff_o_0; + double alpha_d_e, alpha_d_o; + + e_r = er; + e_r_eff_e_0 = er_eff_e_0; + e_r_eff_o_0 = er_eff_o_0; + + alpha_d_e = (20.0 * pi / log(10.0)) * (f / C0) * (e_r / sqrt(e_r_eff_e_0)) * ((e_r_eff_e_0 - 1.0) / (e_r - 1.0)) * tand; + alpha_d_o = (20.0 * pi / log(10.0)) * (f / C0) * (e_r / sqrt(e_r_eff_o_0)) * ((e_r_eff_o_0 - 1.0) / (e_r - 1.0)) * tand; + + atten_dielectric_e = alpha_d_e * l; + atten_dielectric_o = alpha_d_o * l; +} + + +/* + * c_microstrip_attenuation() - compute attenuation of coupled + * microstrips + */ +void c_microstrip::attenuation() +{ + skindepth = skin_depth(); + conductor_losses(); + dielectric_losses(); +} + + +/* + * line_angle() - calculate strips electrical lengths in radians + */ +void c_microstrip::line_angle() +{ + double e_r_eff_e, e_r_eff_o; + double v_e, v_o, lambda_g_e, lambda_g_o; + + e_r_eff_e = er_eff_e; + e_r_eff_o = er_eff_o; + + /* even-mode velocity */ + v_e = C0 / sqrt(e_r_eff_e); + /* odd-mode velocity */ + v_o = C0 / sqrt(e_r_eff_o); + /* even-mode wavelength */ + lambda_g_e = v_e / f; + /* odd-mode wavelength */ + lambda_g_o = v_o / f; + /* electrical angles */ + ang_l_e = 2.0 * pi * l / lambda_g_e; /* in radians */ + ang_l_o = 2.0 * pi * l / lambda_g_o; /* in radians */ +} + + +void c_microstrip::syn_err_fun(double *f1, double *f2, double s_h, double w_h, double e_r, double w_h_se, double w_h_so) +{ + + double g, h; + + g = cosh(0.5 * pi * s_h); + h = cosh(pi * w_h + 0.5 * pi * s_h); + + *f1 = (2.0 / pi) * acosh((2.0 * h - g + 1.0) / (g + 1.0)); + *f2 = (2.0 / pi) * acosh((2.0 * h - g - 1.0) / (g - 1.0)); + if (e_r <= 6.0) { + *f2 += (4.0 / (pi * (1.0 + e_r / 2.0))) * acosh(1.0 + 2.0 * w_h / s_h); + } else { + *f2 += (1.0 / pi) * acosh(1.0 + 2.0 * w_h / s_h); + } + *f1 -= w_h_se; + *f2 -= w_h_so; +} + +/* + * synth_width - calculate widths given Z0 and e_r + * from Akhtarzad S. et al., "The design of coupled microstrip lines", + * IEEE Trans. MTT-23, June 1975 and + * Hinton, J.H., "On design of coupled microstrip lines", IEEE Trans. + * MTT-28, March 1980 + */ +void c_microstrip::synth_width() +{ + double Z0, e_r; + double w_h_se, w_h_so, w_h, a, ce, co, s_h; + double f1, f2, ft1, ft2, j11, j12, j21, j22, d_s_h, d_w_h, err; + double eps = 1e-04; + + f1 = f2 = 0; + e_r = er; + + Z0 = Z0e / 2.0; + /* Wheeler formula for single microstrip synthesis */ + a = exp(Z0 * sqrt(e_r + 1.0) / 42.4) - 1.0; + w_h_se = 8.0 * sqrt(a * ((7.0 + 4.0 / e_r) / 11.0) + ((1.0 + 1.0 / e_r) / 0.81)) / a; + + Z0 = Z0o / 2.0; + /* Wheeler formula for single microstrip synthesis */ + a = exp(Z0 * sqrt(e_r + 1.0) / 42.4) - 1.0; + w_h_so = 8.0 * sqrt(a * ((7.0 + 4.0 / e_r) / 11.0) + ((1.0 + 1.0 / e_r) / 0.81)) / a; + + ce = cosh(0.5 * pi * w_h_se); + co = cosh(0.5 * pi * w_h_so); + /* first guess at s/h */ + s_h = (2.0 / pi) * acosh((ce + co - 2.0) / (co - ce)); + /* first guess at w/h */ + w_h = acosh((ce * co - 1.0) / (co - ce)) / pi - s_h / 2.0; + + s = s_h * h; + w = w_h * h; + + syn_err_fun(&f1, &f2, s_h, w_h, e_r, w_h_se, w_h_so); + + /* rather crude Newton-Rhapson; we need this beacuse the estimate of */ + /* w_h is often quite far from the true value (see Akhtarzad S. et al.) */ + do { + /* compute Jacobian */ + syn_err_fun(&ft1, &ft2, s_h + eps, w_h, e_r, w_h_se, w_h_so); + j11 = (ft1 - f1) / eps; + j21 = (ft2 - f2) / eps; + syn_err_fun(&ft1, &ft2, s_h, w_h + eps, e_r, w_h_se, w_h_so); + j12 = (ft1 - f1) / eps; + j22 = (ft2 - f2) / eps; + + /* compute next step */ + d_s_h = (-f1 * j22 + f2 * j12) / (j11 * j22 - j21 * j12); + d_w_h = (-f2 * j11 + f1 * j21) / (j11 * j22 - j21 * j12); + //g_print("j11 = %e\tj12 = %e\tj21 = %e\tj22 = %e\n", j11, j12, j21, j22); + //g_print("det = %e\n", j11*j22 - j21*j22); + //g_print("d_s_h = %e\td_w_h = %e\n", d_s_h, d_w_h); + + s_h += d_s_h; + w_h += d_w_h; + + /* chech the error */ + syn_err_fun(&f1, &f2, s_h, w_h, e_r, w_h_se, w_h_so); + + err = sqrt(f1 * f1 + f2 * f2); + /* converged ? */ + } while (err > 1e-04); + + + s = s_h * h; + w = w_h * h; +} + + +/* + * Z0_dispersion() - calculate frequency dependency of characteristic + * impedances + */ +void c_microstrip::Z0_dispersion() +{ + double Q_0; + double Q_11, Q_12, Q_13, Q_14, Q_15, Q_16, Q_17, Q_18, Q_19, Q_20, Q_21; + double Q_22, Q_23, Q_24, Q_25, Q_26, Q_27, Q_28, Q_29; + double r_e, q_e, p_e, d_e, C_e; + double e_r_eff_o_f, e_r_eff_o_0; + double e_r_eff_single_f, e_r_eff_single_0, Z0_single_f; + double f_n, g, u, e_r; + double R_1, R_2, R_7, R_10, R_11, R_12, R_15, R_16, tmpf; + + e_r = er; + + u = w / h; /* normalize line width */ + g = s / h; /* normalize line spacing */ + + /* normalized frequency [GHz * mm] */ + f_n = f * h / 1e06; + + e_r_eff_single_f = aux_ms->er_eff; + e_r_eff_single_0 = aux_ms->er_eff_0; + Z0_single_f = aux_ms->Z0; + + e_r_eff_o_f = er_eff_o; + e_r_eff_o_0 = er_eff_o_0; + + Q_11 = 0.893 * (1.0 - 0.3 / (1.0 + 0.7 * (e_r - 1.0))); + Q_12 = 2.121 * (pow(f_n / 20.0, 4.91) / (1.0 + Q_11 * pow(f_n / 20.0, 4.91))) * exp(-2.87 * g) * pow(g, 0.902); + Q_13 = 1.0 + 0.038 * pow(e_r / 8.0, 5.1); + Q_14 = 1.0 + 1.203 * pow(e_r / 15.0, 4.0) / (1.0 + pow(e_r / 15.0, 4.0)); + Q_15 = 1.887 * exp(-1.5 * pow(g, 0.84)) * pow(g, Q_14) / (1.0 + 0.41 * pow(f_n / 15.0, 3.0) * pow(u, 2.0 / Q_13) / (0.125 + pow(u, 1.626 / Q_13))); + Q_16 = (1.0 + 9.0 / (1.0 + 0.403 * pow(e_r - 1.0, 2))) * Q_15; + Q_17 = 0.394 * (1.0 - exp(-1.47 * pow(u / 7.0, 0.672))) * (1.0 - exp(-4.25 * pow(f_n / 20.0, 1.87))); + Q_18 = 0.61 * (1.0 - exp(-2.13 * pow(u / 8.0, 1.593))) / (1.0 + 6.544 * pow(g, 4.17)); + Q_19 = 0.21 * g * g * g * g / ((1.0 + 0.18 * pow(g, 4.9)) * (1.0 + 0.1 * u * u) * (1.0 + pow(f_n / 24.0, 3.0))); + Q_20 = (0.09 + 1.0 / (1.0 + 0.1 * pow(e_r - 1, 2.7))) * Q_19; + Q_21 = std::abs(1.0 - 42.54 * pow(g, 0.133) * exp(-0.812 * g) * pow(u, 2.5) / (1.0 + 0.033 * pow(u, 2.5))); + + r_e = pow(f_n / 28.843, 12); + q_e = 0.016 + pow(0.0514 * e_r * Q_21, 4.524); + p_e = 4.766 * exp(-3.228 * pow(u, 0.641)); + d_e = 5.086 * q_e * (r_e / (0.3838 + 0.386 * q_e)) * (exp(-22.2 * pow(u, 1.92)) / (1.0 + 1.2992 * r_e)) * (pow(e_r - 1.0, 6.0) / (1.0 + 10 * pow(e_r - 1.0, 6.0))); + C_e = 1.0 + 1.275 * (1.0 - exp(-0.004625 * p_e * pow(e_r, 1.674) * pow(f_n / 18.365, 2.745))) - Q_12 + Q_16 - Q_17 + Q_18 + Q_20; + + + R_1 = 0.03891 * pow(e_r, 1.4); + R_2 = 0.267 * pow(u, 7.0); + R_7 = 1.206 - 0.3144 * exp(-R_1) * (1.0 - exp(-R_2)); + R_10 = 0.00044 * pow(e_r, 2.136) + 0.0184; + tmpf = pow(f_n / 19.47, 6.0); + R_11 = tmpf / (1.0 + 0.0962 * tmpf); + R_12 = 1.0 / (1.0 + 0.00245 * u * u); + R_15 = 0.707 * R_10 * pow(f_n / 12.3, 1.097); + R_16 = 1.0 + 0.0503 * e_r * e_r * R_11 * (1.0 - exp(-pow(u / 15.0, 6.0))); + Q_0 = R_7 * (1.0 - 1.1241 * (R_12 / R_16) * exp(-0.026 * pow(f_n, 1.15656) - R_15)); + + /* even-mode frequency-dependent characteristic impedances */ + Z0e = Z0_e_0 * pow(0.9408 * pow(e_r_eff_single_f, C_e) - 0.9603, Q_0) / pow((0.9408 - d_e) * pow(e_r_eff_single_0, C_e) - 0.9603, Q_0); + + Q_29 = 15.16 / (1.0 + 0.196 * pow(e_r - 1.0, 2.0)); + tmpf = pow(e_r - 1.0, 3.0); + Q_28 = 0.149 * tmpf / (94.5 + 0.038 * tmpf); + tmpf = pow(e_r - 1.0, 1.5); + Q_27 = 0.4 * pow(g, 0.84) * (1.0 + 2.5 * tmpf / (5.0 + tmpf)); + tmpf = pow((e_r - 1.0) / 13.0, 12.0); + Q_26 = 30.0 - 22.2 * (tmpf / (1.0 + 3.0 * tmpf)) - Q_29; + tmpf = (e_r - 1.0) * (e_r - 1.0); + Q_25 = (0.3 * f_n * f_n / (10.0 + f_n * f_n)) * (1.0 + 2.333 * tmpf / (5.0 + tmpf)); + Q_24 = 2.506 * Q_28 * pow(u, 0.894) * pow((1.0 + 1.3 * u) * f_n / 99.25, 4.29) / (3.575 + pow(u, 0.894)); + Q_23 = 1.0 + 0.005 * f_n * Q_27 / ((1.0 + 0.812 * pow(f_n / 15.0, 1.9)) * (1.0 + 0.025 * u * u)); + Q_22 = 0.925 * pow(f_n / Q_26, 1.536) / (1.0 + 0.3 * pow(f_n / 30.0, 1.536)); + + /* odd-mode frequency-dependent characteristic impedances */ + Z0o = Z0_single_f + (Z0_o_0 * pow(e_r_eff_o_f / e_r_eff_o_0, Q_22) - Z0_single_f * Q_23) / (1.0 + Q_24 + pow(0.46 * g, 2.2) * Q_25); +} + + +void c_microstrip::calc() +{ + /* compute thickness corrections */ + delta_u_thickness(); + /* get effective dielectric constants */ + er_eff_static(); + /* impedances for even- and odd-mode */ + Z0_even_odd(); + /* calculate freq dependence of er_eff_e, er_eff_o */ + er_eff_freq(); + /* FIXME: (not used) Get effective magnetic permeability */ + mur_eff = calc_mur_eff(); + /* calculate frequency dependence of Z0e, Z0o */ + Z0_dispersion(); + /* calculate losses */ + attenuation(); + /* calculate electrical lengths */ + line_angle(); +} + +/* + * get_microstrip_sub + * get and assign microstrip substrate parameters + * into microstrip structure + */ +void c_microstrip::get_c_microstrip_sub() +{ + er = getProperty ("Er"); + mur = getProperty ("Mur"); + h = getProperty ("H", UNIT_LENGTH, LENGTH_M); + ht = getProperty ("H_t", UNIT_LENGTH, LENGTH_M); + t = getProperty ("T", UNIT_LENGTH, LENGTH_M); + sigma = getProperty ("Cond"); + tand = getProperty ("Tand"); + rough = getProperty ("Rough", UNIT_LENGTH, LENGTH_M); +} + +/* + * get_c_microstrip_comp + * get and assign microstrip component parameters + * into microstrip structure + */ +void c_microstrip::get_c_microstrip_comp() +{ + f = getProperty ("Freq", UNIT_FREQ, FREQ_HZ); +} + +/* + * get_c_microstrip_elec + * get and assign microstrip electrical parameters + * into microstrip structure + */ +void c_microstrip::get_c_microstrip_elec() +{ + Z0e = getProperty ("Z0e", UNIT_RES, RES_OHM); + Z0o = getProperty ("Z0o", UNIT_RES, RES_OHM); + ang_l_e = getProperty ("Ang_l", UNIT_ANG, ANG_RAD); + ang_l_o = getProperty ("Ang_l", UNIT_ANG, ANG_RAD); +} + + +/* + * get_c_microstrip_phys + * get and assign microstrip physical parameters + * into microstrip structure + */ +void c_microstrip::get_c_microstrip_phys() +{ + w = getProperty ("W", UNIT_LENGTH, LENGTH_M); + s = getProperty ("S", UNIT_LENGTH, LENGTH_M); + l = getProperty ("L", UNIT_LENGTH, LENGTH_M); +} + + +void c_microstrip::show_results() +{ + setResult (0, er_eff_e, ""); + setResult (1, er_eff_o, ""); + setResult (2, atten_cond_e, "dB"); + setResult (3, atten_cond_o, "dB"); + setResult (4, atten_dielectric_e, "dB"); + setResult (5, atten_dielectric_o, "dB"); + + double val = convertProperty ("T", skindepth, UNIT_LENGTH, LENGTH_M); + setResult (6, val, getUnit ("T")); +} + + +/* + * analysis function + */ +void c_microstrip::analyze() +{ + /* Get and assign substrate parameters */ + get_c_microstrip_sub(); + /* Get and assign component parameters */ + get_c_microstrip_comp(); + /* Get and assign physical parameters */ + get_c_microstrip_phys(); + + /* compute coupled microstrip parameters */ + calc(); + + /* update electrical parameters */ + setProperty ("Z0e", Z0e, UNIT_RES, RES_OHM); + setProperty ("Z0o", Z0o, UNIT_RES, RES_OHM); + setProperty ("Ang_l", sqrt (ang_l_e * ang_l_o), UNIT_ANG, ANG_RAD); + + /* print results in the subwindow */ + show_results(); +} + + +void c_microstrip::syn_fun(double *f1, double *f2, double s_h, double w_h, double Z0_e, double Z0_o) +{ + s = s_h * h; + w = w_h * h; + + /* compute coupled microstrip parameters */ + calc(); + + *f1 = Z0e - Z0_e; + *f2 = Z0o - Z0_o; +} + +/* + * synthesis function + */ +int c_microstrip::synthesize() +{ + double Z0_e, Z0_o; + double f1, f2, ft1, ft2, j11, j12, j21, j22, d_s_h, d_w_h, err; + double eps = 1e-04; + double w_h, s_h, le, lo; + int iter = 0; + const int maxiter = 1000; + + /* Get and assign substrate parameters */ + get_c_microstrip_sub(); + + /* Get and assign component parameters */ + get_c_microstrip_comp(); + + /* Get and assign electrical parameters */ + get_c_microstrip_elec(); + + /* Get and assign physical parameters */ + /* at present it is required only for getting strips length */ + get_c_microstrip_phys(); + + + /* required value of Z0_e and Z0_o */ + Z0_e = Z0e; + Z0_o = Z0o; + + /* calculate width and use for initial value in Newton's method */ + synth_width(); + w_h = w / h; + s_h = s / h; + f1 = f2 = 0; + + /* rather crude Newton-Rhapson */ + /* might fail due to overshooting or singular jacobian, should implement a better algorithm... */ + /* initial errors values */ + syn_fun(&f1, &f2, s_h, w_h, Z0_e, Z0_o); + do { + /* compute Jacobian */ + syn_fun(&ft1, &ft2, s_h + eps, w_h, Z0_e, Z0_o); + j11 = (ft1 - f1) / eps; + j21 = (ft2 - f2) / eps; + syn_fun(&ft1, &ft2, s_h, w_h + eps, Z0_e, Z0_o); + j12 = (ft1 - f1) / eps; + j22 = (ft2 - f2) / eps; + + /* compute next step; increments of s_h and w_h */ + d_s_h = (-f1 * j22 + f2 * j12) / (j11 * j22 - j21 * j12); + d_w_h = (-f2 * j11 + f1 * j21) / (j11 * j22 - j21 * j12); + + if (!std::isfinite(d_s_h) || !std::isfinite(d_w_h)) { + /* a computed step is infinite: we are lost... */ + iter = maxiter+1; /* just to signal we did not converge */ + break; + } + s_h += d_s_h; + if (s_h <= 0.0) s_h = eps; /* avoid negative values */ + w_h += d_w_h; + if (w_h <= 0.0) w_h = eps; /* avoid negative values */ + + /* compute the error with the new values of s_h and w_h */ + syn_fun(&f1, &f2, s_h, w_h, Z0_e, Z0_o); + err = sqrt(f1 * f1 + f2 * f2); + + iter++; + /* converged ? */ + } while ((err > 1e-04) && (iter < maxiter)); + + /* denormalize computed width and spacing */ + s = s_h * h; + w = w_h * h; + + /* calculate physical length */ + ang_l_e = getProperty ("Ang_l", UNIT_ANG, ANG_RAD); + ang_l_o = getProperty ("Ang_l", UNIT_ANG, ANG_RAD); + le = C0 / f / sqrt(er_eff_e * mur_eff) * ang_l_e / 2.0 / pi; + lo = C0 / f / sqrt(er_eff_o * mur_eff) * ang_l_o / 2.0 / pi; + l = sqrt (le * lo); + + /* update physical parameters */ + setProperty ("W", w, UNIT_LENGTH, LENGTH_M); + setProperty ("S", s, UNIT_LENGTH, LENGTH_M); + setProperty ("L", l, UNIT_LENGTH, LENGTH_M); + + calc(); + /* print results in the subwindow */ + show_results(); + + if (iter > maxiter) + return -1; + else + return 0; +} diff --git a/qucs-transcalc/c_microstrip.h b/qucs-transcalc/c_microstrip.h new file mode 100644 index 00000000..0cee09fd --- /dev/null +++ b/qucs-transcalc/c_microstrip.h @@ -0,0 +1,102 @@ +/* + * c_microstrip.h - coupled microstrip class definition + * + * Copyright (C) 2002 Claudio Girardi + * Copyright (C) 2005 Stefan Jahn + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + +#ifndef _C_MICROSTRIP_H_ +#define _C_MICROSTRIP_H_ + +class c_microstrip : public transline { + public: + c_microstrip(); + ~c_microstrip(); + + private: + double er; /* dielectric constant */ + double h; /* height of substrate */ + double ht; /* height to the top of box */ + double t; /* thickness of top metal */ + double tand; /* Dielectric Loss Tangent */ + double rough; /* Roughness of top metal */ + double w; /* width of lines */ + double w_t_e; /* even-mode thickness-corrected line width */ + double w_t_o; /* odd-mode thickness-corrected line width */ + double l; /* length of lines */ + double s; /* spacing of lines */ + double Z0_e_0; /* static even-mode impedance */ + double Z0_o_0; /* static odd-mode impedance */ + double Z0e; /* even-mode impedance */ + double Z0o; /* odd-mode impedance */ + //double c_e; /* even-mode capacitance */ + //double c_o; /* odd-mode capacitance */ + double ang_l_e; /* even-mode electrical length in angle */ + double ang_l_o; /* odd-mode electrical length in angle */ + double er_eff_e; /* even-mode effective dielectric constant */ + double er_eff_o; /* odd-mode effective dielectric constant */ + double er_eff_e_0; /* static even-mode effective dielectric constant */ + double er_eff_o_0; /* static odd-mode effective dielectric constant */ + //double er_eff; /* FIXME: dummy */ + double mur_eff; /* Effective mag. permeability */ + //double w_eff; /* Effective width of line */ + double atten_dielectric_e; /* even-mode dielectric losses (dB) */ + double atten_cond_e; /* even-mode conductors losses (dB) */ + double atten_dielectric_o; /* odd-mode dielectric losses (dB) */ + double atten_cond_o; /* odd-mode conductors losses (dB) */ + + public: + void analyze (); + int synthesize (); + + private: + double delta_u_thickness_single(double, double); + void delta_u_thickness(); + void compute_single_line(); + double filling_factor_even(double, double, double); + double filling_factor_odd(double, double, double); + double delta_q_cover_even(double); + double delta_q_cover_odd(double); + void er_eff_static(); + double delta_Z0_even_cover(double, double, double); + double delta_Z0_odd_cover(double, double, double); + void Z0_even_odd(); + double calc_mur_eff(); + void er_eff_freq(); + void conductor_losses(); + void dielectric_losses(); + void attenuation(); + void line_angle(); + void syn_err_fun(double *, double *, double, double, double, double, double); + void synth_width(); + void Z0_dispersion(); + void calc(); + void get_c_microstrip_sub(); + void get_c_microstrip_comp(); + void get_c_microstrip_elec(); + void get_c_microstrip_phys(); + void show_results(); + void syn_fun(double *, double *, double, double, double, double); + + + private: + microstrip * aux_ms; +}; + +#endif /* _C_MICROSTRIP_H_ */ diff --git a/qucs-transcalc/coax.cpp b/qucs-transcalc/coax.cpp new file mode 100644 index 00000000..ae592033 --- /dev/null +++ b/qucs-transcalc/coax.cpp @@ -0,0 +1,223 @@ +/* + * coax.cpp - coaxial class implementation + * + * Copyright (C) 2001 Gopal Narayanan + * Copyright (C) 2002 Claudio Girardi + * Copyright (C) 2005, 2006, 2009, 2011 Stefan Jahn + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or (at + * your option) any later version. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + + +/* + * coax.c - Puts up window for microstrip and + * performs the associated calculations + */ + + +#include +#include +#include +#include + +#include "units.h" +#include "transline.h" +#include "coax.h" + +coax::coax() : transline() +{ +} + +coax::~coax() +{ +} + +/* + * get_coax_sub() - get and assign coax substrate parameters into coax + * structure + */ +void coax::get_coax_sub () +{ + er = getProperty ("Er"); + mur = getProperty ("Mur"); + tand = getProperty ("Tand"); + sigma = getProperty ("Sigma"); +} + +/* + * get_coax_comp() - get and assign coax component parameters into + * coax structure + */ +void coax::get_coax_comp () +{ + f = getProperty ("Freq", UNIT_FREQ, FREQ_HZ); +} + +/* + * get_coax_elec() - get and assign coax electrical parameters into + * coax structure + */ +void coax::get_coax_elec () +{ + Z0 = getProperty ("Z0", UNIT_RES, RES_OHM); + ang_l = getProperty ("Ang_l", UNIT_ANG, ANG_RAD); +} + +/* + * get_coax_phys() - get and assign coax physical parameters into coax + * structure + */ +void coax::get_coax_phys () +{ + din = getProperty ("din", UNIT_LENGTH, LENGTH_M); + dout = getProperty ("dout", UNIT_LENGTH, LENGTH_M); + l = getProperty ("L", UNIT_LENGTH, LENGTH_M); +} + +double coax::alphad_coax () +{ + double ad; + ad = (pi/C0) * f * sqrt(er) * tand; + ad = ad * 20.0 / log(10.0); + return ad; +} + +double coax::alphac_coax () +{ + double ac, Rs; + Rs = sqrt((pi * f * mur* MU0)/sigma); + ac = sqrt(er) * (((1/din) + (1/dout))/log(dout/din)) * (Rs/ZF0); + ac = ac * 20.0 / log(10.0); + return ac; +} + +/* + * analyze() - analysis function + */ +void coax::analyze () +{ + double lambda_g; + + /* Get and assign substrate parameters */ + get_coax_sub(); + + /* Get and assign component parameters */ + get_coax_comp(); + + /* Get and assign physical parameters */ + get_coax_phys(); + + if (din != 0.0){ + Z0 = (ZF0/2/pi/sqrt(er))*log(dout/din); + } + + lambda_g = (C0/(f))/sqrt(er * mur); + /* calculate electrical angle */ + ang_l = (2.0 * pi * l)/lambda_g; /* in radians */ + + setProperty ("Z0", Z0, UNIT_RES, RES_OHM); + setProperty ("Ang_l", ang_l, UNIT_ANG, ANG_RAD); + + show_results(); +} + + +/* + * synthesize() - synthesis function + */ +int coax::synthesize () +{ + double lambda_g; + + /* Get and assign substrate parameters */ + get_coax_sub(); + + /* Get and assign component parameters */ + get_coax_comp(); + + /* Get and assign electrical parameters */ + get_coax_elec (); + + /* Get and assign physical parameters */ + get_coax_phys(); + + if (isSelected ("din")) { + /* solve for din */ + din = dout / exp(Z0*sqrt(er)/ZF0*2*pi); + setProperty ("din", din, UNIT_LENGTH, LENGTH_M); + } else if (isSelected ("dout")) { + /* solve for dout */ + dout = din * exp(Z0*sqrt(er)/ZF0*2*pi); + setProperty ("dout", dout, UNIT_LENGTH, LENGTH_M); + } + + lambda_g = (C0/(f))/sqrt(er * mur); + /* calculate physical length */ + l = (lambda_g * ang_l)/(2.0 * pi); /* in m */ + setProperty ("L", l, UNIT_LENGTH, LENGTH_M); + + show_results(); + + return 0; +} + +/* + * show_results() - show results + */ +void coax::show_results() +{ + double fc; + short m, n; + + atten_dielectric = alphad_coax () * l; + atten_cond = alphac_coax () * l; + + setResult (0, atten_cond, "dB"); + setResult (1, atten_dielectric, "dB"); + + n = 1; + fc = C0 / sqrt (er * mur) / (pi_over_2 * (dout + din)/(double) n); + setResult (2, "none"); + if (fc <= f) { + char text[256], txt[256]; + strcpy (text, "TE(1,1) "); + m = 2; + fc = C0 / sqrt (er * mur) / (pi_over_2 * (dout + din)/(double) m); + while ((fc <= f) && (m<10)) { + sprintf(txt, "TE(n,%d) ",m); + strcat(text,txt); + m++; + fc = C0 / sqrt (er * mur) / (pi_over_2 * (dout + din)/(double) m); + } + setResult (2, text); + } + + setResult (3, "none"); + m = 1; + fc = C0 / sqrt (er * mur) / ((dout - din)/(float) m); + if (fc <= f) { + char text[256], txt[256]; + strcpy (text, ""); + while ((fc <= f) && (m<10)) { + sprintf(txt, "TM(n,%d) ",m); + strcat(text,txt); + m++; + fc = C0 / sqrt (er * mur) / ((dout - din)/(double) m); + } + setResult (3, text); + } +} diff --git a/qucs-transcalc/coax.h b/qucs-transcalc/coax.h new file mode 100644 index 00000000..1b558f4c --- /dev/null +++ b/qucs-transcalc/coax.h @@ -0,0 +1,61 @@ +/* + * microstrip.h - coaxial class definition + * + * Copyright (C) 2001 Gopal Narayanan + * Copyright (C) 2005 Stefan Jahn + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + +#ifndef __COAX_H +#define __COAX_H + +class coax : public transline { + public: + coax(); + ~coax(); + + private: + double er; /* dielectric constant */ + double tand; /* Dielectric Loss Tangent */ + double din; /* Inner diameter of cable */ + double dout; /* Outer diameter of cable */ + double l; /* Length of cable */ + double Z0; /* characteristic impedance */ + double ang_l; /* Electrical length in angle */ + double er_eff; /* Effective dielectric constant */ + double atten_dielectric; /* Loss in dielectric (dB) */ + double atten_cond; /* Loss in conductors (dB) */ + double fc; /* Cutoff frequency for higher order modes */ + + public: + void analyze (); + int synthesize (); + + private: + void get_coax_sub(); + void get_coax_comp(); + void get_coax_phys(); + void get_coax_elec(); + void fixdin(); + void fixdout(); + double alphad_coax(); + double alphac_coax(); + void show_results(); +}; + +#endif /* __COAX_H */ diff --git a/qucs-transcalc/coplanar.cpp b/qucs-transcalc/coplanar.cpp new file mode 100644 index 00000000..984a3c94 --- /dev/null +++ b/qucs-transcalc/coplanar.cpp @@ -0,0 +1,302 @@ +/* + * coplanar.cpp - coplanar class implementation + * + * Copyright (C) 2008 Michael Margraf + * Copyright (C) 2005, 2006 Stefan Jahn + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or (at + * your option) any later version. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + + +#include +#include +#include +#include +#include +#include + + +#include "units.h" +#include "transline.h" +#include "coplanar.h" + +coplanar::coplanar() : transline() +{ + backMetal = false; +} + +groundedCoplanar::groundedCoplanar() : coplanar() +{ + backMetal = true; +} + +// ------------------------------------------------------------------- +void coplanar::getProperties() +{ + f = getProperty ("Freq", UNIT_FREQ, FREQ_HZ); + w = getProperty ("W", UNIT_LENGTH, LENGTH_M); + s = getProperty ("S", UNIT_LENGTH, LENGTH_M); + len = getProperty ("L", UNIT_LENGTH, LENGTH_M); + h = getProperty ("H", UNIT_LENGTH, LENGTH_M); + t = getProperty ("T", UNIT_LENGTH, LENGTH_M); + + er = getProperty ("Er"); + tand = getProperty ("Tand"); + sigma = getProperty ("Cond"); + Z0 = getProperty ("Z0", UNIT_RES, RES_OHM); + ang_l = getProperty ("Ang_l", UNIT_ANG, ANG_RAD); +} + +// ------------------------------------------------------------------- +void coplanar::calc() +{ + skindepth = skin_depth(); + + // other local variables (quasi-static constants) + double k1, kk1, kpk1, k2, k3, q1, q2, q3 = 0, qz, er0 = 0; + double zl_factor; + + // compute the necessary quasi-static approx. (K1, K3, er(0) and Z(0)) + k1 = w / (w + s + s); + kk1 = ellipk (k1); + kpk1 = ellipk (sqrt (1 - k1 * k1)); + q1 = kk1 / kpk1; + + // backside is metal + if (backMetal) { + k3 = tanh ((pi / 4) * (w / h)) / tanh ((pi / 4) * (w + s + s) / h); + q3 = KoverKp(k3); + qz = 1 / (q1 + q3); + er0 = 1 + q3 * qz * (er - 1); + zl_factor = ZF0 / 2 * qz; + } + // backside is air + else { + k2 = sinh ((pi / 4) * (w / h)) / sinh ((pi / 4) * (w + s + s) / h); + q2 = KoverKp(k2); + er0 = 1 + (er - 1) / 2 * q2 / q1; + zl_factor = ZF0 / 4 / q1; + } + + // adds effect of strip thickness + if (t > 0) { + double d, ke, qe; + d = (t * 1.25 / pi) * (1 + log (4 * pi * w / t)); + + // modifies k1 accordingly (k1 = ke) + ke = k1 + (1 - k1 * k1) * d / 2 / s; + qe = KoverKp(ke); + // backside is metal + if (backMetal) { + qz = 1 / (qe + q3); + //er0 = 1 + q3 * qz * (er - 1); + zl_factor = ZF0 / 2 * qz; + } + // backside is air + else { + zl_factor = ZF0 / 4 / qe; + } + + // modifies er0 as well + er0 = er0 - (0.7 * (er0 - 1) * t / s) / (q1 + (0.7 * t / s)); + } + + // pre-compute square roots + double sr_er = sqrt (er); + double sr_er0 = sqrt (er0); + + // cut-off frequency of the TE0 mode + double fte = (C0 / 4) / (h * sqrt (er - 1)); + + // dispersion factor G + double p = log (w / h); + double u = 0.54 - (0.64 - 0.015 * p) * p; + double v = 0.43 - (0.86 - 0.54 * p) * p; + double G = exp (u * log (w / s) + v); + + // loss constant factors (computed only once for efficency sake) + double ac = 0; + if (t > 0) { + // equations by GHIONE + double n = (1 - k1) * 8 * pi / (t * (1 + k1)); + double a = w / 2; + double b = a + s; + ac = (pi + log (n * a)) / a + (pi + log (n * b)) / b; + } + double ac_factor = ac / (4 * ZF0 * kk1 * kpk1 * (1 - k1 * k1)); + double ad_factor = (er / (er - 1)) * tand * pi / C0; + + + // .................................................... + double sr_er_f = sr_er0; + + // add the dispersive effects to er0 + sr_er_f += (sr_er - sr_er0) / (1 + G * pow (f / fte, -1.8)); + + // for now, the loss are limited to strip losses (no radiation + // losses yet) losses in neper/length + atten_cond = 20.0 / log(10.0) * len + * ac_factor * sr_er0 * sqrt (pi * MU0 * f / sigma); + atten_dielectric = 20.0 / log(10.0) * len + * ad_factor * f * (sr_er_f * sr_er_f - 1) / sr_er_f; + + ang_l = 2.0 * pi * len * sr_er_f * f / C0; /* in radians */ + + er_eff = sr_er_f * sr_er_f; + Z0 = zl_factor / sr_er_f; +} + +// ------------------------------------------------------------------- +void coplanar::show_results() +{ + setProperty ("Z0", Z0, UNIT_RES, RES_OHM); + setProperty ("Ang_l", ang_l, UNIT_ANG, ANG_RAD); + + setResult (0, er_eff, ""); + setResult (1, atten_cond, "dB"); + setResult (2, atten_dielectric, "dB"); + + double val = convertProperty ("T", skindepth, UNIT_LENGTH, LENGTH_M); + setResult (3, val, getUnit ("T")); +} + +// ------------------------------------------------------------------- +void coplanar::analyze() +{ + getProperties(); + + /* compute coplanar parameters */ + calc(); + + /* print results in the subwindow */ + show_results(); +} + + +// ------------------------------------------------------------------- +int coplanar::synthesize() +{ + double Z0_dest, Z0_current, Z0_result, increment, slope, error; + int iteration; + const int maxiter = 100; + + getProperties(); + + /* required value of Z0 */ + Z0_dest = Z0; + + /* Newton's method */ + iteration = 0; + + /* compute coplanar parameters */ + calc(); + Z0_current = Z0; + + error = std::abs(Z0_dest - Z0_current); + + while (error > MAX_ERROR) { + iteration++; + if(isSelected ("W")) { + increment = w / 100.0; + w += increment; + } + else { + increment = s / 100.0; + s += increment; + } + /* compute coplanar parameters */ + calc(); + Z0_result = Z0; + /* f(w(n)) = Z0 - Z0(w(n)) */ + /* f'(w(n)) = -f'(Z0(w(n))) */ + /* f'(Z0(w(n))) = (Z0(w(n)) - Z0(w(n+delw))/delw */ + /* w(n+1) = w(n) - f(w(n))/f'(w(n)) */ + slope = (Z0_result - Z0_current) / increment; + slope = (Z0_dest - Z0_current) / slope - increment; + if(isSelected ("W")) + w += slope; + else + s += slope; + if (w <= 0.0) + w = increment; + if (s <= 0.0) + s = increment; + /* find new error */ + /* compute coplanar parameters */ + calc(); + Z0_current = Z0; + error = std::abs(Z0_dest - Z0_current); + if (iteration > maxiter) + break; + } + + setProperty ("W", w, UNIT_LENGTH, LENGTH_M); + setProperty ("S", s, UNIT_LENGTH, LENGTH_M); + /* calculate physical length */ + ang_l = getProperty ("Ang_l", UNIT_ANG, ANG_RAD); + len = C0 / f / sqrt(er_eff) * ang_l / 2.0 / pi; /* in m */ + setProperty ("L", len, UNIT_LENGTH, LENGTH_M); + + /* compute coplanar parameters */ + calc(); + + /* print results in the subwindow */ + show_results(); + + if (iteration > maxiter) + return -1; + else + return 0; +} + + + +/* ***************************************************************** + ********** ********** + ********** mathematical functions ********** + ********** ********** + ***************************************************************** */ + +/* The function computes the complete elliptic integral of first kind + K(k) using the arithmetic-geometric mean algorithm (AGM) found e.g. + in Abramowitz and Stegun (17.6.1). + Note that the argument of the function here is the elliptic modulus k + and not the parameter m=k^2 . */ +double coplanar::ellipk (double k) { + if ((k < 0.0) || (k >= 1.0)) + // we use only the range from 0 <= k < 1 + return std::numeric_limits::quiet_NaN(); + + double a = 1.0; + double b = sqrt(1-k*k); + double c = k; + + while (c > NR_EPSI) { + double tmp = (a + b) / 2.0; + c = (a - b) / 2.0; + b = sqrt(a * b); + a = tmp; + } + return (pi_over_2 / a); +} + +double coplanar::KoverKp(double k) { + if ((k < 0.0) || (k >= 1.0)) + return std::numeric_limits::quiet_NaN(); + + return (ellipk(k) / ellipk(sqrt(1-k*k))); +} diff --git a/qucs-transcalc/coplanar.h b/qucs-transcalc/coplanar.h new file mode 100644 index 00000000..e848b577 --- /dev/null +++ b/qucs-transcalc/coplanar.h @@ -0,0 +1,67 @@ +/* + * coplanar.h - microstrip class definition + * + * Copyright (C) 2008 Michael Margraf + * Copyright (C) 2005 Stefan Jahn + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + +#ifndef __COPLANAR_H +#define __COPLANAR_H + +class coplanar : public transline { + public: + coplanar(); + + private: + double er; /* dielectric constant */ + double h; /* height of substrate */ + double t; /* thickness of top metal */ + double tand; /* Dielectric Loss Tangent */ + double w; /* width of line */ + double s; /* width of gap between line and ground */ + double len; /* length of line */ + double Z0; /* characteristic impedance */ + double er_eff; /* effective dielectric constant */ + double ang_l; /* Electrical length in angle */ + double atten_dielectric; /* Loss in dielectric (dB) */ + double atten_cond; /* Loss in conductors (dB) */ + + public: + void analyze(); + int synthesize(); + + protected: + bool backMetal; + + private: + void calc(); + void show_results(); + void getProperties(); + + double ellipk(double); + double KoverKp(double); +}; + + +class groundedCoplanar : public coplanar { + public: + groundedCoplanar(); +}; + +#endif /* __COPLANAR_H */ diff --git a/qucs-transcalc/examples/CMakeLists.txt b/qucs-transcalc/examples/CMakeLists.txt new file mode 100644 index 00000000..6ca0bc36 --- /dev/null +++ b/qucs-transcalc/examples/CMakeLists.txt @@ -0,0 +1,11 @@ + +SET(EXAMPLES +hp.trc +ustrip50fr4.trc +ustrip_jansen.trc +cstrip_kj_1.trc +cstrip_kj_2.trc) + +# installation of transmission lines +INSTALL( FILES ${EXAMPLES} DESTINATION share/qucs-s/tline ) + diff --git a/qucs-transcalc/examples/Makefile.am b/qucs-transcalc/examples/Makefile.am new file mode 100644 index 00000000..ea83717e --- /dev/null +++ b/qucs-transcalc/examples/Makefile.am @@ -0,0 +1,35 @@ +## Process this file with automake to produce Makefile.in +# +# qucs-transcalc/examples/Makefile.am +# +# Automake input file. +# +# Copyright (C) 2005 Stefan Jahn +# +# This is free software; you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation; either version 2, or (at your option) +# any later version. +# +# This software is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this package; see the file COPYING. If not, write to +# the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, +# Boston, MA 02110-1301, USA. +# + +EXAMPLES = hp.trc ustrip50fr4.trc ustrip_jansen.trc cstrip_kj_1.trc \ + cstrip_kj_2.trc + +EXTRA_DIST = $(EXAMPLES) + +# installation of transmission lines +linedatadir = ${prefix}/share/qucs/tline +linedata_DATA = $(EXTRA_DIST) + +CLEANFILES = *~ +MAINTAINERCLEANFILES = Makefile.in diff --git a/qucs-transcalc/examples/cstrip_kj_1.trc b/qucs-transcalc/examples/cstrip_kj_1.trc new file mode 100644 index 00000000..388adeeb --- /dev/null +++ b/qucs-transcalc/examples/cstrip_kj_1.trc @@ -0,0 +1,21 @@ +# QucsTranscalc 0.0.8 cstrip_kj_1.trc +# Generated on Thu Dec 1 2005 at 20:28:35. +# It is not suggested to edit the file, use QucsTranscalc instead. + + + Er 2.35 NA + Mur 1 NA + H 0.79 mm + H_t 1e+20 mil + T 0 mil + Cond 4.1e+07 NA + Tand 0 NA + Rough 0 mil + Freq 30 GHz + W 1.7775 mm + S 0.079 mm + L 1 m + Z0e 79.2774 Ohm + Z0o 37.9688 Ohm + Ang_l 50847.5 Deg + diff --git a/qucs-transcalc/examples/cstrip_kj_2.trc b/qucs-transcalc/examples/cstrip_kj_2.trc new file mode 100644 index 00000000..77f39e6c --- /dev/null +++ b/qucs-transcalc/examples/cstrip_kj_2.trc @@ -0,0 +1,21 @@ +# QucsTranscalc 0.0.8 cstrip_kj_2.trc +# Generated on Thu Dec 1 2005 at 20:30:57. +# It is not suggested to edit the file, use QucsTranscalc instead. + + + Er 9.9 NA + Mur 1 NA + H 0.64 mm + H_t 1e+20 mil + T 5 um + Cond 4.1e+07 NA + Tand 0.0005 NA + Rough 0.25 um + Freq 3 GHz + W 0.6 mm + S 0.5 mm + L 1 m + Z0e 58.8883 Ohm + Z0o 41.3621 Ohm + Ang_l 9214.28 Deg + diff --git a/qucs-transcalc/examples/hp.trc b/qucs-transcalc/examples/hp.trc new file mode 100644 index 00000000..766e3e31 --- /dev/null +++ b/qucs-transcalc/examples/hp.trc @@ -0,0 +1,21 @@ +# QucsTranscalc 0.0.8 hp.trc +# Generated on Thu Dec 1 2005 at 20:02:07. +# It is not suggested to edit the file, use QucsTranscalc instead. + + + Er 9.9 NA + Mur 1 NA + H 645 um + H_t 3.9e+34 mil + T 5 um + Cond 4.1e+07 NA + Tand 0 NA + Rough 0 mil + Freq 2 GHz + W 609 um + S 64.5 mil + L 14.5521 mm + Z0e 52.4548 Ohm + Z0o 47.9635 Ohm + Ang_l 90.0165 Deg + diff --git a/qucs-transcalc/examples/ustrip50fr4.trc b/qucs-transcalc/examples/ustrip50fr4.trc new file mode 100644 index 00000000..b0fbf155 --- /dev/null +++ b/qucs-transcalc/examples/ustrip50fr4.trc @@ -0,0 +1,19 @@ +# QucsTranscalc 0.0.8 ustrip50fr4.trc +# Generated on Thu Dec 1 2005 at 19:53:13. +# It is not suggested to edit the file, use QucsTranscalc instead. + + + Er 4.3 NA + Mur 1 NA + H 210 um + H_t 1e+20 mil + T 30 um + Cond 4.1e+07 NA + Tand 0.025 NA + Rough 0 mil + Freq 1 GHz + W 380 um + L 1 mm + Z0 50.7326 Ohm + Ang_l 2.10486 Deg + diff --git a/qucs-transcalc/examples/ustrip_jansen.trc b/qucs-transcalc/examples/ustrip_jansen.trc new file mode 100644 index 00000000..a31181ca --- /dev/null +++ b/qucs-transcalc/examples/ustrip_jansen.trc @@ -0,0 +1,19 @@ +# QucsTranscalc 0.0.8 ustrip_jansen.trc +# Generated on Thu Dec 1 2005 at 19:56:38. +# It is not suggested to edit the file, use QucsTranscalc instead. + + + Er 9.9 NA + Mur 1 NA + H 0.64 mm + H_t 1e+20 mil + T 5 um + Cond 4.1e+07 NA + Tand 0.0005 NA + Rough 0.25 um + Freq 3 GHz + W 0.6 mm + L 1 m + Z0 50.4142 Ohm + Ang_l 9267.81 Deg + diff --git a/qucs-transcalc/helpdialog.cpp b/qucs-transcalc/helpdialog.cpp new file mode 100644 index 00000000..33d06344 --- /dev/null +++ b/qucs-transcalc/helpdialog.cpp @@ -0,0 +1,72 @@ +/*************************************************************************** + helpdialog.cpp + ------------------ + begin : Sun Apr 03 2005 + copyright : (C) 2005 by Stefan Jahn + email : stefan@lkcc.org + ***************************************************************************/ + +/*************************************************************************** + * * + * This program is free software; you can redistribute it and/or modify * + * it under the terms of the GNU General Public License as published by * + * the Free Software Foundation; either version 2 of the License, or * + * (at your option) any later version. * + * * + ***************************************************************************/ + +#ifdef HAVE_CONFIG_H +# include +#endif + +#include +#include +#include + +#include "helpdialog.h" + + +HelpDialog::HelpDialog(QWidget *parent) + : QDialog(parent) +{ + setWindowTitle("QucsTranscalc "+tr("Help")); + + // -------- set help text into dialog ------------ + QString s(tr("QucsTranscalc is an analysis and synthesis tool for " + "calculating the electrical and physical properties of " + "different kinds of RF and microwave transmission lines.")+ + "\n\n"+ + tr("For each type of transmission line, using dialog boxes, you can " + "enter values for the various parameters, and either calculate its " + "electrical properties, or use the given electrical requirements to " + "synthesize physical parameters of the required transmission line.")); + + + // -------- create dialog widgets ------------ + resize(350, 230); + + vLayout = new QVBoxLayout(this); + vLayout->setMargin(3); + vLayout->setSpacing(3); + + Text = new QTextEdit(s, this); + //Text->setTextFormat(Qt::PlainText); + Text->setReadOnly(true); + Text->setMinimumSize(300,200); + vLayout->addWidget(Text); + + QPushButton *ButtonClose = new QPushButton(tr("Dismiss"), this); + vLayout->addWidget(ButtonClose); + connect(ButtonClose, SIGNAL(clicked()), SLOT(slotClose())); + ButtonClose->setFocus(); +} + +HelpDialog::~HelpDialog() +{ + delete vLayout; +} + +void HelpDialog::slotClose() +{ + accept(); +} diff --git a/qucs-transcalc/helpdialog.h b/qucs-transcalc/helpdialog.h new file mode 100644 index 00000000..6aaafa5d --- /dev/null +++ b/qucs-transcalc/helpdialog.h @@ -0,0 +1,45 @@ +/*************************************************************************** + helpdialog.h + ------------------- + begin : Sun Apr 03 2005 + copyright : (C) 2005 by Stefan Jahn + email : stefan@lkcc.org + ***************************************************************************/ + +/*************************************************************************** + * * + * This program is free software; you can redistribute it and/or modify * + * it under the terms of the GNU General Public License as published by * + * the Free Software Foundation; either version 2 of the License, or * + * (at your option) any later version. * + * * + ***************************************************************************/ + +#ifndef HELPDIALOG_H +#define HELPDIALOG_H + +#include +#include + +class QTextEdit; +class QVBoxLayout; + +/** + *@author Stefan Jahn + */ + +class HelpDialog : public QDialog { + Q_OBJECT +public: + HelpDialog(QWidget *parent = 0); + ~HelpDialog(); + +private slots: + void slotClose(); + +private: + QVBoxLayout *vLayout; + QTextEdit *Text; +}; + +#endif diff --git a/qucs-transcalc/main.cpp b/qucs-transcalc/main.cpp new file mode 100644 index 00000000..2a1cc69e --- /dev/null +++ b/qucs-transcalc/main.cpp @@ -0,0 +1,166 @@ +/*************************************************************************** + main.cpp - description + ------------------- + begin : Sun Feb 27 2005 + copyright : (C) 2005 by Stefan Jahn + email : stefan@lkcc.org + ***************************************************************************/ + +/*************************************************************************** + * * + * This program is free software; you can redistribute it and/or modify * + * it under the terms of the GNU General Public License as published by * + * the Free Software Foundation; either version 2 of the License, or * + * (at your option) any later version. * + * * + ***************************************************************************/ + +#ifdef HAVE_CONFIG_H +# include +#endif + +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "qucstrans.h" + +tQucsSettings QucsSettings; + +extern struct TransUnit TransUnits[]; + + +// ######################################################################### +// Loads the settings file and stores the settings. +bool loadSettings() +{ + QSettings settings("qucs","qucs"); + settings.beginGroup("QucsTranscalc"); + if(settings.contains("x"))QucsSettings.x=settings.value("x").toInt(); + if(settings.contains("y"))QucsSettings.y=settings.value("y").toInt(); + if(settings.contains("dx"))QucsSettings.dx=settings.value("dx").toInt(); + if(settings.contains("dy"))QucsSettings.dy=settings.value("dy").toInt(); + if(settings.contains("Mode"))QucsSettings.Mode=settings.value("Mode").toString(); + if(settings.contains("FreqUnit"))QucsSettings.freq_unit=settings.value("FreqUnit").toInt(); + if(settings.contains("LengthUnit"))QucsSettings.length_unit=settings.value("LengthUnit").toInt(); + if(settings.contains("ResUnit"))QucsSettings.res_unit=settings.value("ResUnit").toInt(); + if(settings.contains("AngUnit"))QucsSettings.ang_unit=settings.value("AngUnit").toInt(); + + settings.endGroup(); + if(settings.contains("font"))QucsSettings.font.fromString(settings.value("font").toString()); + if(settings.contains("Language"))QucsSettings.Language=settings.value("Language").toString(); + if(settings.contains("QucsHomeDir")) + if(settings.value("QucsHomeDir").toString() != "") + QucsSettings.QucsHomeDir.setPath(settings.value("QucsHomeDir").toString()); + + QucsSettings.QucsWorkDir = QucsSettings.QucsHomeDir; + + return true; +} + + +// ######################################################################### +// Saves the settings in the settings file. +bool saveApplSettings(QucsTranscalc *qucs) +{ + QSettings settings ("qucs","qucs"); + settings.beginGroup("QucsTranscalc"); + settings.setValue("x", qucs->x()); + settings.setValue("y", qucs->y()); + settings.setValue("dx", qucs->width()); + settings.setValue("dy", qucs->height()); + settings.setValue("Mode", qucs->getMode()); + settings.setValue("FreqUnit", TransUnits[0].units[QucsSettings.freq_unit]); + settings.setValue("LengthUnit", TransUnits[1].units[QucsSettings.length_unit]); + settings.setValue("ResUnit", TransUnits[2].units[QucsSettings.res_unit]); + settings.setValue("AngUnit", TransUnits[3].units[QucsSettings.ang_unit]); + settings.endGroup(); + + return true; +} + + +// ######################################################################### +// ########## ########## +// ########## Program Start ########## +// ########## ########## +// ######################################################################### + +int main(int argc, char *argv[]) +{ + QApplication a(argc, argv); + + // apply default settings + QucsSettings.x = 100; + QucsSettings.y = 50; + QucsSettings.dx = 540; + QucsSettings.dy = 400; + QucsSettings.font = QFont("Helvetica", 12); + QucsSettings.length_unit = 0; + QucsSettings.res_unit = 0; + QucsSettings.ang_unit = 0; + QucsSettings.freq_unit = 0; + QucsSettings.QucsHomeDir.setPath(QDir::homePath() + "/.qucs"); + + // is application relocated? + char * var = getenv ("QUCSDIR"); + QDir QucsDir; + if (var != NULL) { + QucsDir = QDir (var); + QString QucsDirStr = QucsDir.canonicalPath (); + QucsSettings.LangDir = + QDir::toNativeSeparators(QucsDirStr + "/share/qucs/lang/"); + } else { + QString QucsApplicationPath = QCoreApplication::applicationDirPath(); +#ifdef __APPLE__ + QucsDir = QDir(QucsApplicationPath.section("/bin",0,0)); +#else + QucsDir = QDir(QucsApplicationPath); + QucsDir.cdUp(); +#endif + QucsSettings.LangDir = QucsDir.canonicalPath() + "/share/qucs/lang/"; + } + loadSettings(); + + a.setFont(QucsSettings.font); + + QTranslator tor( 0 ); + QString lang = QucsSettings.Language; + if(lang.isEmpty()) + lang = QString(QLocale::system().name()); + tor.load( QString("qucs_") + lang, QucsSettings.LangDir); + a.installTranslator( &tor ); + + QucsTranscalc *qucs = new QucsTranscalc(); + qucs->raise(); + qucs->resize(QucsSettings.dx, QucsSettings.dy); // size and position ... + qucs->move(QucsSettings.x, QucsSettings.y); // ... before "show" !!! + qucs->show(); + + // load file with all the GUI input values from the Qucs Home + qucs->loadFile(QucsSettings.QucsHomeDir.filePath("transrc")); + qucs->setMode(QucsSettings.Mode); + + // optional file argument + if (argc > 1) { + int _mode = 0; + QString File = argv[1]; + qucs->loadFile(File,&_mode); + } + + int result = a.exec(); + saveApplSettings(qucs); + // save file with all the GUI input values in the Qucs Home + qucs->saveModes(QucsSettings.QucsHomeDir.filePath("transrc")); + delete qucs; + return result; +} diff --git a/qucs-transcalc/microstrip.cpp b/qucs-transcalc/microstrip.cpp new file mode 100644 index 00000000..151cbb65 --- /dev/null +++ b/qucs-transcalc/microstrip.cpp @@ -0,0 +1,578 @@ +/* + * microstrip.cpp - microstrip class implementation + * + * Copyright (C) 2001 Gopal Narayanan + * Copyright (C) 2002 Claudio Girardi + * Copyright (C) 2005, 2006 Stefan Jahn + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or (at + * your option) any later version. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + + +/* microstrip.c - Puts up window for microstrip and + * performs the associated calculations + * Based on the original microstrip.c by Gopal Narayanan + */ + +#include +#include +#include +#include + +#include "units.h" +#include "transline.h" +#include "microstrip.h" + +microstrip::microstrip() : transline() +{ +} + +microstrip::~microstrip() +{ +} + +/* + * Z0_homogeneous() - compute the impedance for a stripline in a + * homogeneous medium, without cover effects + */ +double microstrip::Z0_homogeneous(double u) +{ + double f, Z0; + f = 6.0 + (2.0 * pi - 6.0) * exp(-pow(30.666 / u, 0.7528)); + Z0 = (ZF0 / (2.0 * pi)) * log(f / u + sqrt(1.0 + 4.0 / (u * u))); + return Z0; +} + + +/* + * delta_Z0_cover() - compute the cover effect on impedance for a + * stripline in a homogeneous medium + */ +double microstrip::delta_Z0_cover(double u, double h2h) +{ + double P, Q; + double h2hp1; + h2hp1 = 1.0 + h2h; + P = 270.0 * (1.0 - tanh(1.192 + 0.706 * sqrt(h2hp1) - 1.389 / h2hp1)); + Q = 1.0109 - atanh((0.012 * u + 0.177 * u * u - 0.027 * u * u * u) / (h2hp1 * h2hp1)); + return (P * Q); +} + + +/* + * filling_factor() - compute the filling factor for a microstrip + * without cover and zero conductor thickness + */ +double microstrip::filling_factor(double u, double e_r) +{ + double a, b, q_inf; + double u2, u3, u4; + u2 = u * u; + u3 = u2 * u; + u4 = u3 * u; + a = 1.0 + log((u4 + u2 / 2704) / (u4 + 0.432)) / 49.0 + log(1.0 + u3 / 5929.741) / 18.7; + b = 0.564 * pow((e_r - 0.9) / (e_r + 3.0), 0.053); + q_inf = pow(1.0 + 10.0 / u, -a * b); + return q_inf; +} + + +/* + * delta_q_cover() - compute the cover effect on filling factor + */ +double microstrip::delta_q_cover(double h2h) +{ + double q_c; + q_c = tanh(1.043 + 0.121 * h2h - 1.164 / h2h); + return q_c; +} + + +/* + * delta_q_thickness() - compute the thickness effect on filling factor + */ +double microstrip::delta_q_thickness(double u, double t_h) +{ + double q_t; + q_t = (2.0 * log(2.0) / pi) * (t_h / sqrt(u)); + return q_t; +} + + +/* + * e_r_effective() - compute effective dielectric constant from + * material e_r and filling factor + */ +double microstrip::e_r_effective(double e_r, double q) +{ + double e_r_eff; + e_r_eff = 0.5 * (e_r + 1.0) + 0.5 * q * (e_r - 1.0); + return e_r_eff; +} + + +/* + * delta_u_thickness - compute the thickness effect on normalized width + */ +double microstrip::delta_u_thickness(double u, double t_h, double e_r) +{ + double delta_u; + if (t_h > 0.0) { + /* correction for thickness for a homogeneous microstrip */ + delta_u = (t_h / pi) * log(1.0 + (4.0 * e) * pow(tanh(sqrt(6.517 * u)), 2.0) / t_h); + /* correction for strip on a substrate with relative permettivity e_r */ + delta_u = 0.5 * delta_u * (1.0 + 1.0 / cosh(sqrt(e_r - 1.0))); + } else { + delta_u = 0.0; + } + return delta_u; +} + + +/* + * microstrip_Z0() - compute microstrip static impedance + */ +void microstrip::microstrip_Z0() +{ + double e_r, h2, h2h, u, t_h; + double Z0_h_r, Z0; + double delta_u_1, delta_u_r, q_inf, q_c, q_t, e_r_eff, e_r_eff_t, q; + + e_r = er; + h2 = ht; + h2h = h2 / h; + u = w / h; + t_h = t / h; + + /* compute normalized width correction for e_r = 1.0 */ + delta_u_1 = delta_u_thickness(u, t_h, 1.0); + /* compute homogeneous stripline impedance */ + Z0_h_1 = Z0_homogeneous(u + delta_u_1); + /* compute normalized width corection */ + delta_u_r = delta_u_thickness(u, t_h, e_r); + u += delta_u_r; + /* compute homogeneous stripline impedance */ + Z0_h_r = Z0_homogeneous(u); + + /* filling factor, with width corrected for thickness */ + q_inf = filling_factor(u, e_r); + /* cover effect */ + q_c = delta_q_cover(h2h); + /* thickness effect */ + q_t = delta_q_thickness(u, t_h); + /* resultant filling factor */ + q = (q_inf - q_t) * q_c; + + /* e_r corrected for thickness and non homogeneous material */ + e_r_eff_t = e_r_effective(e_r, q); + + /* effective dielectric constant */ + e_r_eff = e_r_eff_t * pow(Z0_h_1 / Z0_h_r, 2.0); + + /* characteristic impedance, corrected for thickness, cover */ + /* and non homogeneous material */ + Z0 = Z0_h_r / sqrt(e_r_eff_t); + + w_eff = u * h; + er_eff_0 = e_r_eff; + Z0_0 = Z0; +} + + +/* + * e_r_dispersion() - computes the dispersion correction factor for + * the effective permeability + */ +double microstrip::e_r_dispersion(double u, double e_r, double f_n) +{ + double P_1, P_2, P_3, P_4, P; + + P_1 = 0.27488 + u * (0.6315 + 0.525 / pow(1.0 + 0.0157 * f_n, 20.0)) - 0.065683 * exp(-8.7513 * u); + P_2 = 0.33622 * (1.0 - exp(-0.03442 * e_r)); + P_3 = 0.0363 * exp(-4.6 * u) * (1.0 - exp(-pow(f_n / 38.7, 4.97))); + P_4 = 1.0 + 2.751 * (1.0 - exp(-pow(e_r / 15.916, 8.0))); + + P = P_1 * P_2 * pow((P_3 * P_4 + 0.1844) * f_n, 1.5763); + + return P; +} + + +/* + * Z0_dispersion() - computes the dispersion correction factor for the + * characteristic impedance + */ +double microstrip::Z0_dispersion(double u, double e_r, double e_r_eff_0, double e_r_eff_f, double f_n) +{ + double R_1, R_2, R_3, R_4, R_5, R_6, R_7, R_8, R_9, R_10, R_11, R_12, R_13, R_14, R_15, R_16, R_17, D, tmpf; + + R_1 = 0.03891 * pow(e_r, 1.4); + R_2 = 0.267 * pow(u, 7.0); + R_3 = 4.766 * exp(-3.228 * pow(u, 0.641)); + R_4 = 0.016 + pow(0.0514 * e_r, 4.524); + R_5 = pow(f_n / 28.843, 12.0); + R_6 = 22.2 * pow(u, 1.92); + R_7 = 1.206 - 0.3144 * exp(-R_1) * (1.0 - exp(-R_2)); + R_8 = 1.0 + 1.275 * (1.0 - exp(-0.004625 * R_3 * pow(e_r, 1.674) * pow(f_n / 18.365, 2.745))); + tmpf = pow(e_r - 1.0, 6.0); + R_9 = 5.086 * R_4 * (R_5 / (0.3838 + 0.386 * R_4)) * (exp(-R_6) / (1.0 + 1.2992 * R_5)) * (tmpf / (1.0 + 10.0 * tmpf)); + R_10 = 0.00044 * pow(e_r, 2.136) + 0.0184; + tmpf = pow(f_n / 19.47, 6.0); + R_11 = tmpf / (1.0 + 0.0962 * tmpf); + R_12 = 1.0 / (1.0 + 0.00245 * u * u); + R_13 = 0.9408 * pow(e_r_eff_f, R_8) - 0.9603; + R_14 = (0.9408 - R_9) * pow(e_r_eff_0, R_8) - 0.9603; + R_15 = 0.707 * R_10 * pow(f_n / 12.3, 1.097); + R_16 = 1.0 + 0.0503 * e_r * e_r * R_11 * (1.0 - exp(-pow(u / 15.0, 6.0))); + R_17 = R_7 * (1.0 - 1.1241 * (R_12 / R_16) * exp(-0.026 * pow(f_n, 1.15656) - R_15)); + + D = pow(R_13 / R_14, R_17); + + return D; +} + + +/* + * dispersion() - compute frequency dependent parameters of + * microstrip + */ +void microstrip::dispersion() +{ + double e_r, e_r_eff_0; + double u, f_n, P, e_r_eff_f, D, Z0_f; + + e_r = er; + e_r_eff_0 = er_eff_0; + u = w / h; + + /* normalized frequency [GHz * mm] */ + f_n = f * h / 1e06; + + P = e_r_dispersion(u, e_r, f_n); + /* effective dielectric constant corrected for dispersion */ + e_r_eff_f = e_r - (e_r - e_r_eff_0) / (1.0 + P); + + D = Z0_dispersion(u, e_r, e_r_eff_0, e_r_eff_f, f_n); + Z0_f = Z0_0 * D; + + er_eff = e_r_eff_f; + Z0 = Z0_f; +} + + +/* + * conductor_losses() - compute microstrip conductor losses per unit + * length + */ +double microstrip::conductor_losses() +{ + double e_r_eff_0, delta; + double K, R_s, Q_c, alpha_c; + + e_r_eff_0 = er_eff_0; + delta = skindepth; + + if (f > 0.0) { + /* current distribution factor */ + K = exp(-1.2 * pow(Z0_h_1 / ZF0, 0.7)); + /* skin resistance */ + R_s = 1.0 / (sigma * delta); + + /* correction for surface roughness */ + R_s *= 1.0 + ((2.0 / pi) * atan(1.40 * pow((rough / delta), 2.0))); + /* strip inductive quality factor */ + Q_c = (pi * Z0_h_1 * w * f) / (R_s * C0 * K); + alpha_c = (20.0 * pi / log(10.0)) * f * sqrt(e_r_eff_0) / (C0 * Q_c); + } else { + alpha_c = 0.0; + } + + return alpha_c; +} + + +/* + * dielectric_losses() - compute microstrip dielectric losses per unit + * length + */ +double microstrip::dielectric_losses() +{ + double e_r, e_r_eff_0; + double alpha_d; + + e_r = er; + e_r_eff_0 = er_eff_0; + + alpha_d = (20.0 * pi / log(10.0)) * (f / C0) * (e_r / sqrt(e_r_eff_0)) * ((e_r_eff_0 - 1.0) / (e_r - 1.0)) * tand; + + return alpha_d; +} + + +/* + * attenuation() - compute attenuation of microstrip + */ +void microstrip::attenuation() +{ + skindepth = skin_depth(); + + atten_cond = conductor_losses() * l; + atten_dielectric = dielectric_losses() * l; +} + + +/* + * mur_eff_ms() - returns effective magnetic permeability + */ +void microstrip::mur_eff_ms() +{ + double mureff; + + mureff = (2.0 * mur) / ((1.0 + mur) + ((1.0 - mur) * pow((1.0 + (10.0 * h / w)), -0.5))); + + mur_eff = mureff; +} + + +/* + * synth_width - calculate width given Z0 and e_r + */ +double microstrip::synth_width() +{ + double e_r, a, b; + double w_h, w; + + + e_r = er; + + + a = ((Z0 / ZF0 / 2 / pi) * sqrt((e_r + 1) / 2.)) + ((e_r - 1) / (e_r + 1) * (0.23 + (0.11 / e_r))); + b = ZF0 / 2 * pi / (Z0 * sqrt(e_r)); + + if (a > 1.52) { + w_h = 8 * exp(a) / (exp(2. * a) - 2); + } else { + w_h = (2. / pi) * (b - 1. - log((2 * b) - 1.) + ((e_r - 1) / (2 * e_r)) * (log(b - 1.) + 0.39 - 0.61 / e_r)); + } + + if (h > 0.0) { + w = w_h * h; + return w; + } else { + w = 0; + } + return w; +} + + +/* + * line_angle() - calculate microstrip length in radians + */ +void microstrip::line_angle() +{ + double e_r_eff; + double v, lambda_g; + + e_r_eff = er_eff; + + /* velocity */ + v = C0 / sqrt(e_r_eff * mur_eff); + /* wavelength */ + lambda_g = v / f; + /* electrical angles */ + ang_l = 2.0 * pi * l / lambda_g; /* in radians */ +} + + +void microstrip::calc() +{ + /* effective permeability */ + mur_eff_ms(); + /* static impedance */ + microstrip_Z0(); + /* calculate freq dependence of er and Z0 */ + dispersion(); + /* calculate electrical lengths */ + line_angle(); + /* calculate losses */ + attenuation(); +} + + +/* + * get_microstrip_sub () - get and assign microstrip substrate + * parameters into microstrip structure + */ +void microstrip::get_microstrip_sub() +{ + er = getProperty ("Er"); + mur = getProperty ("Mur"); + h = getProperty ("H", UNIT_LENGTH, LENGTH_M); + ht = getProperty ("H_t", UNIT_LENGTH, LENGTH_M); + t = getProperty ("T", UNIT_LENGTH, LENGTH_M); + sigma = getProperty ("Cond"); + tand = getProperty ("Tand"); + rough = getProperty ("Rough", UNIT_LENGTH, LENGTH_M); +} + +/* + * get_microstrip_comp() - get and assign microstrip component + * parameters into microstrip structure + */ +void microstrip::get_microstrip_comp() +{ + f = getProperty ("Freq", UNIT_FREQ, FREQ_HZ); +} + +/* + * get_microstrip_elec() - get and assign microstrip electrical + * parameters into microstrip structure + */ +void microstrip::get_microstrip_elec() +{ + Z0 = getProperty ("Z0", UNIT_RES, RES_OHM); + ang_l = getProperty ("Ang_l", UNIT_ANG, ANG_RAD); +} + + +/* + * get_microstrip_phys() - get and assign microstrip physical + * parameters into microstrip structure + */ +void microstrip::get_microstrip_phys() +{ + w = getProperty ("W", UNIT_LENGTH, LENGTH_M); + l = getProperty ("L", UNIT_LENGTH, LENGTH_M); +} + + +void microstrip::show_results() +{ + setProperty ("Z0", Z0, UNIT_RES, RES_OHM); + setProperty ("Ang_l", ang_l, UNIT_ANG, ANG_RAD); + + setResult (0, er_eff, ""); + setResult (1, atten_cond, "dB"); + setResult (2, atten_dielectric, "dB"); + + double val = convertProperty ("T", skindepth, UNIT_LENGTH, LENGTH_M); + setResult (3, val, getUnit ("T")); +} + +/* + * analysis function + */ +void microstrip::analyze() +{ + /* Get and assign substrate parameters */ + get_microstrip_sub(); + + /* Get and assign component parameters */ + get_microstrip_comp(); + + /* Get and assign physical parameters */ + get_microstrip_phys(); + + /* compute microstrip parameters */ + calc(); + + /* print results in the subwindow */ + show_results(); +} + + +/* + * synthesis function + */ +int microstrip::synthesize() +{ + double Z0_dest, Z0_current, Z0_result, increment, slope, error; + int iteration; + const int maxiter = 100; + + /* Get and assign substrate parameters */ + get_microstrip_sub(); + + /* Get and assign component parameters */ + get_microstrip_comp(); + + /* Get and assign electrical parameters */ + get_microstrip_elec(); + + /* Get and assign physical parameters */ + /* at present it is required only for getting strips length */ + get_microstrip_phys(); + + + /* calculate width and use for initial value in Newton's method */ + w = synth_width(); + + /* required value of Z0 */ + Z0_dest = Z0; + + /* Newton's method */ + iteration = 0; + + /* compute microstrip parameters */ + calc(); + Z0_current = Z0; + + error = std::abs(Z0_dest - Z0_current); + + while (error > MAX_ERROR) { + iteration++; + increment = (w / 100.0); + w += increment; + /* compute microstrip parameters */ + calc(); + Z0_result = Z0; + /* f(w(n)) = Z0 - Z0(w(n)) */ + /* f'(w(n)) = -f'(Z0(w(n))) */ + /* f'(Z0(w(n))) = (Z0(w(n)) - Z0(w(n+delw))/delw */ + /* w(n+1) = w(n) - f(w(n))/f'(w(n)) */ + slope = (Z0_result - Z0_current) / increment; + /* printf("%g\n",slope); */ + w += (Z0_dest - Z0_current) / slope - increment; + /* printf("ms->w = %g\n", ms->w); */ + /* find new error */ + /* compute microstrip parameters */ + calc(); + Z0_current = Z0; + error = std::abs(Z0_dest - Z0_current); + /* printf("Iteration = %d\n",iteration); + printf("w = %g\t Z0 = %g\n",ms->w, Z0_current); */ + if (iteration > maxiter) + break; + } + + setProperty ("W", w, UNIT_LENGTH, LENGTH_M); + /* calculate physical length */ + ang_l = getProperty ("Ang_l", UNIT_ANG, ANG_RAD); + l = C0 / f / sqrt(er_eff * mur_eff) * ang_l / 2.0 / pi; /* in m */ + setProperty ("L", l, UNIT_LENGTH, LENGTH_M); + + /* compute microstrip parameters */ + calc(); + + /* print results in the subwindow */ + show_results(); + + if (iteration > maxiter) + return -1; + else + return 0; +} diff --git a/qucs-transcalc/microstrip.h b/qucs-transcalc/microstrip.h new file mode 100644 index 00000000..da500a10 --- /dev/null +++ b/qucs-transcalc/microstrip.h @@ -0,0 +1,93 @@ +/* + * microstrip.h - microstrip class definition + * + * Copyright (C) 2001 Gopal Narayanan + * Copyright (C) 2005 Stefan Jahn + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + +#ifndef __MICROSTRIP_H +#define __MICROSTRIP_H + +class microstrip : public transline { + public: + microstrip(); + ~microstrip(); + + friend class c_microstrip; + + private: + double er; /* dielectric constant */ + double h; /* height of substrate */ + double ht; /* height to the top of box */ + double t; /* thickness of top metal */ + double tand; /* Dielectric Loss Tangent */ + double rough; /* Roughness of top metal */ + double w; /* width of line */ + double l; /* length of line */ + double Z0_0; /* static characteristic impedance */ + double Z0; /* characteristic impedance */ + double ang_l; /* Electrical length in angle */ + double er_eff_0; /* Static effective dielectric constant */ + double er_eff; /* Effective dielectric constant */ + double mur_eff; /* Effective mag. permeability */ + double w_eff; /* Effective width of line */ + double atten_dielectric; /* Loss in dielectric (dB) */ + double atten_cond; /* Loss in conductors (dB) */ + + /* private params */ + double Z0_h_1; /* homogeneous stripline impedance */ + + public: + void analyze(); + int synthesize(); + + private: + double er_eff_freq(); + double alpha_c(); + double alpha_c_roughness(); + double alpha_dielectric(); + double char_impedance_ht(); + double synth_width(); + double ereff_dispersion(); + double Z0_dispersion(); + double Z0_homogeneous(double); + double delta_Z0_cover(double, double); + double filling_factor(double, double); + double delta_q_cover(double); + double delta_q_thickness(double, double); + double e_r_effective(double, double); + double delta_u_thickness(double, double, double); + double e_r_dispersion(double, double, double); + double Z0_dispersion(double, double, double, double, double); + double conductor_losses(); + double dielectric_losses(); + void microstrip_Z0(); + void dispersion(); + void attenuation(); + void mur_eff_ms(); + void line_angle(); + void calc(); + void get_microstrip_sub(); + void get_microstrip_comp(); + void get_microstrip_elec(); + void get_microstrip_phys(); + void show_results(); +}; + +#endif /* __MICROSTRIP_H */ diff --git a/qucs-transcalc/optionsdialog.cpp b/qucs-transcalc/optionsdialog.cpp new file mode 100644 index 00000000..8968b86c --- /dev/null +++ b/qucs-transcalc/optionsdialog.cpp @@ -0,0 +1,109 @@ +/*************************************************************************** + optionsdialog.cpp + -------------------- + begin : Sun Apr 03 2005 + copyright : (C) 2005 by Stefan Jahn + email : stefan@lkcc.org + ***************************************************************************/ + +/*************************************************************************** + * * + * This program is free software; you can redistribute it and/or modify * + * it under the terms of the GNU General Public License as published by * + * the Free Software Foundation; either version 2 of the License, or * + * (at your option) any later version. * + * * + ***************************************************************************/ + +#ifdef HAVE_CONFIG_H +# include +#endif + +#include +#include +#include +#include +#include + +#include "optionsdialog.h" +#include "qucstrans.h" + +extern struct TransUnit TransUnits[]; + +OptionsDialog::OptionsDialog(QWidget *parent) + : QDialog(parent) +{ + setWindowTitle("QucsTranscalc "+tr("Options")); + + // -------- create dialog widgets ------------ + vLayout = new QVBoxLayout(this); + vLayout->setMargin(3); + vLayout->setSpacing(3); + + QGroupBox * unitsGroup = new QGroupBox(tr("Units")); + QVBoxLayout * l = new QVBoxLayout(); + l->setSpacing(3); + QLabel * lfr = new QLabel(tr("Frequency")); + lfr->setAlignment (Qt::AlignRight); + l->addWidget(lfr); + QLabel * lle = new QLabel(tr("Length")); + lle->setAlignment (Qt::AlignRight); + l->addWidget(lle); + QLabel * lre = new QLabel(tr("Resistance")); + lre->setAlignment (Qt::AlignRight); + l->addWidget(lre); + QLabel * lan = new QLabel(tr("Angle")); + lan->setAlignment (Qt::AlignRight); + l->addWidget(lan); + QVBoxLayout * r = new QVBoxLayout(); + r->setSpacing(3); + for (int j = 0; j < 4; j++) { + units[j] = new QComboBox(); //?r); + r->addWidget(units[j]); + for (int i = 0; TransUnits[j].units[i] != NULL; i++) + units[j]->addItem(TransUnits[j].units[i]); + } + units[0]->setCurrentIndex(QucsSettings.freq_unit); + units[1]->setCurrentIndex(QucsSettings.length_unit); + units[2]->setCurrentIndex(QucsSettings.res_unit); + units[3]->setCurrentIndex(QucsSettings.ang_unit); + + QHBoxLayout * h1 = new QHBoxLayout(); + h1->addLayout(l); + h1->addLayout(r); + unitsGroup->setLayout(h1); + + vLayout->addWidget(unitsGroup); + + QHBoxLayout * h2 = new QHBoxLayout(); + + QPushButton *ButtonSave = new QPushButton(tr("Save as Default")); + connect(ButtonSave, SIGNAL(clicked()), SLOT(slotSave())); + + QPushButton *ButtonClose = new QPushButton(tr("Dismiss")); + connect(ButtonClose, SIGNAL(clicked()), SLOT(slotClose())); + ButtonClose->setFocus(); + + h2->addWidget(ButtonSave); + h2->addWidget(ButtonClose); + vLayout->addLayout(h2); +} + +OptionsDialog::~OptionsDialog() +{ + delete vLayout; +} + +void OptionsDialog::slotClose() +{ + accept(); +} + +void OptionsDialog::slotSave() +{ + QucsSettings.freq_unit = units[0]->currentIndex(); + QucsSettings.length_unit = units[1]->currentIndex(); + QucsSettings.res_unit = units[2]->currentIndex(); + QucsSettings.ang_unit = units[3]->currentIndex(); + accept(); +} diff --git a/qucs-transcalc/optionsdialog.h b/qucs-transcalc/optionsdialog.h new file mode 100644 index 00000000..3258fee6 --- /dev/null +++ b/qucs-transcalc/optionsdialog.h @@ -0,0 +1,46 @@ +/*************************************************************************** + optionsdialog.h + ------------------- + begin : Sun Apr 03 2005 + copyright : (C) 2005 by Stefan Jahn + email : stefan@lkcc.org + ***************************************************************************/ + +/*************************************************************************** + * * + * This program is free software; you can redistribute it and/or modify * + * it under the terms of the GNU General Public License as published by * + * the Free Software Foundation; either version 2 of the License, or * + * (at your option) any later version. * + * * + ***************************************************************************/ + +#ifndef OPTIONSDIALOG_H +#define OPTIONSDIALOG_H + +#include +#include + +class QVBoxLayout; +class QComboBox; + +/** + *@author Stefan Jahn + */ + +class OptionsDialog : public QDialog { + Q_OBJECT +public: + OptionsDialog(QWidget *parent = 0); + ~OptionsDialog(); + +private slots: + void slotClose(); + void slotSave(); + +private: + QVBoxLayout *vLayout; + QComboBox *units[4]; +}; + +#endif diff --git a/qucs-transcalc/qucstrans.1.cmake.in b/qucs-transcalc/qucstrans.1.cmake.in new file mode 100644 index 00000000..6b957dad --- /dev/null +++ b/qucs-transcalc/qucstrans.1.cmake.in @@ -0,0 +1,52 @@ +.TH QucsTranscalc "1" "April 2005" "Debian/GNU Linux" "User Commands" +.SH NAME +QucsTranscalc \- A transmission line calculator. +.SH SYNOPSIS +.B qucstrans +[\fIOPTION\fR] ... +.SH DESCRIPTION + +\fBQucs\fR is an integrated circuit simulator which means you are able +to setup a circuit with a graphical user interface (GUI) and simulate +the large-signal, small-signal and noise behaviour of the circuit. +After that simulation has finished you can view the simulation results +on a presentation page or window. + +The software aims to support all kinds of circuit simulation types, +e.g. DC, AC, S-parameter, harmonic balance analysis, noise analysis, +etc. + +\fBQucsTranscalc\fR is the transmission line tool used by Qucs. The +application was somewhat inspired by \fBtranscalc\fR by Claudio +Girardi and Gopal Narayanan. It is based on transcalc's latest CVS +version with some minor fixes incorporated. The GUI has been ported +from the GIMP toolkit (GTK) to Qt. + +It is an analysis and synthesis tool for calculating the electrical +and physical properties of different kinds of RF and microwave +transmission lines. + +For each type of transmission line, using dialog boxes, the user can +enter values for the various parameters, and either calculate its +electrical properties, or use the given electrical requirements to +synthesize physical parameters of the required transmission line. + +Available transmission lines are: Microstrip, Rectangular Waveguide, +Coaxial Line, Coplanar and Coupled Microstrips. + +.SH AVAILABILITY +The latest version of Qucs can always be obtained from +\fB${QUCS_URL}\fR +.SH "REPORTING BUGS" +Known bugs are documented within the BUGS file. Report bugs to +\fB${QUCS_BUGREPORT}\fR +.SH COPYRIGHT +Copyright \(co 2005 Stefan Jahn +.PP +This is free software; see the source for copying conditions. There is NO +warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. +.SH AUTHORS +Written by Gopal Narayanan , Claudio Girardi +, Michael Margraf + and Stefan Jahn +. diff --git a/qucs-transcalc/qucstrans.1.in b/qucs-transcalc/qucstrans.1.in new file mode 100644 index 00000000..08e7014d --- /dev/null +++ b/qucs-transcalc/qucstrans.1.in @@ -0,0 +1,52 @@ +.TH QucsTranscalc "1" "April 2005" "Debian/GNU Linux" "User Commands" +.SH NAME +QucsTranscalc \- A transmission line calculator. +.SH SYNOPSIS +.B qucstrans +[\fIOPTION\fR] ... +.SH DESCRIPTION + +\fBQucs\fR is an integrated circuit simulator which means you are able +to setup a circuit with a graphical user interface (GUI) and simulate +the large-signal, small-signal and noise behaviour of the circuit. +After that simulation has finished you can view the simulation results +on a presentation page or window. + +The software aims to support all kinds of circuit simulation types, +e.g. DC, AC, S-parameter, harmonic balance analysis, noise analysis, +etc. + +\fBQucsTranscalc\fR is the transmission line tool used by Qucs. The +application was somewhat inspired by \fBtranscalc\fR by Claudio +Girardi and Gopal Narayanan. It is based on transcalc's latest CVS +version with some minor fixes incorporated. The GUI has been ported +from the GIMP toolkit (GTK) to Qt. + +It is an analysis and synthesis tool for calculating the electrical +and physical properties of different kinds of RF and microwave +transmission lines. + +For each type of transmission line, using dialog boxes, the user can +enter values for the various parameters, and either calculate its +electrical properties, or use the given electrical requirements to +synthesize physical parameters of the required transmission line. + +Available transmission lines are: Microstrip, Rectangular Waveguide, +Coaxial Line, Coplanar and Coupled Microstrips. + +.SH AVAILABILITY +The latest version of Qucs can always be obtained from +\fB@PACKAGE_URL@\fR +.SH "REPORTING BUGS" +Known bugs are documented within the BUGS file. Report bugs to +\fB@PACKAGE_BUGREPORT@\fR +.SH COPYRIGHT +Copyright \(co 2005 Stefan Jahn +.PP +This is free software; see the source for copying conditions. There is NO +warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. +.SH AUTHORS +Written by Gopal Narayanan , Claudio Girardi +, Michael Margraf + and Stefan Jahn +. diff --git a/qucs-transcalc/qucstrans.cpp b/qucs-transcalc/qucstrans.cpp new file mode 100644 index 00000000..dec88073 --- /dev/null +++ b/qucs-transcalc/qucstrans.cpp @@ -0,0 +1,1510 @@ +/*************************************************************************** + qucstrans.cpp - description + ------------------- + begin : Sun Feb 27 2005 + copyright : (C) 2005, 2006 by Stefan Jahn + email : stefan@lkcc.org + ***************************************************************************/ + +/*************************************************************************** + * * + * This program is free software; you can redistribute it and/or modify * + * it under the terms of the GNU General Public License as published by * + * the Free Software Foundation; either version 2 of the License, or * + * (at your option) any later version. * + * * + ***************************************************************************/ + +#ifdef HAVE_CONFIG_H +# include +#endif + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "qucstrans.h" +#include "helpdialog.h" +#include "optionsdialog.h" +#include "transline.h" +#include "units.h" +#include "microstrip.h" +#include "coplanar.h" +#include "coax.h" +#include "rectwaveguide.h" +#include "c_microstrip.h" +#include "stripline.h" + +// Defines maximum number of entries in each property category. +static const int TransMaxBox[MAX_TRANS_BOXES] = { 9, 1, 4, 3 }; + +// Helper #defines for the below transmission line types. +#define TRANS_RADIOS { -1, -1, -1, -1 } +#define TRANS_QOBJS NULL, NULL, NULL, NULL +#define TRANS_END { NULL, 0, NULL, { NULL }, 0, TRANS_QOBJS } +#define TRANS_RESULT { NULL, NULL, NULL } +#define TRANS_RESULTS { TRANS_RESULT } + +// Defines the available transmission line types. +struct TransType TransLineTypes[] = { + { ModeMicrostrip, "Microstrip", "microstrip.png", NULL, + { { { + { "Er", 2.94, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Mur", 1, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "H", 10, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "H_t", 1e20, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "T", 0.1, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "Cond", 4.1e7, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Tand", 0, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Rough", 0, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Freq", 1, NULL, TRANS_FREQS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "W", 10, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "L", 100, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Z0", 50, NULL, TRANS_OHMS, 0, TRANS_QOBJS }, + { "Ang_l", 90, NULL, TRANS_ANGLES, 0, TRANS_QOBJS }, + TRANS_END + } } }, + 4, TRANS_RESULTS, TRANS_RADIOS + }, + { ModeCoplanar, "Coplanar", "cpw.png", NULL, + { { { + { "Er", 2.94, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "H", 10, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "T", 0.1, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "Cond", 4.1e7, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Tand", 0, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Freq", 1, NULL, TRANS_FREQS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "W", 10, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "S", 5, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "L", 100, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Z0", 50, NULL, TRANS_OHMS, 0, TRANS_QOBJS }, + { "Ang_l", 90, NULL, TRANS_ANGLES, 0, TRANS_QOBJS }, + TRANS_END + } } }, + 4, TRANS_RESULTS, { 1, 0, -1, -1 } + }, + { ModeGroundedCoplanar, "GroundedCoplanar", "cpw_back.png", NULL, + { { { + { "Er", 2.94, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "H", 10, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "T", 0.1, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "Cond", 4.1e7, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Tand", 0, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Freq", 1, NULL, TRANS_FREQS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "W", 10, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "S", 5, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "L", 100, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Z0", 50, NULL, TRANS_OHMS, 0, TRANS_QOBJS }, + { "Ang_l", 90, NULL, TRANS_ANGLES, 0, TRANS_QOBJS }, + TRANS_END + } } }, + 4, TRANS_RESULTS, { 1, 0, -1, -1 } + }, + { ModeRectangular, "Rectangular", "rectwaveguide.png", NULL, + { { { + { "Er", 1, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Mur", 1, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Cond", 4.1e7, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Tand", 0, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "TanM", 0, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Freq", 10, NULL, TRANS_FREQS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "a", 1000, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "b", 500, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "L", 4000, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Z0", 50, NULL, TRANS_OHMS, 0, TRANS_QOBJS }, + { "Ang_l", 90, NULL, TRANS_ANGLES, 0, TRANS_QOBJS }, + TRANS_END + } } }, + 3, TRANS_RESULTS, { 0, 1, -1, -1 } + }, + { ModeCoaxial, "Coaxial", "coax.png", NULL, + { { { + { "Er", 2.1, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Mur", 1, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Tand", 0.002, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Sigma", 4.1e7, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Freq", 10, NULL, TRANS_FREQS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "din", 40, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "dout", 134, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "L", 1000, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Z0", 50, NULL, TRANS_OHMS, 0, TRANS_QOBJS }, + { "Ang_l", 90, NULL, TRANS_ANGLES, 0, TRANS_QOBJS }, + TRANS_END + } } }, + 2, TRANS_RESULTS, { 0, 1, -1, -1 } + }, + { ModeCoupledMicrostrip, "CoupledMicrostrip", "c_microstrip.png", NULL, + { { { + { "Er", 4.3, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Mur", 1, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "H", 8.27, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "H_t", 1e20, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "T", 0.68, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "Cond", 4.1e7, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Tand", 0, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Rough", 0, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Freq", 10, NULL, TRANS_FREQS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "W", 8.66, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "S", 5.31, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "L", 1000, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Z0e", 50, NULL, TRANS_OHMS, 0, TRANS_QOBJS }, + { "Z0o", 50, NULL, TRANS_OHMS, 0, TRANS_QOBJS }, + { "Ang_l", 90, NULL, TRANS_ANGLES, 0, TRANS_QOBJS }, + TRANS_END + } } }, + 7, TRANS_RESULTS, TRANS_RADIOS + }, + { ModeStripline, "Stripline", "stripline.png", NULL, + { { { + { "Er", 4.5, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "Mur", 1, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "h", 27.56, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "Tand", 0.018, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + { "T", 1.42, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "Sigma", 5.8e7, NULL, TRANS_NONES, 0, TRANS_QOBJS }, + + TRANS_END + } }, + { { + { "Freq", 10, NULL, TRANS_FREQS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "W", 19.69, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + { "L", 100, NULL, TRANS_LENGTHS, 0, TRANS_QOBJS }, + TRANS_END + } }, + { { + { "Z0", 50, NULL, TRANS_OHMS, 0, TRANS_QOBJS }, + { "Ang_l", 90, NULL, TRANS_ANGLES, 0, TRANS_QOBJS }, + TRANS_END + } } }, + 2, TRANS_RESULTS, { 0, 1, -1, -1 } + }, + { ModeNone, NULL, NULL, NULL, { { { TRANS_END } } }, 0, + TRANS_RESULTS, TRANS_RADIOS } +}; + +struct TransUnit TransUnits[] = { + { "Frequency", TRANS_FREQS }, + { "Length", TRANS_LENGTHS }, + { "Resistance", TRANS_OHMS }, + { "Angle", TRANS_ANGLES }, +}; + +/* Constructor setups the GUI. */ +QucsTranscalc::QucsTranscalc() { + QWidget *centralWidget = new QWidget(this); + setCentralWidget(centralWidget); + + // set application icon + setWindowIcon(QPixmap(":/bitmaps/big.qucs.xpm")); + setWindowTitle("Qucs Transcalc " PACKAGE_VERSION); + + // create file menu + QMenu *fileMenu = new QMenu(tr("&File")); + + QAction *fileLoad = new QAction(tr("&Load"), this); + fileLoad->setShortcut(Qt::CTRL+Qt::Key_L); + fileMenu->addAction(fileLoad); + connect(fileLoad, SIGNAL(triggered(bool)), SLOT(slotFileLoad())); + + QAction *fileSave = new QAction (tr("&Save"), this); + fileSave->setShortcut(Qt::CTRL+Qt::Key_S); + fileMenu->addAction(fileSave); + connect(fileSave, SIGNAL(triggered(bool)), SLOT(slotFileSave())); + + fileMenu->addSeparator(); + + QAction *fileOption = new QAction (tr("&Options"), this); + fileOption->setShortcut(Qt::CTRL+Qt::Key_O); + fileMenu->addAction(fileOption); + connect(fileOption, SIGNAL(triggered(bool)), SLOT(slotOptions())); + + fileMenu->addSeparator(); + + QAction *fileQuit = new QAction (tr("&Quit"), this); + fileQuit->setShortcut(Qt::CTRL+Qt::Key_Q); + fileMenu->addAction(fileQuit); + connect(fileQuit, SIGNAL(triggered(bool)), SLOT(slotQuit())); + + // create execute menu + QMenu *execMenu = new QMenu(tr("&Execute")); + + QAction *execCopy = new QAction(tr("&Copy to Clipboard"), this); + execCopy->setShortcut(Qt::Key_F2); + execMenu->addAction(execCopy); + connect(execCopy, SIGNAL(triggered(bool)), SLOT(slotCopyToClipBoard())); + + QAction *execAnalyze = new QAction(tr("&Analyze"), this); + execAnalyze->setShortcut(Qt::Key_F3); + execMenu->addAction(execAnalyze); + connect(execAnalyze, SIGNAL(triggered(bool)), SLOT(slotAnalyze())); + + QAction *execSynthesize = new QAction (tr("&Synthesize"), this); + execSynthesize->setShortcut(Qt::Key_F4); + execMenu->addAction(execSynthesize); + connect(execSynthesize, SIGNAL(triggered(bool)), SLOT(slotSynthesize())); + + // create help menu + QMenu *helpMenu = new QMenu(tr("&Help")); + QAction *helpHelp = new QAction(tr("&Help"), this); + helpHelp->setShortcut(Qt::Key_F1); + helpMenu->addAction(helpHelp); + connect(helpHelp, SIGNAL(triggered(bool)), SLOT(slotHelpIntro())); + + QAction *helpAbout = new QAction(tr("About"), this); + helpMenu->addAction(helpAbout); + connect(helpAbout, SIGNAL(triggered(bool)), SLOT(slotAbout())); + + // setup menu bar + menuBar()->addMenu(fileMenu); + menuBar()->addMenu(execMenu); + menuBar()->addSeparator(); + menuBar()->addMenu(helpMenu); + + // === left + // seletion combo and figure + QVBoxLayout *vl = new QVBoxLayout(); + + // transmission line type choice + QGroupBox * lineGroup = new QGroupBox (tr("Transmission Line Type")); + tranType = new QComboBox (lineGroup); + tranType->insertItem (0, tr("Microstrip Line")); + tranType->insertItem (1, tr("Coplanar Waveguide")); + tranType->insertItem (2, tr("Grounded Coplanar")); + tranType->insertItem (3, tr("Rectangular Waveguide")); + tranType->insertItem (4, tr("Coaxial Line")); + tranType->insertItem (5, tr("Coupled Microstrip")); + tranType->insertItem (6, tr("Stripline")); + connect(tranType, SIGNAL(activated(int)), SLOT(slotSelectType(int))); + // setup transmission line picture + pix = new QLabel (lineGroup); + pix->setPixmap(QPixmap(":/bitmaps/microstrip.png")); + + QVBoxLayout *vfig = new QVBoxLayout(); + vfig->addWidget(tranType); + vfig->addWidget(pix); + vfig->setSpacing(3); + lineGroup->setLayout(vfig); + + vl->addWidget(lineGroup); + + // init additional translations + setupTranslations (); + + // set current mode + mode = ModeMicrostrip; + + // === middle + QVBoxLayout *vm = new QVBoxLayout(); + vm->setSpacing (3); + + // substrate parameter box + QGroupBox * substrate = new QGroupBox (tr("Substrate Parameters")); + vm->addWidget(substrate); + // Pass the GroupBox > create Grid layout > Add widgets > set layout + createPropItems (substrate, TRANS_SUBSTRATE); + // component parameter box + QGroupBox * component = new QGroupBox (tr("Component Parameters")); + vm->addWidget(component); + createPropItems (component, TRANS_COMPONENT); + + // === right + QVBoxLayout *vr = new QVBoxLayout(); + vr->setSpacing (3); + + // physical parameter box + QGroupBox * physical = new QGroupBox (tr("Physical Parameters")); + vr->addWidget(physical); + createPropItems (physical, TRANS_PHYSICAL); + + // analyze/synthesize buttons + QHBoxLayout * h2 = new QHBoxLayout(); + QPushButton * analyze = new QPushButton (tr("Analyze")); + h2->addWidget(analyze); + analyze->setToolTip(tr("Derive Electrical Parameters")); + connect(analyze, SIGNAL(clicked()), SLOT(slotAnalyze())); + + QPushButton * synthesize = new QPushButton (tr("Synthesize")); + h2->addWidget(synthesize); + synthesize->setToolTip(tr("Compute Physical Parameters")); + connect(synthesize, SIGNAL(clicked()), SLOT(slotSynthesize())); + vr->addLayout(h2); + + // electrical parameter box + QGroupBox * electrical = new QGroupBox (tr("Electrical Parameters")); + vr->addWidget(electrical); + createPropItems (electrical, TRANS_ELECTRICAL); + + calculated = new QGroupBox (tr("Calculated Results")); + vr->addWidget(calculated); + + // status line + //statBar = new QStatusBar (this); + //QLabel * statText = new QLabel ("Ready.", statBar); + statusBar()->showMessage (tr("Ready.")); + //statBar->setFixedHeight (statText->height ()); + //delete statText; + + QVBoxLayout *vmain = new QVBoxLayout(); + + QHBoxLayout *hmain = new QHBoxLayout(); + hmain->addLayout(vl); + hmain->addLayout(vm); + hmain->addLayout(vr); + + vmain->addLayout(hmain); + //vmain->addWidget(statBar); + + centralWidget->setLayout(vmain); + + // setup calculated result bix + createResultItems (calculated); + updateSelection (); + // instantiate transmission lines + TransLineTypes[0].line = new microstrip (); + TransLineTypes[0].line->setApplication (this); + TransLineTypes[1].line = new coplanar (); + TransLineTypes[1].line->setApplication (this); + TransLineTypes[2].line = new groundedCoplanar (); + TransLineTypes[2].line->setApplication (this); + TransLineTypes[3].line = new rectwaveguide (); + TransLineTypes[3].line->setApplication (this); + TransLineTypes[4].line = new coax (); + TransLineTypes[4].line->setApplication (this); + TransLineTypes[5].line = new c_microstrip (); + TransLineTypes[5].line->setApplication (this); + TransLineTypes[6].line = new stripline (); + TransLineTypes[6].line->setApplication (this); + +} + +/* Destructor destroys the application. */ +QucsTranscalc::~QucsTranscalc() +{ + for (int i = 0; i < MAX_TRANS_TYPES; i++) { + if (TransLineTypes[i].line) + delete TransLineTypes[i].line; + } +} + +/* The function puts translatable text into the transmission line + structures. */ +void QucsTranscalc::setupTranslations () { + // calculated results + int i = 0;//Microstrip + TransLineTypes[i].result[0].name = new QString(tr("ErEff")); + TransLineTypes[i].result[1].name = new QString(tr("Conductor Losses")); + TransLineTypes[i].result[2].name = new QString(tr("Dielectric Losses")); + TransLineTypes[i].result[3].name = new QString(tr("Skin Depth")); + + i++;//CPW + TransLineTypes[i].result[0].name = new QString(tr("ErEff")); + TransLineTypes[i].result[1].name = new QString(tr("Conductor Losses")); + TransLineTypes[i].result[2].name = new QString(tr("Dielectric Losses")); + TransLineTypes[i].result[3].name = new QString(tr("Skin Depth")); + + i++;//GCPW + TransLineTypes[i].result[0].name = new QString(tr("ErEff")); + TransLineTypes[i].result[1].name = new QString(tr("Conductor Losses")); + TransLineTypes[i].result[2].name = new QString(tr("Dielectric Losses")); + TransLineTypes[i].result[3].name = new QString(tr("Skin Depth")); + + i++;//Rectangular waveguide + TransLineTypes[i].result[0].name = new QString(tr("ErEff")); + TransLineTypes[i].result[1].name = new QString(tr("Conductor Losses")); + TransLineTypes[i].result[2].name = new QString(tr("Dielectric Losses")); + TransLineTypes[i].result[3].name = new QString(tr("TE-Modes")); + TransLineTypes[i].result[4].name = new QString(tr("TM-Modes")); + + i++;//Coaxial line + TransLineTypes[i].result[0].name = new QString(tr("Conductor Losses")); + TransLineTypes[i].result[1].name = new QString(tr("Dielectric Losses")); + TransLineTypes[i].result[2].name = new QString(tr("TE-Modes")); + TransLineTypes[i].result[3].name = new QString(tr("TM-Modes")); + + i++;//Coupled microstripx + TransLineTypes[i].result[0].name = new QString(tr("ErEff Even")); + TransLineTypes[i].result[1].name = new QString(tr("ErEff Odd")); + TransLineTypes[i].result[2].name = new QString(tr("Conductor Losses Even")); + TransLineTypes[i].result[3].name = new QString(tr("Conductor Losses Odd")); + TransLineTypes[i].result[4].name = new QString(tr("Dielectric Losses Even")); + TransLineTypes[i].result[5].name = new QString(tr("Dielectric Losses Odd")); + TransLineTypes[i].result[6].name = new QString(tr("Skin Depth")); + + i++;//Stripline + TransLineTypes[i].result[0].name = new QString(tr("Conductor Losses")); + TransLineTypes[i].result[1].name = new QString(tr("Dielectric Losses")); + TransLineTypes[i].result[2].name = new QString(tr("Skin Depth")); + + // extra tool tips + struct TransType * t = TransLineTypes; + //Microstrip + t->array[0].item[0].tip = new QString(tr("Relative Permittivity")); + t->array[0].item[1].tip = new QString(tr("Relative Permeability")); + t->array[0].item[2].tip = new QString(tr("Height of Substrate")); + t->array[0].item[3].tip = new QString(tr("Height of Box Top")); + t->array[0].item[4].tip = new QString(tr("Strip Thickness")); + t->array[0].item[5].tip = new QString(tr("Strip Conductivity")); + t->array[0].item[6].tip = new QString(tr("Dielectric Loss Tangent")); + t->array[0].item[7].tip = new QString(tr("Conductor Roughness")); + t->array[1].item[0].tip = new QString(tr("Frequency")); + t->array[2].item[0].tip = new QString(tr("Line Width")); + t->array[2].item[1].tip = new QString(tr("Line Length")); + t->array[3].item[0].tip = new QString(tr("Characteristic Impedance")); + t->array[3].item[1].tip = new QString(tr("Electrical Length")); + + t++;//CPW + t->array[0].item[0].tip = new QString(tr("Relative Permittivity")); + t->array[0].item[1].tip = new QString(tr("Height of Substrate")); + t->array[0].item[2].tip = new QString(tr("Strip Thickness")); + t->array[0].item[3].tip = new QString(tr("Strip Conductivity")); + t->array[0].item[4].tip = new QString(tr("Dielectric Loss Tangent")); + t->array[1].item[0].tip = new QString(tr("Frequency")); + t->array[2].item[0].tip = new QString(tr("Line Width")); + t->array[2].item[1].tip = new QString(tr("Gap Width")); + t->array[2].item[2].tip = new QString(tr("Line Length")); + t->array[3].item[0].tip = new QString(tr("Characteristic Impedance")); + t->array[3].item[1].tip = new QString(tr("Electrical Length")); + + t++;//GCPW + t->array[0].item[0].tip = new QString(tr("Relative Permittivity")); + t->array[0].item[1].tip = new QString(tr("Height of Substrate")); + t->array[0].item[2].tip = new QString(tr("Strip Thickness")); + t->array[0].item[3].tip = new QString(tr("Strip Conductivity")); + t->array[0].item[4].tip = new QString(tr("Dielectric Loss Tangent")); + t->array[1].item[0].tip = new QString(tr("Frequency")); + t->array[2].item[0].tip = new QString(tr("Line Width")); + t->array[2].item[1].tip = new QString(tr("Gap Width")); + t->array[2].item[2].tip = new QString(tr("Line Length")); + t->array[3].item[0].tip = new QString(tr("Characteristic Impedance")); + t->array[3].item[1].tip = new QString(tr("Electrical Length")); + + t++;//Rectangular waveguide + t->array[0].item[0].tip = new QString(tr("Relative Permittivity")); + t->array[0].item[1].tip = new QString(tr("Relative Permeability")); + t->array[0].item[2].tip = new QString(tr("Conductivity of Metal")); + t->array[0].item[3].tip = new QString(tr("Dielectric Loss Tangent")); + t->array[0].item[4].tip = new QString(tr("Magnetic Loss Tangent")); + t->array[1].item[0].tip = new QString(tr("Frequency")); + t->array[2].item[0].tip = new QString(tr("Width of Waveguide")); + t->array[2].item[1].tip = new QString(tr("Height of Waveguide")); + t->array[2].item[2].tip = new QString(tr("Waveguide Length")); + t->array[3].item[0].tip = new QString(tr("Characteristic Impedance")); + t->array[3].item[1].tip = new QString(tr("Electrical Length")); + + t++;//Coaxial line + t->array[0].item[0].tip = new QString(tr("Relative Permittivity")); + t->array[0].item[1].tip = new QString(tr("Relative Permeability")); + t->array[0].item[2].tip = new QString(tr("Dielectric Loss Tangent")); + t->array[0].item[3].tip = new QString(tr("Conductivity of Metal")); + t->array[1].item[0].tip = new QString(tr("Frequency")); + t->array[2].item[0].tip = new QString(tr("Inner Diameter")); + t->array[2].item[1].tip = new QString(tr("Outer Diameter")); + t->array[2].item[2].tip = new QString(tr("Length")); + t->array[3].item[0].tip = new QString(tr("Characteristic Impedance")); + t->array[3].item[1].tip = new QString(tr("Electrical Length")); + + t++;//Coupled microstrip + t->array[0].item[0].tip = new QString(tr("Relative Permittivity")); + t->array[0].item[1].tip = new QString(tr("Relative Permeability")); + t->array[0].item[2].tip = new QString(tr("Height of Substrate")); + t->array[0].item[3].tip = new QString(tr("Height of Box Top")); + t->array[0].item[4].tip = new QString(tr("Strip Thickness")); + t->array[0].item[5].tip = new QString(tr("Strip Conductivity")); + t->array[0].item[6].tip = new QString(tr("Dielectric Loss Tangent")); + t->array[0].item[7].tip = new QString(tr("Conductor Roughness")); + t->array[1].item[0].tip = new QString(tr("Frequency")); + t->array[2].item[0].tip = new QString(tr("Line Width")); + t->array[2].item[1].tip = new QString(tr("Gap Width")); + t->array[2].item[2].tip = new QString(tr("Length")); + t->array[3].item[0].tip = new QString(tr("Even-Mode Impedance")); + t->array[3].item[1].tip = new QString(tr("Odd-Mode Impedance")); + t->array[3].item[2].tip = new QString(tr("Electrical Length")); + + t++;//Stripline + t->array[0].item[0].tip = new QString(tr("Relative Permittivity")); + t->array[0].item[1].tip = new QString(tr("Relative Permeability")); + t->array[0].item[2].tip = new QString(tr("Dielectric Loss Tangent")); + t->array[0].item[3].tip = new QString(tr("Conductivity of Metal")); + t->array[0].item[4].tip = new QString(tr("Conductor thickness")); + t->array[0].item[5].tip = new QString(tr("Substrate height")); + t->array[1].item[0].tip = new QString(tr("Frequency")); + t->array[2].item[0].tip = new QString(tr("Width")); + t->array[2].item[1].tip = new QString(tr("Length")); + t->array[3].item[0].tip = new QString(tr("Characteristic Impedance")); + t->array[3].item[1].tip = new QString(tr("Electrical Length")); +} + +/* Creates a property item 'val' in a parameter category specified by + its 'box'. */ +void QucsTranscalc::createPropItem (QGridLayout * parentGrid, TransValue * val, + int box, QButtonGroup * group) { + Q_UNUSED(group); + QRadioButton * r = NULL; + QLabel * l; + QLineEdit * e; + QComboBox * c; + QDoubleValidator * v = new QDoubleValidator (this); + + // name label + l = new QLabel (val->name); + parentGrid->addWidget(l, parentGrid->rowCount(), 0); + + l->setAlignment (Qt::AlignRight); + if (val->tip) l->setToolTip(*(val->tip)); + val->label = l; + + // editable value text + e = new QLineEdit (); + parentGrid->addWidget(e, parentGrid->rowCount()-1, 1); + e->setText (QString::number (val->value)); + e->setAlignment (Qt::AlignRight); + e->setValidator (v); + connect(e, SIGNAL(textChanged(const QString&)), SLOT(slotValueChanged())); + if (!val->name) e->setDisabled (true); + val->lineedit = e; + + // unit choice + c = new QComboBox (); + parentGrid->addWidget(c, parentGrid->rowCount()-1, 2); + if (!val->units[0]) { + c->addItem ("NA"); + c->setDisabled(true); + } + else { + int nounit = 0; + for (int i = 0; val->units[i]; i++) { + c->addItem (val->units[i]); + if (!strcmp (val->units[i], "NA")) nounit++; + } + c->setDisabled (nounit != 0); + c->setCurrentIndex (0); + } + connect(c, SIGNAL(activated(int)), SLOT(slotValueChanged())); + val->combobox = c; + + // special synthesize-computation choice + if (box == TRANS_PHYSICAL) { + r = new QRadioButton (); + r->setDisabled (true); + r->setHidden(true); + val->radio = r; + parentGrid->addWidget(r, parentGrid->rowCount()-1, 3); + } +} + +/* The function changes the content of a GUI's property entry + depending on the given transmission line property. */ +void QucsTranscalc::updatePropItem (TransValue * val) { + // update label text + val->label->setText (val->name); + if (val->tip) val->label->setToolTip(*(val->tip)); + // update editable value text + val->lineedit->setText (QString::number (val->value)); + val->lineedit->setDisabled (!val->name); + + // update unit choice + val->combobox->clear(); + if (!val->units[0]) { + val->combobox->addItem ("NA"); + val->combobox->setDisabled (true); + } + else { + int nounit = 0; + for (int i = 0; val->units[i]; i++) { + val->combobox->addItem (val->units[i]); + if (!strcmp (val->units[i], "NA")) nounit++; + } + val->combobox->setCurrentIndex(val->unit); + val->combobox->setDisabled (nounit); + } +} + +/* Switches to the current type of transmission line layout. */ +void QucsTranscalc::setMode (int _mode) { + // change necessary? + if (mode == _mode) return; + storeValues (); + mode = _mode; + setUpdatesEnabled(false); + updateMode (); + // update selection and results + updateSelection (); + updateResultItems (); + + slotAnalyze(); + setUpdatesEnabled(true); + repaint(); +} + +/* Updates the property items of the current mode. */ +void QucsTranscalc::updateMode (void) { + // go through each type of parameter category + for (int box = 0; box < MAX_TRANS_BOXES; box++) { + int last = 0, idx = getTypeIndex(); + struct TransValue * val = TransLineTypes[idx].array[box].item; + // update each property item + for (int i = 0; i < TransMaxBox[box]; i++) { + // fix uninitialized memory + if (val->name == NULL) last++; + if (last) { + val->name = NULL; + val->value = 0; + val->tip = NULL; + val->units[0] = NULL; + } + updatePropItem (val); + val++; + } + } +} + +/* Updates the current choice of physical property selection. */ +void QucsTranscalc::updateSelection () { + int idx = getTypeIndex (); + struct TransValue * val = TransLineTypes[idx].array[TRANS_PHYSICAL].item; + for (int i = 0; i < TransMaxBox[TRANS_PHYSICAL]; i++) { + if (TransLineTypes[idx].radio[i] != -1) { + val->radio->setHidden(false); + if (TransLineTypes[idx].radio[i] == 1) { + val->radio->setDown (true); + val->radio->setChecked (true); + val->radio->setToolTip(tr("Selected for Calculation")); + } + else { + val->radio->setDown (false); + val->radio->setChecked (false); + val->radio->setToolTip(tr("Check item for Calculation")); + } + val->radio->setDisabled (false); + } + else { + val->radio->setHidden(true); + val->radio->setDown (false); + val->radio->setChecked (false); + val->radio->setDisabled (true); + } + val++; + } +} + +/* The function creates the property items for the given category of + transmission line parameters. */ +void QucsTranscalc::createPropItems (QGroupBox *parent, int box) { + struct TransValue * val, * dup; + int last = 0, idx = getTypeIndex (); + val = TransLineTypes[idx].array[box].item; + QGridLayout *boxGrid = new QGridLayout(); + QButtonGroup * group = new QButtonGroup(); + connect(group, SIGNAL(buttonPressed(int)), SLOT(slotRadioChecked(int))); + + boxGrid->setSpacing(2); + parent->setLayout(boxGrid); + + // go through each parameter category + for (int i = 0; i < TransMaxBox[box]; i++) { + // fix uninitialized memory + + if (val->name == NULL) last++; + if (last) { + val->name = NULL; + val->value = 0; + val->tip = NULL; + val->units[0] = NULL; + } + + createPropItem (boxGrid, val, box, group); + + // publish the newly created widgets to the other transmission lines + for (int _mode = 0; _mode < MAX_TRANS_TYPES; _mode++) { + if (idx != _mode) { + dup = & TransLineTypes[_mode].array[box].item[i]; + dup->label = val->label; + dup->lineedit = val->lineedit; + dup->combobox = val->combobox; + dup->radio = val->radio; + // dup->value = val->value; + } + } + val++; + } +} + +/* Creates a single result item. */ +void QucsTranscalc::createResultItem (QGridLayout * parentGrid, TransResult * res) { + QLabel * n = new QLabel (res->name ? *(res->name) + ":" : QString()); + parentGrid->addWidget(n, parentGrid->count(),0,1,1); + n->setAlignment (Qt::AlignRight); + res->label = n; + + QLabel * v = new QLabel (); + parentGrid->addWidget(v, parentGrid->count()-1,1,1,1); + v->setAlignment (Qt::AlignLeft); + res->value = v; + if (!res->name) { + v->hide (); + n->hide (); + } +} + +/* Updates the given result item. */ +void QucsTranscalc::updateResultItem (TransResult * res) { + if (res->name) { + res->label->setText (*(res->name) + ":"); + res->label->show (); + res->value->show (); + } else { + res->label->hide (); + res->value->hide (); + } +} + +/* Creates all the result items. */ +void QucsTranscalc::createResultItems (QGroupBox * parent) { + struct TransResult * res, * dup; + int idx = getTypeIndex (); + res = & TransLineTypes[idx].result[0]; + + QGridLayout *boxGrid = new QGridLayout(); + + boxGrid->setSpacing(2); + parent->setLayout(boxGrid); + + for (int i = 0; i < MAX_TRANS_RESULTS; i++) { + createResultItem (boxGrid, res); + for (int _mode = 0; _mode < MAX_TRANS_TYPES; _mode++) { + if (idx != _mode) { + dup = & TransLineTypes[_mode].result[i]; + dup->label = res->label; + dup->value = res->value; + } + } + res++; + } +} + +/* Puts the given result into its widget. */ +void QucsTranscalc::setResult (int line, const char * text) { + struct TransResult * res; + res = & TransLineTypes[getTypeIndex ()].result[line]; + res->value->setText (text); +} + +/* Updates the current result items. */ +void QucsTranscalc::updateResultItems() { + int idx = getTypeIndex (); + struct TransResult * res = TransLineTypes[idx].result; + for (int i = 0; i < MAX_TRANS_RESULTS; i++) { + updateResultItem (res); + res++; + } +} + +/* Returns the appropriate array index for the given mode. */ +int QucsTranscalc::getTypeIndex () { + struct TransType * type = TransLineTypes; + for (int i = 0; type->type != ModeNone; i++) { + if (type->type == mode) return i; + type++; + } + return 0; +} + +/* Returns the property value specified by its name. */ +struct TransValue * QucsTranscalc::findProperty (QString prop) { + struct TransValue * val = NULL; + for (int box = 0; box < MAX_TRANS_BOXES; box++) { + val = TransLineTypes[getTypeIndex ()].array[box].item; + for (int i = 0; val->name; i++) { + if (prop == val->name) return val; + val++; + } + } + return NULL; +} + +/* Sets the given property to the given value. */ +void QucsTranscalc::setProperty (QString prop, double value) { + struct TransValue * val = findProperty (prop); + if (val) { + val->lineedit->setText (QString::number (value)); + val->value = value; + } +} + +/* Returns the given property value. */ +double QucsTranscalc::getProperty (QString prop) { + struct TransValue * val = findProperty (prop); + if (val) { + QString str = val->lineedit->text (); + val->value = str.toDouble (); + return val->value; + } + return 0; +} + +/* Sets the given property unit. */ +void QucsTranscalc::setUnit (QString prop, const char * unit) { + struct TransValue * val = findProperty (prop); + if (val) { + if (!unit) { + val->combobox->setCurrentIndex (0); + val->unit = 0; + } + else for (int i = 0; val->units[i]; i++) { + if (!strcmp (unit, val->units[i])) { + val->combobox->setCurrentIndex (i); + val->unit = i; + break; + } + } + } +} + +/* Returns the given property unit. */ +char * QucsTranscalc::getUnit (QString prop) { + struct TransValue * val = findProperty (prop); + if (val) { + QString str = val->combobox->currentText (); + for (int i = 0; val->units[i]; i++) { + if (str == val->units[i]) { + val->unit = i; + return (char *) val->units[i]; + } + } + } + return NULL; +} + +/* Returns the given property selection. */ +bool QucsTranscalc::isSelected (QString prop) { + struct TransValue * val = findProperty (prop); + if (val) { + if (val->radio) + if (val->radio->isChecked ()) { + return true; + } + } + return false; +} + +void QucsTranscalc::slotAbout() +{ + QMessageBox::about(this, tr("About..."), + "QucsTranscalc Version " PACKAGE_VERSION "\n"+ + tr("Transmission Line Calculator for Qucs\n")+ + tr("Copyright (C) 2001 by Gopal Narayanan\n")+ + tr("Copyright (C) 2002 by Claudio Girardi\n")+ + tr("Copyright (C) 2005 by Stefan Jahn\n")+ + tr("Copyright (C) 2008 by Michael Margraf\n")+ + "\nThis is free software; see the source for copying conditions." + "\nThere is NO warranty; not even for MERCHANTABILITY or " + "\nFITNESS FOR A PARTICULAR PURPOSE."); +} + +void QucsTranscalc::slotQuit() +{ + int tmp; + tmp = x(); // call size and position function in order to ... + tmp = y(); // ... set them correctly before closing the ... + tmp = width(); // dialog !!! Otherwise the frame of the window ... + tmp = height(); // will not be recognized (a X11 problem). + Q_UNUSED(tmp); + + storeValues (); + qApp->quit(); +} + +void QucsTranscalc::closeEvent(QCloseEvent *Event) +{ + int tmp; + tmp = x(); // call size and position function in order to ... + tmp = y(); // ... set them correctly before closing the ... + tmp = width(); // dialog !!! Otherwise the frame of the window ... + tmp = height(); // will not be recognized (a X11 problem). + Q_UNUSED(tmp); + + storeValues (); + Event->accept(); +} + +void QucsTranscalc::slotSelectType (int Type) +{ + pix->setPixmap(QPixmap(":/bitmaps/" + QString(TransLineTypes[Type].bitmap))); + setMode (Type); + statusBar()->showMessage(tr("Ready.")); +} + +void QucsTranscalc::slotAnalyze() +{ + if (TransLineTypes[getTypeIndex()].line) + TransLineTypes[getTypeIndex()].line->analyze(); + statusBar()->showMessage(tr("Values are consistent.")); +} + +void QucsTranscalc::slotSynthesize() +{ + int status = -1; + + if (TransLineTypes[getTypeIndex()].line) + status = TransLineTypes[getTypeIndex()].line->synthesize(); + + if (status == 0) + statusBar()->showMessage(tr("Values are consistent.")); + else + statusBar()->showMessage(tr("Failed to converge!")); +} + +void QucsTranscalc::slotValueChanged() +{ + statusBar()->showMessage(tr("Values are inconsistent.")); +} + +// Load transmission line values from the given file. +bool QucsTranscalc::loadFile(QString fname, int * _mode) { + QFile file(QDir::toNativeSeparators(fname)); + if(!file.open(QIODevice::ReadOnly)) return false; // file doesn't exist + + QTextStream stream(&file); + QString Line, Name, Unit; + double Value; + + int oldmode = mode; + while(!stream.atEnd()) { + Line = stream.readLine(); + for (int i = 0; i < MAX_TRANS_TYPES; i++) { + if (Line == "<" + QString(TransLineTypes[i].description) + ">") { + if (_mode) { + *_mode = TransLineTypes[i].type; + setMode (*_mode); + updatePixmap (mode); + } else { + mode = TransLineTypes[i].type; + } + while(!stream.atEnd()) { + Line = stream.readLine(); + if (Line == "") + break; + Line = Line.simplified(); + Name = Line.section(' ',0,0); + Value = Line.section(' ',1,1).toDouble(); + Unit = Line.section(' ',2,2); + setProperty (Name, Value); + setUnit (Name, Unit.toLatin1()); + } + break; + } + } + } + if (!_mode) mode = oldmode; + updateMode (); + file.close(); + return true; +} + +// Saves current transmission line values into the given file. +bool QucsTranscalc::saveFile(QString fname) { + QFile file (QDir::toNativeSeparators(fname)); + if(!file.open (QIODevice::WriteOnly)) return false; // file not writable + QTextStream stream (&file); + + // some lines of documentation + stream << "# QucsTranscalc " << PACKAGE_VERSION << " " << fname << "\n"; + stream << "# Generated on " << QDate::currentDate().toString() + << " at " << QTime::currentTime().toString() << ".\n"; + stream << "# It is not suggested to edit the file, use QucsTranscalc " + << "instead.\n\n"; + + storeValues (); + saveMode (stream); + file.close (); + return true; +} + +/* Writes the transmission line values for the current modes into the + given stream. */ +void QucsTranscalc::saveMode(QTextStream& stream) { + struct TransType * t = &TransLineTypes[getTypeIndex ()]; + struct TransValue * val = NULL; + stream << "<" << t->description << ">\n"; + for (int box = 0; box < MAX_TRANS_BOXES; box++) { + val = t->array[box].item; + while (val->name) { + stream << " " << val->name << " " << val->value << " " + << val->units[val->unit] << "\n"; + val++; + } + } + stream << "description << ">\n"; +} + +// Writes the transmission line values for all modes into the given file. +bool QucsTranscalc::saveModes(QString fname) { + int oldmode = mode; + + QFile file(fname); + if(!file.open(QIODevice::WriteOnly)) { + QMessageBox::warning(this, QObject::tr("Warning"), + QObject::tr("Cannot save GUI settings in\n") + fname); + return false; + } + + QTextStream stream(&file); + stream << "QucsTranscalc " PACKAGE_VERSION " GUI Settings File\n"; + for (int i = 0; i < MAX_TRANS_TYPES; i++) { + mode = TransLineTypes[i].type; + saveMode (stream); + } + file.close(); + + mode = oldmode; + + return true; +} + +// Saves the GUI values into internal data structures. +void QucsTranscalc::storeValues (void) { + struct TransType * t = &TransLineTypes[getTypeIndex ()]; + struct TransValue * val; + for (int box = 0; box < MAX_TRANS_BOXES; box++) { + val = t->array[box].item; + int i = 0; + while (val->name) { + getProperty (val->name); + getUnit (val->name); + if (box == TRANS_PHYSICAL) { + if (val->radio->isEnabled()) { + if (val->radio->isChecked()) + t->radio[i] = 1; + else + t->radio[i] = 0; + } + else t->radio[i] = -1; + } + i++; + val++; + } + } +} + +void QucsTranscalc::slotFileLoad() +{ + statusBar()->showMessage(tr("Loading file...")); + int _mode = 0; + QString s = QFileDialog::getOpenFileName(this, tr("Enter a Filename"), + QucsSettings.QucsWorkDir.path(), + tr("Transcalc File") + " (*.trc)"); + if (!s.isEmpty()) { + QucsSettings.QucsWorkDir.setPath(QDir::cleanPath(s)); + if (!loadFile (s, &_mode)) { + QMessageBox::critical (this, tr("Error"), + tr("Cannot load file:")+" '"+s+"'!"); + } + } + else statusBar()->showMessage(tr("Loading aborted."), 2000); + + statusBar()->showMessage(tr("Ready.")); +} + +void QucsTranscalc::slotFileSave() +{ + statusBar()->showMessage(tr("Saving file...")); + + QString s = QFileDialog::getSaveFileName(this, tr("Enter a Filename"), + QucsSettings.QucsWorkDir.path(), + tr("Transcalc File") + " (*.trc)"); + if (!s.isEmpty()) { + QucsSettings.QucsWorkDir.setPath(QDir::cleanPath(s)); + if (!saveFile (s)) { + QMessageBox::critical (this, tr("Error"), + tr("Cannot save file:")+" '"+s+"'!"); + } + } + else statusBar()->showMessage(tr("Saving aborted."), 2000); + + statusBar()->showMessage(tr("Ready.")); +} + +// Returns the current textual mode. +QString QucsTranscalc::getMode (void) { + return TransLineTypes[getTypeIndex ()].description; +} + +// Sets the mode according to the given textual mode. +void QucsTranscalc::setMode (QString _mode) { + for (int i = 0; i < MAX_TRANS_TYPES; i++) { + if (TransLineTypes[i].description == _mode) { + setMode (TransLineTypes[i].type); + updatePixmap (mode); + break; + } + } +} + +// Updates the combobox and pixmap for the current mode. +void QucsTranscalc::updatePixmap (int _mode) { + pix->setPixmap(QPixmap(":/bitmaps/" + QString(TransLineTypes[_mode].bitmap))); + tranType->setCurrentIndex(_mode); +} + +void QucsTranscalc::slotHelpIntro() +{ + HelpDialog *d = new HelpDialog(this); + d->show(); +} + +void QucsTranscalc::slotOptions() +{ + OptionsDialog *d = new OptionsDialog(this); + d->show(); +} + +// Translates a given textual unit into a array index. +int QucsTranscalc::translateUnit(const char * unit, int type) { + for (int i = 0; TransUnits[type].units[i]; i++) { + if (unit && !strcmp (unit, TransUnits[type].units[i])) + return i; + } + return 0; +} + +void QucsTranscalc::slotRadioChecked(int id) +{ + int idx = getTypeIndex (); + for (int i = 0; i < TransMaxBox[TRANS_PHYSICAL]; i++) { + if (TransLineTypes[idx].radio[i] != -1) { + TransLineTypes[idx].radio[i] = 0; + if (i == id) { + TransLineTypes[idx].radio[i] = 1; + } + } + } + updateSelection (); +} + +void QucsTranscalc::slotCopyToClipBoard() +{ + int created = 0; + QString s = "\n"; + + // create microstrip schematic + if (mode == ModeMicrostrip) { + transline * l = TransLineTypes[ModeMicrostrip].line; + s += "\n"; + s += " \n"; + s +=" \n"; + s += " \n"; + s += " \n"; + s += QString(" \n"). + arg(l->getProperty("Er")). + arg(l->getProperty("H", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("T", UNIT_LENGTH, LENGTH_UM)). + arg(l->getProperty("Tand")). + arg(1 / l->getProperty("Cond")). + arg(l->getProperty("Rough", UNIT_LENGTH, LENGTH_M)); + s += " <.SP SPTC1 1 90 240 0 51 0 0 "; + double freq = l->getProperty("Freq", UNIT_FREQ, FREQ_GHZ); + if (freq > 0) + s += QString("\"log\" 1 \"%1 GHz\" 1 \"%2 GHz\" 1 "). + arg(freq / 10).arg(freq * 10); + else + s += "\"lin\" 1 \"0 GHz\" 1 \"10 GHz\" 1 "; + s += "\"51\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n"; + s += QString(" \n"). + arg(l->getProperty("W", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("L", UNIT_LENGTH, LENGTH_MM)); + s += " \n"; + s += "\n"; + s += "\n"; + s += " <90 100 150 100 \"\" 0 0 0 \"\">\n"; + s += " <90 100 90 120 \"\" 0 0 0 \"\">\n"; + s += " <210 100 270 100 \"\" 0 0 0 \"\">\n"; + s += " <270 100 270 120 \"\" 0 0 0 \"\">\n"; + s += "\n"; + created++; + } + + // create coupled microstrip schematic + else if (mode == ModeCoupledMicrostrip) { + transline * l = TransLineTypes[ModeCoupledMicrostrip].line; + s += "\n"; + s += " \n"; + s += " \n"; + s += " \n"; + s += " \n"; + s += " \n"; + s += " \n"; + s += " \n"; + s += " \n"; + s += QString(" \n"). + arg(l->getProperty("Er")). + arg(l->getProperty("H", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("T", UNIT_LENGTH, LENGTH_UM)). + arg(l->getProperty("Tand")). + arg(1 / l->getProperty("Cond")). + arg(l->getProperty("Rough", UNIT_LENGTH, LENGTH_M)); + s += " <.SP SPTC1 1 100 290 0 51 0 0 "; + double freq = l->getProperty("Freq", UNIT_FREQ, FREQ_GHZ); + if (freq > 0) + s += QString("\"log\" 1 \"%1 GHz\" 1 \"%2 GHz\" 1 "). + arg(freq / 10).arg(freq * 10); + else + s += "\"lin\" 1 \"0 GHz\" 1 \"10 GHz\" 1 "; + s += "\"51\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n"; + s += QString(" \n"). + arg(l->getProperty("W", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("L", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("S", UNIT_LENGTH, LENGTH_MM)); + s += "\n"; + s += "\n"; + s += " <100 80 160 80 \"\" 0 0 0 \"\">\n"; + s += " <100 80 100 100 \"\" 0 0 0 \"\">\n"; + s += " <140 140 140 170 \"\" 0 0 0 \"\">\n"; + s += " <140 140 160 140 \"\" 0 0 0 \"\">\n"; + s += " <320 80 320 100 \"\" 0 0 0 \"\">\n"; + s += " <220 80 320 80 \"\" 0 0 0 \"\">\n"; + s += " <280 140 280 190 \"\" 0 0 0 \"\">\n"; + s += " <220 140 280 140 \"\" 0 0 0 \"\">\n"; + s += "\n"; + created++; + } + + // create coaxial line schematic + else if (mode == ModeCoaxial) { + transline * l = TransLineTypes[ModeCoaxial].line; + s += "\n"; + s += " \n"; + s +=" \n"; + s += " \n"; + s += " \n"; + s += " <.SP SPTC1 1 90 240 0 51 0 0 "; + double freq = l->getProperty("Freq", UNIT_FREQ, FREQ_GHZ); + if (freq > 0) + s += QString("\"log\" 1 \"%1 GHz\" 1 \"%2 GHz\" 1 "). + arg(freq / 10).arg(freq * 10); + else + s += "\"lin\" 1 \"0 GHz\" 1 \"10 GHz\" 1 "; + s += "\"51\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n"; + s += QString(" \n"). + arg(l->getProperty("Er")). + arg(1 / l->getProperty("Sigma")). + arg(l->getProperty("Mur")). + arg(l->getProperty("dout", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("din", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("L", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("Tand")); + s += "\n"; + s += "\n"; + s += " <90 100 150 100 \"\" 0 0 0 \"\">\n"; + s += " <90 100 90 120 \"\" 0 0 0 \"\">\n"; + s += " <210 100 270 100 \"\" 0 0 0 \"\">\n"; + s += " <270 100 270 120 \"\" 0 0 0 \"\">\n"; + s += "\n"; + created++; + } + + // create coplanar schematic + else if (mode == ModeCoplanar) { + transline * l = TransLineTypes[ModeCoplanar].line; + s += "\n"; + s += " \n"; + s +=" \n"; + s += " \n"; + s += " \n"; + s += QString(" \n"). + arg(l->getProperty("Er")). + arg(l->getProperty("H", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("T", UNIT_LENGTH, LENGTH_UM)). + arg(l->getProperty("Tand")). + arg(1 / l->getProperty("Cond")); + s += " <.SP SPTC1 1 90 240 0 51 0 0 "; + double freq = l->getProperty("Freq", UNIT_FREQ, FREQ_GHZ); + if (freq > 0) + s += QString("\"log\" 1 \"%1 GHz\" 1 \"%2 GHz\" 1 "). + arg(freq / 10).arg(freq * 10); + else + s += "\"lin\" 1 \"0 GHz\" 1 \"10 GHz\" 1 "; + s += "\"51\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n"; + s += QString(" \n"). + arg(l->getProperty("W", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("S", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("L", UNIT_LENGTH, LENGTH_MM)); + s += " \n"; + s += "\n"; + s += "\n"; + s += " <90 100 150 100 \"\" 0 0 0 \"\">\n"; + s += " <90 100 90 120 \"\" 0 0 0 \"\">\n"; + s += " <210 100 270 100 \"\" 0 0 0 \"\">\n"; + s += " <270 100 270 120 \"\" 0 0 0 \"\">\n"; + s += "\n"; + created++; + } + + // create coplanar schematic + else if (mode == ModeGroundedCoplanar) { + transline * l = TransLineTypes[ModeGroundedCoplanar].line; + s += "\n"; + s += " \n"; + s +=" \n"; + s += " \n"; + s += " \n"; + s += QString(" \n"). + arg(l->getProperty("Er")). + arg(l->getProperty("H", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("T", UNIT_LENGTH, LENGTH_UM)). + arg(l->getProperty("Tand")). + arg(1 / l->getProperty("Cond")); + s += " <.SP SPTC1 1 90 240 0 51 0 0 "; + double freq = l->getProperty("Freq", UNIT_FREQ, FREQ_GHZ); + if (freq > 0) + s += QString("\"log\" 1 \"%1 GHz\" 1 \"%2 GHz\" 1 "). + arg(freq / 10).arg(freq * 10); + else + s += "\"lin\" 1 \"0 GHz\" 1 \"10 GHz\" 1 "; + s += "\"51\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n"; + s += QString(" \n"). + arg(l->getProperty("W", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("S", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("L", UNIT_LENGTH, LENGTH_MM)); + s += " \n"; + s += "\n"; + s += "\n"; + s += " <90 100 150 100 \"\" 0 0 0 \"\">\n"; + s += " <90 100 90 120 \"\" 0 0 0 \"\">\n"; + s += " <210 100 270 100 \"\" 0 0 0 \"\">\n"; + s += " <270 100 270 120 \"\" 0 0 0 \"\">\n"; + s += "\n"; + created++; + } + + // rectangular waveguide schematic + else if (mode == ModeRectangular) { + transline * l = TransLineTypes[ModeRectangular].line; + s += "\n"; + s += " \n"; + s +=" \n"; + s += " \n"; + s += " \n"; + s += " <.SP SPTC1 1 90 240 0 51 0 0 "; + double freq = l->getProperty("Freq", UNIT_FREQ, FREQ_GHZ); + if (freq > 0) + s += QString("\"log\" 1 \"%1 GHz\" 1 \"%2 GHz\" 1 "). + arg(freq / 10).arg(freq * 10); + else + s += "\"lin\" 1 \"0 GHz\" 1 \"10 GHz\" 1 "; + s += "\"51\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n"; + s += QString(" \n"). + arg(l->getProperty("a", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("b", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("L", UNIT_LENGTH, LENGTH_MM)). + arg(l->getProperty("Er")). + arg(l->getProperty("Mur")). + arg(l->getProperty("Tand")). + arg(1 / l->getProperty("Cond")); + s += " \n"; + s += "\n"; + s += "\n"; + s += " <90 100 150 100 \"\" 0 0 0 \"\">\n"; + s += " <90 100 90 120 \"\" 0 0 0 \"\">\n"; + s += " <210 100 270 100 \"\" 0 0 0 \"\">\n"; + s += " <270 100 270 120 \"\" 0 0 0 \"\">\n"; + s += "\n"; + created++; + } + + // put resulting transmission line schematic into clipboard + QClipboard *cb = QApplication::clipboard(); + cb->setText(s); + + // put a message into status line + if (created) + statusBar()->showMessage(tr("Schematic copied into clipboard."), 2000); + else + statusBar()->showMessage(tr("Transmission line type not available."), 2000); +} diff --git a/qucs-transcalc/qucstrans.h b/qucs-transcalc/qucstrans.h new file mode 100644 index 00000000..e5975eb4 --- /dev/null +++ b/qucs-transcalc/qucstrans.h @@ -0,0 +1,200 @@ +/*************************************************************************** + qucstrans.h - description + ------------------- + begin : Sun Feb 27 2005 + copyright : (C) 2005 by Stefan Jahn + email : stefan@lkcc.org + ***************************************************************************/ + +/*************************************************************************** + * * + * This program is free software; you can redistribute it and/or modify * + * it under the terms of the GNU General Public License as published by * + * the Free Software Foundation; either version 2 of the License, or * + * (at your option) any later version. * + * * + ***************************************************************************/ + +#ifndef QUCSTRANS_H +#define QUCSTRANS_H + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +class QComboBox; +class QLineEdit; +class QLabel; +class QGroupBox; +class QVBoxLayout; +class QRadioButton; +class QGridLayout; +class QStatusBar; +class QTextStream; +class QButtonGroup; +class QStackedWidget; + +class transline; + +// Current limit defintions. +#define MAX_TRANS_BOXES 4 +#define MAX_TRANS_TYPES 7 //Number of transmission lines +#define MAX_TRANS_RESULTS 7 + +// Types of transmission line properties. +#define TRANS_SUBSTRATE 0 +#define TRANS_COMPONENT 1 +#define TRANS_PHYSICAL 2 +#define TRANS_ELECTRICAL 3 + +#define TRANS_FREQS { "GHz", "Hz", "kHz", "MHz", NULL } +#define TRANS_OHMS { "Ohm", "kOhm", NULL } +#define TRANS_ANGLES { "Deg", "Rad", NULL } +#define TRANS_LENGTHS { "mil", "cm", "mm", "m", "um", "in", "ft", NULL } +#define TRANS_NONES { "NA", NULL } + +// Application settings. +struct tQucsSettings { + int x, y, dx, dy; // position and size of main window + QFont font; // font + QString LangDir; // translation directory + QString Language; + int length_unit; // default length unit + int freq_unit; // default frequency unit + int res_unit; // default resistance unit + int ang_unit; // default angle unit + QString Mode; // current mode + QDir QucsWorkDir; + QDir QucsHomeDir; +}; + +extern tQucsSettings QucsSettings; + +// Transmission line value. +struct TransValue { + const char * name; // label + double value; // value + QString * tip; // tool tip description + const char * units[8]; // unit choise + int unit; // unit index + QLabel * label; // Qt label widget + QLineEdit * lineedit; // Qt value widget + QComboBox * combobox; // Qt unit widget + QRadioButton * radio; // Qt fixed widget +}; + +// Array of transmission line values. +struct TransArray { + struct TransValue item[12]; +}; + +// Extraneous calculation results. +struct TransResult { + QString * name; // textual description + QLabel * label; // Qt name widget + QLabel * value; // Qt value widget +}; + +// Transmission line types. +enum TransMode { + ModeMicrostrip = 0, + ModeCoplanar = 1, + ModeGroundedCoplanar = 2, + ModeRectangular = 3, + ModeCoaxial = 4, + ModeCoupledMicrostrip = 5, + ModeStripline = 6, + ModeNone +}; + +// A transmission line type structure. +struct TransType { + int type; // type of transmission line (see TransMode) + const char * description; // description + const char * bitmap; // bitmap file name + transline * line; // transmission line instance + struct TransArray array[MAX_TRANS_BOXES]; + int results; // number of extraneous results + struct TransResult result[MAX_TRANS_RESULTS]; + int radio[4]; +}; + +struct TransUnit { + const char * description; + const char * units[8]; +}; + +/** + *@author Stefan Jahn + */ + +class QucsTranscalc : public QMainWindow { + Q_OBJECT +public: + QucsTranscalc(); + ~QucsTranscalc(); + + void setProperty (QString, double); + double getProperty (QString); + void setUnit (QString, const char *); + char * getUnit (QString); + void setResult (int, const char *); + bool isSelected (QString); + + void saveMode (QTextStream&); + bool saveModes (QString); + bool loadFile (QString, int * _mode = 0); + QString getMode (void); + void setMode (QString); + static int translateUnit(const char *, int); + +private slots: + void slotAbout(); + void slotQuit(); + void slotSelectType(int); + void slotSynthesize(); + void slotAnalyze(); + void slotValueChanged(); + void slotFileLoad(); + void slotFileSave(); + void slotHelpIntro(); + void slotOptions(); + void slotRadioChecked(int); + void slotCopyToClipBoard(); + +private: + void updateSelection (); + void createPropItem (QGridLayout *, TransValue *, int, QButtonGroup *); + void createResultItem (QGridLayout *, TransResult *); + void updateResultItem (TransResult *); + void createResultItems (QGroupBox *); + void updateResultItems (); + void createPropItems (QGroupBox *, int); + int getTypeIndex (); + void updatePropItem (TransValue *); + void setMode (int); + struct TransValue * findProperty (QString); + void setupTranslations (); + bool saveFile (QString); + void updateMode (void); + void storeValues (void); + void updatePixmap (int); + + private: + void closeEvent (QCloseEvent*); + +private: + QStatusBar * statBar; + QLabel * pix; + QComboBox * tranType; + QGroupBox * calculated; + int mode; +}; + +#endif /* QUCSTRANS_H */ diff --git a/qucs-transcalc/qucstrans_.qrc b/qucs-transcalc/qucstrans_.qrc new file mode 100644 index 00000000..7c87b8ad --- /dev/null +++ b/qucs-transcalc/qucstrans_.qrc @@ -0,0 +1,12 @@ + + + bitmaps/c_microstrip.png + bitmaps/coax.png + bitmaps/cpw_back.png + bitmaps/cpw.png + bitmaps/microstrip.png + bitmaps/stripline.png + bitmaps/rectwaveguide.png + bitmaps/big.qucs.xpm + + diff --git a/qucs-transcalc/rectwaveguide.cpp b/qucs-transcalc/rectwaveguide.cpp new file mode 100644 index 00000000..5686cfa9 --- /dev/null +++ b/qucs-transcalc/rectwaveguide.cpp @@ -0,0 +1,352 @@ +/* + * rectwaveguide.cpp - rectangular waveguide class implementation + * + * Copyright (C) 2001 Gopal Narayanan + * Copyright (C) 2005, 2006 Stefan Jahn + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + +#include +#include +#include + +#include "units.h" +#include "transline.h" +#include "rectwaveguide.h" + +rectwaveguide::rectwaveguide() : transline() +{ +} + +rectwaveguide::~rectwaveguide() +{ +} + +/* + * returns k + */ +double rectwaveguide::kval () +{ + double kval; + kval = 2.0 * pi * f * sqrt (mur * er) / C0; + return kval; +} + +/* + * given mode numbers m and n + * returns cutoff kc value + */ +double rectwaveguide::kc (int m, int n) +{ + double kcval; + kcval = sqrt (pow ((m * pi / a), 2.0) + pow ((n * pi / b), 2.0)); + return kcval; +} + +/* + * given mode numbers m and n + * returns cutoff fc value + */ +double rectwaveguide::fc (int m, int n) +{ + double fcval; + fcval = kc (m, n) * C0 / (2.0 * pi * sqrt (mur * er)); + return fcval; +} + +/* + * alphac - returns attenuation due to conductor losses for all propagating + * modes in the waveguide + */ +double rectwaveguide::alphac () +{ + double Rs, f_c; + double ac; + short m, n, mmax, nmax; + + Rs = sqrt ((pi * f * mur * MU0) / sigma); + ac = 0.0; + mmax = (int) floor (f / fc (1,0)); + nmax = mmax; + + /* below from Ramo, Whinnery & Van Duzer */ + + /* TE(m,n) modes */ + for (n = 0; n<= nmax; n++){ + for (m = 1; m <= mmax; m++){ + f_c = fc(m, n); + if (f > f_c) { + switch (n) { + case 0: + ac += (Rs/(b * ZF0 * sqrt(1.0 - pow((f_c/f),2.0)))) * + (1.0 + ((2 * b/a)*pow((f_c/f),2.0))); + break; + default: + ac += ((2. * Rs)/(b * ZF0 * sqrt(1.0 - pow((f_c/f),2.0)))) * + (((1. + (b/a))*pow((f_c/f),2.0)) + + ((1. - pow((f_c/f),2.0)) * (((b/a)*(((b/a)*pow(m,2.)) + pow(n,2.)))/ + (pow((b*m/a),2.0) + pow(n,2.0))))); + break; + } + } + } + } + + /* TM(m,n) modes */ + for (n = 1; n<= nmax; n++) { + for (m = 1; m<= mmax; m++) { + f_c = fc(m, n); + if (f > f_c) { + ac += ((2. * Rs)/(b * ZF0 * sqrt(1.0 - pow((f_c/f),2.0)))) * + (((pow(m,2.0)*pow((b/a),3.0)) + pow(n,2.))/ + ((pow((m*b/a),2.)) + pow(n,2.0))); + } + } + } + + ac = ac * 20.0 * log10 (exp (1.0)); /* convert from Np/m to db/m */ + return ac; +} + +/* + * alphac_cutoff - returns attenuation for a cutoff wg + */ +double rectwaveguide::alphac_cutoff () +{ + double acc; + acc = sqrt (pow (kc(1,0), 2.0) - pow (kval (), 2.0)); + acc = 20 * log10 (exp (1.0)) * acc; + return acc; +} + +/* + * returns attenuation due to dielectric losses + */ +double rectwaveguide::alphad() +{ + double k, beta; + double ad; + + k = kval (); + beta = sqrt (pow (k, 2.0) - pow (kc (1,0), 2.0)); + + ad = (pow (k, 2.0) * tand) / (2.0 * beta); + ad = ad * 20.0 * log10 (exp (1.0)); /* convert from Np/m to db/m */ + return ad; +} + +/* + * get_rectwaveguide_sub + * get and assign rectwaveguide substrate parameters + * into rectwaveguide structure + */ +void rectwaveguide::get_rectwaveguide_sub () +{ + er = getProperty ("Er"); + mur = getProperty ("Mur"); + sigma = getProperty ("Cond"); + tand = getProperty ("Tand"); + tanm = getProperty ("TanM"); +} + +/* + * get_rectwaveguide_comp + * get and assign rectwaveguide component parameters + * into rectwaveguide structure + */ +void rectwaveguide::get_rectwaveguide_comp () +{ + f = getProperty ("Freq", UNIT_FREQ, FREQ_HZ); +} + +/* + * get_rectwaveguide_elec + * get and assign rectwaveguide electrical parameters + * into rectwaveguide structure + */ +void rectwaveguide::get_rectwaveguide_elec () +{ + Z0 = getProperty ("Z0", UNIT_RES, RES_OHM); + ang_l = getProperty ("Ang_l", UNIT_ANG, ANG_RAD); +} + +/* + * get_rectwaveguide_phys + * get and assign rectwaveguide physical parameters + * into rectwaveguide structure + */ +void rectwaveguide::get_rectwaveguide_phys () +{ + a = getProperty ("a", UNIT_LENGTH, LENGTH_M); + b = getProperty ("b", UNIT_LENGTH, LENGTH_M); + l = getProperty ("L", UNIT_LENGTH, LENGTH_M); +} + +/* + * analyze - analysis function + */ +void rectwaveguide::analyze () +{ + double lambda_g; + double k; + double beta; + + /* Get and assign substrate parameters */ + get_rectwaveguide_sub(); + + /* Get and assign component parameters */ + get_rectwaveguide_comp(); + + /* Get and assign physical parameters */ + get_rectwaveguide_phys(); + + k = kval (); + + if (kc (1,0) <= k) { + /* propagating modes */ + beta = sqrt (pow (k, 2.0) - pow (kc (1,0), 2.0)); + lambda_g = 2.0 * pi / beta; + /* Z0 = (k * ZF0) / beta; */ + Z0 = k * ZF0 * sqrt(mur/er) / beta; + + /* calculate electrical angle */ + lambda_g = 2.0 * pi / beta; + ang_l = 2.0 * pi * l / lambda_g; /* in radians */ + atten_cond = alphac () * l; + atten_dielectric = alphad () * l; + er_eff = (1.0 - pow ((fc (1,0) / f), 2.0)); + } else { + /* evanascent modes */ + Z0 = 0; + ang_l = 0; + er_eff = 0; + atten_dielectric = 0.0; + atten_cond = alphac_cutoff () * l; + } + + setProperty ("Z0", Z0, UNIT_RES, RES_OHM); + setProperty ("Ang_l", ang_l, UNIT_ANG, ANG_RAD); + + show_results (); +} + +/* + * synthesize - synthesis function + */ +int rectwaveguide::synthesize () +{ + double lambda_g, k, beta; + + /* Get and assign substrate parameters */ + get_rectwaveguide_sub(); + + /* Get and assign component parameters */ + get_rectwaveguide_comp(); + + /* Get and assign electrical parameters */ + get_rectwaveguide_elec(); + + /* Get and assign physical parameters */ + get_rectwaveguide_phys(); + + + if (isSelected ("b")) { + /* solve for b */ + b = Z0 * a * sqrt(1.0 - pow((fc(1,0)/f),2.0))/ + (2. * ZF0); + setProperty ("b", b, UNIT_LENGTH, LENGTH_M); + } else if (isSelected ("a")) { + /* solve for a */ + a = sqrt(pow((2.0 * ZF0 * b/Z0), 2.0) + + pow((C0/(2.0 * f)),2.0)); + setProperty ("a", a, UNIT_LENGTH, LENGTH_M); + } + + k = kval (); + beta = sqrt(pow(k,2.) - pow(kc(1,0),2.0)); + lambda_g = (2. * pi)/beta; + l = (ang_l * lambda_g)/(2.0 * pi); /* in m */ + + setProperty ("L", l, UNIT_LENGTH, LENGTH_M); + + if (kc(1,0) <= k) { + /*propagating modes */ + beta = sqrt(pow(k,2.) - pow(kc(1,0),2.0)); + lambda_g = (2. * pi)/beta; + atten_cond = alphac () * l; + atten_dielectric = alphad () * l; + er_eff = (1.0 - pow((fc(1,0)/f),2.0)); + } else { + /*evanascent modes */ + Z0 = 0; + ang_l = 0; + er_eff = 0; + atten_dielectric = 0.0; + atten_cond = alphac_cutoff () * l; + } + + show_results (); + + return 0; +} + +void rectwaveguide::show_results () +{ + short m, n, mmax, nmax; + + setResult (0, er_eff, ""); + setResult (1, atten_cond, "dB"); + setResult (2, atten_dielectric, "dB"); + + setResult (3, "none"); + if (f >= (2.*fc(1,0))) { + char text[256], txt[256]; + strcpy (text, ""); + /* multiple modes possible in waveguide */ + /* mmax = floor(f/fc(1,0));*/ + mmax = 5; + nmax = mmax; + for (m = 2; m<= mmax; m++) { + for (n=0; n<= nmax; n++) { + if (f >= (fc(m,n))){ + sprintf(txt,"TE(%u,%u) ",m, n); + strcat(text,txt); + } + } + } + setResult (3, text); + } + + setResult (4, "none"); + if (f >= fc(1,1)){ /*TM(1,1) mode possible*/ + char text[256], txt[256]; + strcpy (text, ""); + /* mmax = floor(f/fc(1,1));*/ + mmax = 5; + nmax = mmax; + for (m = 1; m<= mmax; m++) { + for (n=1; n<= nmax; n++) { + if (f >= (fc(m,n))){ + sprintf(txt,"TM(%u,%u) ",m, n); + strcat(text,txt); + } + } + } + setResult (3, text); + } +} diff --git a/qucs-transcalc/rectwaveguide.h b/qucs-transcalc/rectwaveguide.h new file mode 100644 index 00000000..c31da7c3 --- /dev/null +++ b/qucs-transcalc/rectwaveguide.h @@ -0,0 +1,65 @@ +/* + * rectwaveguide.h - rectangular waveguide class definition + * + * Copyright (C) 2001 Gopal Narayanan + * Copyright (C) 2005 Stefan Jahn + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + +#ifndef __RECTWAVEGUIDE_H +#define __RECTWAVEGUIDE_H + +class rectwaveguide : public transline { + public: + rectwaveguide(); + ~rectwaveguide(); + + private: + double er; /* dielectric constant */ + double tand; /* Dielectric Loss Tangent */ + double tanm; /* Magnetic Loss Tangent */ + double a; /* width of waveguide */ + double b; /* height of waveguide */ + double l; /* length of waveguide */ + double Z0; /* characteristic impedance */ + double ang_l; /* Electrical length in angle */ + double er_eff; /* Effective dielectric constant */ + //double mur_eff; /* Effective mag. permeability */ + double atten_dielectric; /* Loss in dielectric (dB) */ + double atten_cond; /* Loss in conductors (dB) */ + //double fc10; /* Cutoff frequency for TE10 mode */ + + public: + void analyze (); + int synthesize (); + + private: + double kval (); + double kc (int, int); + double fc (int, int); + double alphac (); + double alphac_cutoff (); + double alphad (); + void get_rectwaveguide_sub (); + void get_rectwaveguide_comp (); + void get_rectwaveguide_phys (); + void get_rectwaveguide_elec (); + void show_results (); +}; + +#endif /* __RECTWAVEGUIDE_H */ diff --git a/qucs-transcalc/stripline.cpp b/qucs-transcalc/stripline.cpp new file mode 100644 index 00000000..8ee310e4 --- /dev/null +++ b/qucs-transcalc/stripline.cpp @@ -0,0 +1,222 @@ +/* + * stripline.cpp - stripline class implementation + * + * Copyright (C) 2016 Andres Martinez-Mera + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or (at + * your option) any later version. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + + +#include +#include +#include +#include + +#include "units.h" +#include "transline.h" +#include "stripline.h" + +stripline::stripline() : transline() +{ +} + +stripline::~stripline() +{ +} + +/* + * get_stripline_sub() - get and assign stripline substrate parameters into stripline + * structure + */ +void stripline::get_stripline_sub () +{ + er = getProperty ("Er"); + mur = getProperty ("Mur"); + tand = getProperty ("Tand"); + sigma = getProperty ("Sigma"); + t = getProperty ("T", UNIT_LENGTH, LENGTH_M); + h = getProperty ("h", UNIT_LENGTH, LENGTH_M); +} + +/* + * get_stripline_comp() - get and assign stripline component parameters into + * stripline structure + */ +void stripline::get_stripline_comp () +{ + f = getProperty ("Freq", UNIT_FREQ, FREQ_HZ); + lambda_0 = C0/f; +} + +/* + * get_stripline_elec() - get and assign stripline electrical parameters into + * stripline structure + */ +void stripline::get_stripline_elec () +{ + Z0 = getProperty ("Z0", UNIT_RES, RES_OHM); + ang_l = getProperty ("Ang_l", UNIT_ANG, ANG_RAD); +} + +/* + * get_stripline_phys() - get and assign stripline physical parameters into stripline + * structure + */ +void stripline::get_stripline_phys () +{ + W = getProperty ("W", UNIT_LENGTH, LENGTH_M); + l = getProperty ("L", UNIT_LENGTH, LENGTH_M); +} + +/* This function calculates the dielectric attenuation coefficient*/ +double stripline::alphad_stripline () +{ + return 27.3*sqrt(er)*tand/lambda_0; +} + +/* This function calculates the conductor attenuation coefficient*/ +double stripline::alphac_stripline () +{ + double w = 2*pi*f, alpha_c; + double mu = mur*pi*4e-7; + double Rs = sqrt(w*mu/(2*sigma));//Skin effect resistance + skindepth = sqrt(2/(w*mu*sigma)); + double A = 1 + 2*W/(2*h-t) + (1/pi)*((2*h+t)/(2*h-t))*log((4*h-t)/t); + double B = 1 + (2*h/(0.5*W+0.7*t))*(0.5 + 0.414*t/W + (1/(2*pi))*log(4*pi*W/t)); + + (Z0 < 120/sqrt(er)) ? + alpha_c = (23.4e-3*Rs*er*Z0*A)/(30*pi*(2*h-t)) : + alpha_c = 1.4*Rs*B/(Z0*2*h); + + return alpha_c; +} + +/* + * analyze() - Reads both the substrate and the physical properties of the line and calculates Z0 and the electrical length + */ +void stripline::analyze () +{ + /* Get and assign substrate parameters */ + get_stripline_sub(); + + /* Get and assign component parameters */ + get_stripline_comp(); + + /* Get and assign physical parameters */ + get_stripline_phys(); + + calculateZ0(); // Z0 + getStriplineLength();// Line length + alphac_stripline();// alpha_c and skin depth + alphad_stripline();// alpha_d + setProperty ("Z0", Z0, UNIT_RES, RES_OHM); + setProperty ("Ang_l", ang_l, UNIT_ANG, ANG_RAD); + show_results(); +} + + +// This function calculates the characteristic impedance of a symmetric stripline according to [1], eq. 4.80-4.84 +double stripline::getZ0fromWidth(double W_) +{ + double x = t/(2.*h); + double m= 2/(1 + 2*x/3*(1-x)); + double B = (x/(pi*(1-x)))*(1 - 0.5*log(pow(x/(2.-x),2) + pow((0.0796*x)/((W_/(2.*h)) + 1.1*x),m))); + double A = 1/((W_/(2*h - t)) + B); + // Line impedance + return (30/sqrt(er))*log(1 + (4/pi)*A*((8/pi)*A + sqrt(pow((8/pi)*A,2) + 6.27))); +} + +/* This function calculates the static line impedance*/ +void stripline::calculateZ0() +{ + Z0 = getZ0fromWidth(W); +} + +void stripline::getStriplineLength() +{ + double lambda_g = (C0/(f))/sqrt(er * mur); + /* calculate electrical angle */ + ang_l = (2.0 * pi * l)/lambda_g; /* in radians */ +} + +/* + * + */ +int stripline::synthesize () +{ + get_stripline_sub();//Substrate + get_stripline_elec();//Z0 and electrical length + get_stripline_phys();//Line size + alphac_stripline();// alpha_c and skin depth + alphad_stripline();// alpha_d + + + // Zero-thickness approximation + double B = exp(Z0*sqrt(er)/30)-1; + double C = sqrt(4*B+6.27); + double A = 2*B/C; + double A1 = 8/(pi*A); + + double x = t/(2*h); + double m= 2/(1 + 2*x/3*(1-x)); + double We = (2*h - t)*A1; + double A2 = (x/(pi*(1-x))) * (1 - 0.5*log( (x*x/(4-2*x+x*x)) + pow((0.0796*x/( (We/(2*h)) + 1.1*x)),m) )); + + double Wi = (A1 - A2)*(2*h-t);//Width given by the zero-thickness approximation (Initial guess for the refinement) + + double Zi; + double dW = Wi*1e-4;//Differential width + unsigned int MAX_ITER = 100, iter = 0; + double MAX_ERR = 1e-5; + double Zi_1 = 0, Zi_diff, step; + + + // The Newton-Raphson method is employed to refine the width given by the zero-thickness approximation with the analysis formulae. + while ((std::abs(Z0 - Zi_1) > MAX_ERR)&&(iter < MAX_ITER))//Stop condition: |Z0 - Z_i| < MAX_ERR or max. number iterations exceeded + { + Zi = getZ0fromWidth(Wi+dW); + Zi_diff = (Zi - Zi_1) / dW; + step = (Z0 - Zi_1) / Zi_diff - dW; + Wi += step+dW; + if (Wi <= 0.0) Wi = dW; + Zi_1 = getZ0fromWidth(Wi); + iter++; + } + + setProperty ("W", Wi, UNIT_LENGTH, LENGTH_M); + double lambda_g = lambda_0/sqrt(er * mur); + /* calculate physical length */ + l = (lambda_g * ang_l)/(2.0 * pi); /* in m */ + setProperty ("L", l, UNIT_LENGTH, LENGTH_M); + + show_results(); + return 0; +} + +/* + * show_results() - show results + */ +void stripline::show_results() +{ + atten_dielectric = alphad_stripline () * l; + atten_cond = alphac_stripline () * l; + + setResult (0, atten_cond, "dB"); + setResult (1, atten_dielectric, "dB"); + double val = convertProperty ("T", skindepth, UNIT_LENGTH, LENGTH_M); + setResult (2, val, getUnit ("T"));//Skin depth +} diff --git a/qucs-transcalc/stripline.h b/qucs-transcalc/stripline.h new file mode 100644 index 00000000..91524a3e --- /dev/null +++ b/qucs-transcalc/stripline.h @@ -0,0 +1,70 @@ +/* + * stripline.h - stripline class definition + * + * Copyright (C) 2016 Andres Martinez-Mera + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + +#ifndef __STRIPLINE_H +#define __STRIPLINE_H + +/* +Reference: +[1] Analysis Methods for RF, Microwave and Milimeter-Wave Planar Transmission Line Structures, +Cam Nguyen. John Wiley and Sons Inc., 2001, Pages 76 - 78 +*/ + +class stripline : public transline { + public: + stripline(); + ~stripline(); + + private: + double er; /* dielectric constant */ + double tand; /* Dielectric Loss Tangent */ + double t; /* Thickness */ + double h; /* Substrate height */ + double W; /* Width of the conductor strip */ + double l; /* Length of cable */ + double Z0; /* characteristic impedance */ + double ang_l; /* Electrical length in angle */ + double er_eff; /* Effective dielectric constant */ + double atten_dielectric; /* Loss in dielectric (dB) */ + double atten_cond; /* Loss in conductors (dB) */ + double lambda_0; /* Wavelength */ + + public: + void analyze (); + int synthesize (); + + private: + void get_stripline_sub(); + void get_stripline_comp(); + void get_stripline_phys(); + void get_stripline_elec(); + void fixdin(); + void fixdout(); + double alphad_stripline(); + double alphac_stripline(); + void show_results(); + void getStriplineLength(); + void calculateZ0(); + double getZ0fromWidth(double); +}; + +#endif /* __stripline_H */ diff --git a/qucs-transcalc/transline.cpp b/qucs-transcalc/transline.cpp new file mode 100644 index 00000000..ae499af7 --- /dev/null +++ b/qucs-transcalc/transline.cpp @@ -0,0 +1,187 @@ +/* + * transline.cpp - base for a transmission line implementation + * + * Copyright (C) 2005 Stefan Jahn + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + +#include "qucstrans.h" +#include "transline.h" +#include "units.h" + +// Unit conversion array for length. +static double conv_length[7][7] = { + { 1.0, 2.54e-3, 2.54e-2, 2.54e-5, 25.4, 1.e-3, 1./12000}, + {1./2.54e-3, 1.0, 10.0, 1.e-2, 1.e4, 1./2.54, 1./30.48}, + {1./2.54e-2, 1./10., 1.0, 1.e-3, 1.e3, 1./25.4, 1./304.8}, + {1./2.54e-5, 1.e2, 1.e3, 1.0, 1.e6, 1./2.54e-2, 1./0.3048}, + {1./25.4, 1.e-4, 1.e-3, 1.e-6, 1.0, 1./2.54e4, 1./3.048e5}, + {1.e3, 2.54, 25.4, 2.54e-2, 2.54e4, 1.0, 1./12.}, + {1.2e4, 30.48, 304.8, 0.3048, 3.048e5, 12.0, 1.0} +}; + +// Unit conversion array for frequencies. +static double conv_freq[4][4] = { + { 1.0, 1.e9, 1.e6, 1.e3}, + { 1.e-9, 1.0, 1.e-3, 1.e-6}, + { 1.e-6, 1.e3, 1.0, 1.e-3}, + { 1.e-3, 1.e6, 1.e3, 1.0} +}; + +// Unit conversion array for resistances. +static double conv_res[2][2] = { + {1.0, 1.e-3}, + {1.e3, 1.0} +}; + +// Unit conversion array for angles. +static double conv_ang[2][2] = { + {1.0, pi/180.0}, + {180.0/pi, 1.0} +}; + +/* Constructor creates a transmission line instance. */ +transline::transline () { + app = 0; + mur = 1.0; +} + +/* Destructor destroys a transmission line instance. */ +transline::~transline () { +} + +/* Sets the application instance. */ +void transline::setApplication (QucsTranscalc * a) { + app = a; +} + +/* Sets a named property to the given value, access through the + application. */ +void transline::setProperty (const char * prop, double value) { + app->setProperty (prop, value); +} + +/* Sets a named property to a given value. Depending on the source + and destination unit the value gets previously converted. */ +void transline::setProperty (const char * prop, double value, int type, + int srcunit) { + int dstunit = translateUnit (getUnit (prop)); + if (type == UNIT_LENGTH) + value *= conv_length[srcunit][dstunit]; + else if (type == UNIT_RES) + value *= conv_res[srcunit][dstunit]; + else if (type == UNIT_ANG) + value *= conv_ang[srcunit][dstunit]; + else if (type == UNIT_FREQ) + value *= conv_freq[srcunit][dstunit]; + setProperty (prop, value); +} + +/* Converts the given value/unit pair into a text representation and + puts this into the given result line. */ +void transline::setResult (int line, double value, const char * unit) { + char text[256]; + sprintf (text, "%g %s", value, unit); + app->setResult (line, text); +} + +/* Puts the text into the given result line. */ +void transline::setResult (int line, const char * text) { + app->setResult (line, text); +} + +/* Returns a named property value. */ +double transline::getProperty (const char * prop) { + return app->getProperty (prop); +} + +/* Returns a named property selection. */ +bool transline::isSelected (const char * prop) { + return app->isSelected (prop); +} + +/* Returns a named property value. Depending on the source and + destination unit the actual value is converted. */ +double transline::getProperty (const char * prop, int type, int dstunit) { + int srcunit = translateUnit (getUnit (prop)); + double value = getProperty (prop); + if (type == UNIT_LENGTH) + return value * conv_length[srcunit][dstunit]; + else if (type == UNIT_RES) + return value * conv_res[srcunit][dstunit]; + else if (type == UNIT_ANG) + return value * conv_ang[srcunit][dstunit]; + else if (type == UNIT_FREQ) + return value * conv_freq[srcunit][dstunit]; + return value; +} + +/* The function converts the given value depending on the requested + unit and its source unit. */ +double transline::convertProperty (const char * prop, double value, int type, + int srcunit) { + int dstunit = translateUnit (getUnit (prop)); + if (type == UNIT_LENGTH) + value *= conv_length[srcunit][dstunit]; + else if (type == UNIT_RES) + value *= conv_res[srcunit][dstunit]; + else if (type == UNIT_ANG) + value *= conv_ang[srcunit][dstunit]; + else if (type == UNIT_FREQ) + value *= conv_freq[srcunit][dstunit]; + return value; +} + +/* Returns the unit of the given property. */ +char * transline::getUnit (const char * prop) { + return app->getUnit (prop); +} + +/* The function translates the given textual unit into an + identifier. */ +int transline::translateUnit (char * text) { + if (!strcmp (text, "mil")) return LENGTH_MIL; + else if (!strcmp (text, "cm")) return LENGTH_CM; + else if (!strcmp (text, "mm")) return LENGTH_MM; + else if (!strcmp (text, "m")) return LENGTH_M; + else if (!strcmp (text, "um")) return LENGTH_UM; + else if (!strcmp (text, "in")) return LENGTH_IN; + else if (!strcmp (text, "ft")) return LENGTH_FT; + + else if (!strcmp (text, "GHz")) return FREQ_GHZ; + else if (!strcmp (text, "Hz")) return FREQ_HZ; + else if (!strcmp (text, "kHz")) return FREQ_KHZ; + else if (!strcmp (text, "MHz")) return FREQ_MHZ; + + else if (!strcmp (text, "Ohm")) return RES_OHM; + else if (!strcmp (text, "kOhm")) return RES_KOHM; + + else if (!strcmp (text, "Deg")) return ANG_DEG; + else if (!strcmp (text, "Rad")) return ANG_RAD; + return -1; +} + +/* + * skin_depth - calculate skin depth + */ +double transline::skin_depth() +{ + double depth; + depth = 1.0 / (sqrt(pi * f * mur * MU0 * sigma)); + return depth; +} diff --git a/qucs-transcalc/transline.h b/qucs-transcalc/transline.h new file mode 100644 index 00000000..2967b17d --- /dev/null +++ b/qucs-transcalc/transline.h @@ -0,0 +1,60 @@ +/* + * transline.h - base for a transmission line class definition + * + * Copyright (C) 2005 Stefan Jahn + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + +#ifndef __TRANSLINE_H +#define __TRANSLINE_H + +class QucsTranscalc; + + +class transline { + public: + transline (); + virtual ~transline (); + + void setApplication (QucsTranscalc *); + void setProperty (const char *, double); + void setProperty (const char *, double, int, int); + double getProperty (const char *); + double getProperty (const char *, int, int); + double convertProperty (const char *, double, int, int); + void setResult (int, double, const char *); + void setResult (int, const char *); + int translateUnit (char *); + char * getUnit (const char *); + bool isSelected (const char *); + + virtual int synthesize () { return -1; }; + virtual void analyze () { }; + + protected: + double f; /* Frequency of operation */ + double sigma; /* Conductivity of the metal */ + double mur; /* mag. permeability */ + double skindepth; /* Skin depth */ + double skin_depth(); + + private: + QucsTranscalc * app; +}; + +#endif /* __TRANSLINE_H */ diff --git a/qucs-transcalc/units.h b/qucs-transcalc/units.h new file mode 100644 index 00000000..0b3f2158 --- /dev/null +++ b/qucs-transcalc/units.h @@ -0,0 +1,75 @@ +/* + * units.h - some conversion definitions + * + * Copyright (C) 2001 Gopal Narayanan + * Copyright (C) 2005, 2006 Stefan Jahn + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this package; see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, + * Boston, MA 02110-1301, USA. + * + */ + +#ifndef __UNITS_H +#define __UNITS_H + +#include + + +static const double pi = 3.1415926535897932384626433832795029; /* pi */ +static const double pi_over_2 = 1.5707963267948966192313216916397514; /* pi/2 */ +static const double e = 2.7182818284590452353602874713526625; /* e */ + +static const double MU0 = 12.566370614e-7; /* magnetic constant */ +static const double C0 = 299792458.0; /* speed of light in vacuum */ +static const double ZF0 = 376.73031346958504364963; /* wave resistance in vacuum */ + +#ifndef INFINITY +const double INFINITY = -log (0.0); +#endif + +static const double NR_EPSI = 2.2204460492503131e-16; +static const double MAX_ERROR = 0.000001; + +// Types of units. +#define UNIT_FREQ 0 +#define UNIT_LENGTH 1 +#define UNIT_RES 2 +#define UNIT_ANG 3 + +// Frequency units. +#define FREQ_GHZ 0 +#define FREQ_HZ 1 +#define FREQ_KHZ 2 +#define FREQ_MHZ 3 + +// Length units. +#define LENGTH_MIL 0 +#define LENGTH_CM 1 +#define LENGTH_MM 2 +#define LENGTH_M 3 +#define LENGTH_UM 4 +#define LENGTH_IN 5 +#define LENGTH_FT 6 + +// Resistance units. +#define RES_OHM 0 +#define RES_KOHM 1 + +// Angle units. +#define ANG_DEG 0 +#define ANG_RAD 1 + + +#endif /* __UNITS_H */ diff --git a/qucs/qucs_actions.cpp b/qucs/qucs_actions.cpp index 2a813551..a8a311c6 100644 --- a/qucs/qucs_actions.cpp +++ b/qucs/qucs_actions.cpp @@ -322,7 +322,7 @@ void QucsApp::slotEditPaste(bool on) editPaste->setChecked(false); // release toolbar button editPaste->blockSignals(false); return; - } + } else { // if it's not a text doc, prevent the user from editing // while we perform the paste operation @@ -814,7 +814,7 @@ void QucsApp::slotCallActiveFilter() // Is called to start the transmission line calculation program. void QucsApp::slotCallLine() { - launchTool("qucstrans", "line calculation",QStringList(),true); + launchTool(QUCS_NAME "trans", "line calculation",QStringList(),true); } // ------------------------------------------------------------------------ @@ -865,7 +865,7 @@ void QucsApp::launchTool(const QString& prog, const QString& progDesc, const QSt tool->setWorkingDirectory(tooldir); qDebug() << "Command :" << cmd; tool->start(cmd,args); - + if(!tool->waitForStarted(1000) ) { QMessageBox::critical(this, tr("Error"), tr("Cannot start %1 program! \n\n(%2)").arg(progDesc, cmd));