From e490f733c8783b52c60adcc271ecf74132e8b3d8 Mon Sep 17 00:00:00 2001 From: Vadim Kuznetsov Date: Sun, 25 Sep 2022 15:17:43 +0300 Subject: [PATCH] Import powercombining tool from Qt4 Qucs --- CMakeLists.txt | 1 + qucs-powercombining/CMakeLists.txt | 96 + .../bitmaps/Bagley_idealTL.svg | 566 +++++ .../bitmaps/Bagley_microstrip.svg | 510 ++++ .../bitmaps/Branchline_idealTL.svg | 578 +++++ .../bitmaps/Branchline_microstrip.svg | 510 ++++ .../bitmaps/DoubleBoxBranchline_idealTL.svg | 923 ++++++++ .../DoubleBoxBranchline_microstrip.svg | 793 +++++++ qucs-powercombining/bitmaps/Gysel_idealTL.svg | 889 +++++++ .../bitmaps/Gysel_microstrip.svg | 808 +++++++ .../bitmaps/MultistageWilkinsonLC.svg | 1877 +++++++++++++++ .../bitmaps/MultistageWilkinson_idealTL.svg | 1248 ++++++++++ .../MultistageWilkinson_microstrip.svg | 1110 +++++++++ qucs-powercombining/bitmaps/Tee_idealTL.svg | 350 +++ .../bitmaps/Tee_microstrip.svg | 284 +++ .../bitmaps/TravellingWave_idealTL.svg | 1235 ++++++++++ .../bitmaps/TravellingWave_microstrip.svg | 1101 +++++++++ qucs-powercombining/bitmaps/Tree_idealTL.svg | 1432 +++++++++++ .../bitmaps/Tree_microstrip.svg | 1294 ++++++++++ qucs-powercombining/bitmaps/WilkinsonLC.svg | 240 ++ .../bitmaps/Wilkinson_idealTL.svg | 573 +++++ .../bitmaps/Wilkinson_microstrip.svg | 519 ++++ qucs-powercombining/bitmaps/big.qucs.xpm | 225 ++ qucs-powercombining/main.cpp | 35 + .../qucspowercombining.1.cmake.in | 51 + qucs-powercombining/qucspowercombining.1.in | 51 + qucs-powercombining/qucspowercombining.qrc | 25 + .../qucspowercombiningtool.cpp | 2085 +++++++++++++++++ qucs-powercombining/qucspowercombiningtool.h | 113 + 29 files changed, 19522 insertions(+) create mode 100644 qucs-powercombining/CMakeLists.txt create mode 100644 qucs-powercombining/bitmaps/Bagley_idealTL.svg create mode 100644 qucs-powercombining/bitmaps/Bagley_microstrip.svg create mode 100644 qucs-powercombining/bitmaps/Branchline_idealTL.svg create mode 100644 qucs-powercombining/bitmaps/Branchline_microstrip.svg create mode 100644 qucs-powercombining/bitmaps/DoubleBoxBranchline_idealTL.svg create mode 100644 qucs-powercombining/bitmaps/DoubleBoxBranchline_microstrip.svg create mode 100644 qucs-powercombining/bitmaps/Gysel_idealTL.svg create mode 100644 qucs-powercombining/bitmaps/Gysel_microstrip.svg create mode 100644 qucs-powercombining/bitmaps/MultistageWilkinsonLC.svg create mode 100644 qucs-powercombining/bitmaps/MultistageWilkinson_idealTL.svg create mode 100644 qucs-powercombining/bitmaps/MultistageWilkinson_microstrip.svg create mode 100644 qucs-powercombining/bitmaps/Tee_idealTL.svg create mode 100644 qucs-powercombining/bitmaps/Tee_microstrip.svg create mode 100644 qucs-powercombining/bitmaps/TravellingWave_idealTL.svg create mode 100644 qucs-powercombining/bitmaps/TravellingWave_microstrip.svg create mode 100644 qucs-powercombining/bitmaps/Tree_idealTL.svg create mode 100644 qucs-powercombining/bitmaps/Tree_microstrip.svg create mode 100644 qucs-powercombining/bitmaps/WilkinsonLC.svg create mode 100644 qucs-powercombining/bitmaps/Wilkinson_idealTL.svg create mode 100644 qucs-powercombining/bitmaps/Wilkinson_microstrip.svg create mode 100644 qucs-powercombining/bitmaps/big.qucs.xpm create mode 100644 qucs-powercombining/main.cpp create mode 100644 qucs-powercombining/qucspowercombining.1.cmake.in create mode 100644 qucs-powercombining/qucspowercombining.1.in create mode 100644 qucs-powercombining/qucspowercombining.qrc create mode 100644 qucs-powercombining/qucspowercombiningtool.cpp create mode 100644 qucs-powercombining/qucspowercombiningtool.h diff --git a/CMakeLists.txt b/CMakeLists.txt index 5a515a66..1d3fdb6d 100755 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -38,6 +38,7 @@ ADD_SUBDIRECTORY( qucs-attenuator ) ADD_SUBDIRECTORY( qucs-filter ) ADD_SUBDIRECTORY( library ) ADD_SUBDIRECTORY( qucs-transcalc ) +ADD_SUBDIRECTORY( qucs-powercombining ) #ADD_SUBDIRECTORY( examples ) ADD_SUBDIRECTORY( translations ) diff --git a/qucs-powercombining/CMakeLists.txt b/qucs-powercombining/CMakeLists.txt new file mode 100644 index 00000000..aeb0e3d8 --- /dev/null +++ b/qucs-powercombining/CMakeLists.txt @@ -0,0 +1,96 @@ +PROJECT(qucspowercombining 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 "2016") +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_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 +FIND_PACKAGE( Qt5 COMPONENTS Core Gui Widgets Svg REQUIRED) + +INCLUDE_DIRECTORIES( + ${Qt5Core_INCLUDE_DIRS} + ${Qt5Widgets_INCLUDE_DIRS} + ${Qt5Svg_INCLUDE_DIRS} + ) + +# bug, the find package does not seem to set the QT_LIBRARIES, do it manually +SET(QT_LIBRARIES ${Qt5Core_LIBRARIES} ${Qt5Widgets_LIBRARIES} ${Qt5Svg_LIBRARIES} ) + +ADD_DEFINITIONS(${QT_DEFINITIONS}) + +SET(QUCS-POWCOMB_SRCS + qucspowercombiningtool.cpp + main.cpp +) + +SET(QUCS-POWCOMB_HDRS + qucspowercombiningtool.h +) + +SET(QUCS-POWCOMB_MOC_HDRS + qucspowercombiningtool.h +) + +QT5_WRAP_CPP(QUCS-POWCOMB_MOC_SRCS ${QUCS-POWCOMB_MOC_HDRS}) + +SET(RESOURCES qucspowercombining.qrc) +QT5_ADD_RESOURCES(RESOURCES_SRCS ${RESOURCES}) + + +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 qucspowercombining.icns) + + # set where in the bundle to put the icns file + SET_SOURCE_FILES_PROPERTIES(${CMAKE_CURRENT_SOURCE_DIR}/../qucs/bitmaps/qucspowercombining.icns PROPERTIES MACOSX_PACKAGE_LOCATION Resources) + # include the icns file in the target + SET(QUCS-POWCOMB_SRCS ${QUCS-POWCOMB_SRCS} ${CMAKE_CURRENT_SOURCE_DIR}/../qucs/bitmaps/qucspowercombining.icns) + +ENDIF(APPLE) + +ADD_EXECUTABLE(${QUCS_NAME}powercombining MACOSX_BUNDLE WIN32 + ${QUCS-POWCOMB_SRCS} + ${QUCS-POWCOMB_HDRS} + ${QUCS-POWCOMB_MOC_SRCS} + ${RESOURCES_SRCS} ) + +TARGET_LINK_LIBRARIES(${QUCS_NAME}powercombining ${QT_LIBRARIES}) + +INSTALL(TARGETS ${QUCS_NAME}powercombining + BUNDLE DESTINATION bin COMPONENT Runtime + RUNTIME DESTINATION bin COMPONENT Runtime + ) + diff --git a/qucs-powercombining/bitmaps/Bagley_idealTL.svg b/qucs-powercombining/bitmaps/Bagley_idealTL.svg new file mode 100644 index 00000000..0d6804a1 --- /dev/null +++ b/qucs-powercombining/bitmaps/Bagley_idealTL.svg @@ -0,0 +1,566 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/Bagley_microstrip.svg b/qucs-powercombining/bitmaps/Bagley_microstrip.svg new file mode 100644 index 00000000..485210d1 --- /dev/null +++ b/qucs-powercombining/bitmaps/Bagley_microstrip.svg @@ -0,0 +1,510 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/Branchline_idealTL.svg b/qucs-powercombining/bitmaps/Branchline_idealTL.svg new file mode 100644 index 00000000..aefeef86 --- /dev/null +++ b/qucs-powercombining/bitmaps/Branchline_idealTL.svg @@ -0,0 +1,578 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/Branchline_microstrip.svg b/qucs-powercombining/bitmaps/Branchline_microstrip.svg new file mode 100644 index 00000000..f556a2e1 --- /dev/null +++ b/qucs-powercombining/bitmaps/Branchline_microstrip.svg @@ -0,0 +1,510 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/DoubleBoxBranchline_idealTL.svg b/qucs-powercombining/bitmaps/DoubleBoxBranchline_idealTL.svg new file mode 100644 index 00000000..fa27d01a --- /dev/null +++ b/qucs-powercombining/bitmaps/DoubleBoxBranchline_idealTL.svg @@ -0,0 +1,923 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/DoubleBoxBranchline_microstrip.svg b/qucs-powercombining/bitmaps/DoubleBoxBranchline_microstrip.svg new file mode 100644 index 00000000..868c2ef5 --- /dev/null +++ b/qucs-powercombining/bitmaps/DoubleBoxBranchline_microstrip.svg @@ -0,0 +1,793 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/Gysel_idealTL.svg b/qucs-powercombining/bitmaps/Gysel_idealTL.svg new file mode 100644 index 00000000..cad5239b --- /dev/null +++ b/qucs-powercombining/bitmaps/Gysel_idealTL.svg @@ -0,0 +1,889 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/Gysel_microstrip.svg b/qucs-powercombining/bitmaps/Gysel_microstrip.svg new file mode 100644 index 00000000..af9bdc1b --- /dev/null +++ b/qucs-powercombining/bitmaps/Gysel_microstrip.svg @@ -0,0 +1,808 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/MultistageWilkinsonLC.svg b/qucs-powercombining/bitmaps/MultistageWilkinsonLC.svg new file mode 100644 index 00000000..c487592d --- /dev/null +++ b/qucs-powercombining/bitmaps/MultistageWilkinsonLC.svg @@ -0,0 +1,1877 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/MultistageWilkinson_idealTL.svg b/qucs-powercombining/bitmaps/MultistageWilkinson_idealTL.svg new file mode 100644 index 00000000..fb7d7ff8 --- /dev/null +++ b/qucs-powercombining/bitmaps/MultistageWilkinson_idealTL.svg @@ -0,0 +1,1248 @@ + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/MultistageWilkinson_microstrip.svg b/qucs-powercombining/bitmaps/MultistageWilkinson_microstrip.svg new file mode 100644 index 00000000..021f53a5 --- /dev/null +++ b/qucs-powercombining/bitmaps/MultistageWilkinson_microstrip.svg @@ -0,0 +1,1110 @@ + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/Tee_idealTL.svg b/qucs-powercombining/bitmaps/Tee_idealTL.svg new file mode 100644 index 00000000..2090f2d8 --- /dev/null +++ b/qucs-powercombining/bitmaps/Tee_idealTL.svg @@ -0,0 +1,350 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/Tee_microstrip.svg b/qucs-powercombining/bitmaps/Tee_microstrip.svg new file mode 100644 index 00000000..c135d711 --- /dev/null +++ b/qucs-powercombining/bitmaps/Tee_microstrip.svg @@ -0,0 +1,284 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/TravellingWave_idealTL.svg b/qucs-powercombining/bitmaps/TravellingWave_idealTL.svg new file mode 100644 index 00000000..bc30b9cc --- /dev/null +++ b/qucs-powercombining/bitmaps/TravellingWave_idealTL.svg @@ -0,0 +1,1235 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/TravellingWave_microstrip.svg b/qucs-powercombining/bitmaps/TravellingWave_microstrip.svg new file mode 100644 index 00000000..8d3cde3c --- /dev/null +++ b/qucs-powercombining/bitmaps/TravellingWave_microstrip.svg @@ -0,0 +1,1101 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/Tree_idealTL.svg b/qucs-powercombining/bitmaps/Tree_idealTL.svg new file mode 100644 index 00000000..3afecadb --- /dev/null +++ b/qucs-powercombining/bitmaps/Tree_idealTL.svg @@ -0,0 +1,1432 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/Tree_microstrip.svg b/qucs-powercombining/bitmaps/Tree_microstrip.svg new file mode 100644 index 00000000..6df280f0 --- /dev/null +++ b/qucs-powercombining/bitmaps/Tree_microstrip.svg @@ -0,0 +1,1294 @@ + + + + + + image/svg+xml + + Qt Svg Document + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/WilkinsonLC.svg b/qucs-powercombining/bitmaps/WilkinsonLC.svg new file mode 100644 index 00000000..8083fa37 --- /dev/null +++ b/qucs-powercombining/bitmaps/WilkinsonLC.svg @@ -0,0 +1,240 @@ + + + + + + + + image/svg+xml + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/Wilkinson_idealTL.svg b/qucs-powercombining/bitmaps/Wilkinson_idealTL.svg new file mode 100644 index 00000000..f2542c74 --- /dev/null +++ b/qucs-powercombining/bitmaps/Wilkinson_idealTL.svg @@ -0,0 +1,573 @@ + + + + + + image/svg+xml + + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/Wilkinson_microstrip.svg b/qucs-powercombining/bitmaps/Wilkinson_microstrip.svg new file mode 100644 index 00000000..75229071 --- /dev/null +++ b/qucs-powercombining/bitmaps/Wilkinson_microstrip.svg @@ -0,0 +1,519 @@ + + + + + + image/svg+xml + + + + + + Qt Svg Document + Generated with Qt + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/qucs-powercombining/bitmaps/big.qucs.xpm b/qucs-powercombining/bitmaps/big.qucs.xpm new file mode 100644 index 00000000..b69901cf --- /dev/null +++ b/qucs-powercombining/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-powercombining/main.cpp b/qucs-powercombining/main.cpp new file mode 100644 index 00000000..fcfb94d1 --- /dev/null +++ b/qucs-powercombining/main.cpp @@ -0,0 +1,35 @@ +/* + * main.cpp - Power combining tool main + * + * copyright (C) 2016 Andres Martinez-Mera + * + * 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. + * + * + */ + +#include +#include "qucspowercombiningtool.h" + +int main(int argc, char *argv[]) +{ + QApplication app(argc, argv); + QucsPowerCombiningTool *PowerCombiningTool = new QucsPowerCombiningTool(); + PowerCombiningTool->raise(); + PowerCombiningTool->resize(350, 350); + PowerCombiningTool->show(); + return app.exec(); +} diff --git a/qucs-powercombining/qucspowercombining.1.cmake.in b/qucs-powercombining/qucspowercombining.1.cmake.in new file mode 100644 index 00000000..561f6e6f --- /dev/null +++ b/qucs-powercombining/qucspowercombining.1.cmake.in @@ -0,0 +1,51 @@ +.TH QucsPowerCombiningTool "1" "September 2018" "Debian/GNU Linux" "User Commands" +.SH NAME +QucsPowerCombiningTool \- A power combiner synthesis tool. +.SH SYNOPSIS +.B qucspowercombining +[\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. + +\fBQucsPowerCombiningTool\fR is a synthesis tool focused on the design of power +combiners/dividers which is integrated into the \fBQucs\fR environment. +The GUI consists of a dialog which includes the design parameters of the combiner and +a graphical description of the combiner type in the dialog. After the synthesis is done, +the schematic is copied into the system-wide clipboard. In order to run a simulation, please create +a new schematic in \fBQucs\fR and then press CTRL-V (paste from clipboard). The schematic will be +inserted in \fBQucs\fR and ready for simulation. + +The available power combiner types are: +Wilkinson, +Multistage Wilkinson, +T-junction, +Branchline, +Double-box Branchline, +Bagley, +Gysel, +Travelling wave combiner +and Tree combiner + +.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 2018 Qucs Team +.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. + + diff --git a/qucs-powercombining/qucspowercombining.1.in b/qucs-powercombining/qucspowercombining.1.in new file mode 100644 index 00000000..fb1b48ab --- /dev/null +++ b/qucs-powercombining/qucspowercombining.1.in @@ -0,0 +1,51 @@ +.TH QucsPowerCombiningTool "1" "September 2018" "Debian/GNU Linux" "User Commands" +.SH NAME +QucsPowerCombiningTool \- A power combiner synthesis tool. +.SH SYNOPSIS +.B qucspowercombining +[\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. + +\fBQucsPowerCombiningTool\fR is a synthesis tool focused on the design of power +combiners/dividers which is integrated into the \fBQucs\fR environment. +The GUI consists of a dialog which includes the design parameters of the combiner and +a graphical description of the combiner type in the dialog. After the synthesis is done, +the schematic is copied into the system-wide clipboard. In order to run a simulation, please create +a new schematic in \fBQucs\fR and then press CTRL-V (paste from clipboard). The schematic will be +inserted in \fBQucs\fR and ready for simulation. + +The available power combiner types are: +Wilkinson, +Multistage Wilkinson, +T-junction, +Branchline, +Double-box Branchline, +Bagley, +Gysel, +Travelling wave combiner +and Tree combiner + +.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 2018 Qucs Team +.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. + + diff --git a/qucs-powercombining/qucspowercombining.qrc b/qucs-powercombining/qucspowercombining.qrc new file mode 100644 index 00000000..d80aa4dc --- /dev/null +++ b/qucs-powercombining/qucspowercombining.qrc @@ -0,0 +1,25 @@ + + + bitmaps/Wilkinson_idealTL.svg + bitmaps/Wilkinson_microstrip.svg + bitmaps/WilkinsonLC.svg + bitmaps/MultistageWilkinson_idealTL.svg + bitmaps/MultistageWilkinson_microstrip.svg + bitmaps/MultistageWilkinsonLC.svg + bitmaps/Tree_idealTL.svg + bitmaps/Tree_microstrip.svg + bitmaps/TravellingWave_idealTL.svg + bitmaps/TravellingWave_microstrip.svg + bitmaps/Branchline_idealTL.svg + bitmaps/Branchline_microstrip.svg + bitmaps/DoubleBoxBranchline_idealTL.svg + bitmaps/DoubleBoxBranchline_microstrip.svg + bitmaps/Bagley_idealTL.svg + bitmaps/Bagley_microstrip.svg + bitmaps/Gysel_idealTL.svg + bitmaps/Gysel_microstrip.svg + bitmaps/Tee_idealTL.svg + bitmaps/Tee_microstrip.svg + bitmaps/big.qucs.xpm + + diff --git a/qucs-powercombining/qucspowercombiningtool.cpp b/qucs-powercombining/qucspowercombiningtool.cpp new file mode 100644 index 00000000..bccbf9a1 --- /dev/null +++ b/qucs-powercombining/qucspowercombiningtool.cpp @@ -0,0 +1,2085 @@ +/* + * qucspowercombiningtool.cpp - Power combining tool implementation + * + * copyright (C) 2017 Andres Martinez-Mera + * + * 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. + * + * + */ +#ifdef HAVE_CONFIG_H +#include +#endif + +#include + +#include "qucspowercombiningtool.h" +#include "../qucs/qucs.h" +#include "../qucs/misc.h" +#include "../qucs-filter/material_props.h" + + + +//--------------------------------------------------------- +// Constructor. It sets up the user interface +QucsPowerCombiningTool::QucsPowerCombiningTool() +{ + setWindowTitle("Qucs Power Combining Tool " PACKAGE_VERSION); + // set application icon + setWindowIcon(QPixmap(":/images/bitmaps/big.qucs.xpm")); + centralWidget = new QWidget(); + QHBoxLayout *hbox = new QHBoxLayout(); + SpecificationsgroupBox = new QGroupBox("Implementation"); + MicrostripgroupBox = new QGroupBox("Microstrip substrate"); + ImagegroupBox = new QGroupBox("Preview"); + + //Add elements to SpecificationsgroupBox + QVBoxLayout *VboxImplementation = new QVBoxLayout(); + //Topology + QHBoxLayout *hboxImpl1 = new QHBoxLayout(); + TopoLabel = new QLabel("Topology"); + TopoCombo = new QComboBox(); + TopoCombo->addItem("Wilkinson"); + TopoCombo->addItem("Multistage Wilkinson"); + TopoCombo->addItem("T-junction"); + TopoCombo->addItem("Branchline"); + TopoCombo->addItem("Double box branchline"); + TopoCombo->addItem("Bagley"); + TopoCombo->addItem("Gysel"); + TopoCombo->addItem("Travelling Wave"); + TopoCombo->addItem("Tree"); + hboxImpl1->addWidget(TopoLabel); + hboxImpl1->addWidget(TopoCombo); + VboxImplementation->addLayout(hboxImpl1); + + //Number of outputs + QHBoxLayout *hboxImpl2 = new QHBoxLayout(); + NLabel =new QLabel("Number of outputs"); + BranchesCombo = new QComboBox(); + BranchesCombo->addItem("2"); + BranchesCombo->setEnabled(false); + BranchesCombo->setFixedWidth(60); + hboxImpl2->addWidget(NLabel); + hboxImpl2->addWidget(BranchesCombo); + VboxImplementation->addLayout(hboxImpl2); + + + //Ref impedance + QHBoxLayout *hboxImpl3 = new QHBoxLayout(); + RefImp = new QLabel("Z0"); + RefImplineEdit = new QLineEdit("50"); + RefImplineEdit->setFixedWidth(40); + OhmLabel = new QLabel(QChar(0xa9, 0x03)); + OhmLabel->setFixedWidth(15); + hboxImpl3->addWidget(RefImp); + hboxImpl3->addWidget(RefImplineEdit); + hboxImpl3->addWidget(OhmLabel); + VboxImplementation->addLayout(hboxImpl3); + + + //Frequency + QHBoxLayout *hboxImpl4 = new QHBoxLayout(); + FreqLabel = new QLabel("Frequency"); + FreqlineEdit = new QLineEdit("1"); + FreqlineEdit->setFixedWidth(40); + FreqScaleCombo=new QComboBox(); + FreqScaleCombo->addItem("GHz"); + FreqScaleCombo->addItem("MHz"); + FreqScaleCombo->addItem("kHz"); + FreqScaleCombo->addItem("Hz"); + FreqScaleCombo->setFixedWidth(60); + hboxImpl4->addWidget(FreqLabel); + hboxImpl4->addWidget(FreqlineEdit); + hboxImpl4->addWidget(FreqScaleCombo); + VboxImplementation->addLayout(hboxImpl4); + + // Output power ratio + QHBoxLayout *hboxImpl5 = new QHBoxLayout(); + K1Label = new QLabel("Output Power ratio"); + K1lineEdit=new QLineEdit("0"); + K1lineEdit->setFixedWidth(40); + K1LabeldB = new QLabel("dB"); + K1LabeldB->setFixedWidth(20); + hboxImpl5->addWidget(K1Label); + hboxImpl5->addWidget(K1lineEdit); + hboxImpl5->addWidget(K1LabeldB); + VboxImplementation->addLayout(hboxImpl5); + + // Number of stages + QHBoxLayout *hboxImpl5_ = new QHBoxLayout(); + NStagesCombo = new QComboBox(); + NStagesCombo->addItem("2"); + NStagesCombo->setFixedWidth(60); + NStagesLabel = new QLabel("Number of stages"); + hboxImpl5_->addWidget(NStagesLabel); + hboxImpl5_->addWidget(NStagesCombo); + VboxImplementation->addLayout(hboxImpl5_); + NStagesLabel->hide(); + NStagesCombo->hide(); + + //Ideal transmission line attenuation coeffient + QHBoxLayout *hboxImplAlpha = new QHBoxLayout(); + AlphaLabel = new QLabel("Attenuation coefficient"); + AlphalineEdit=new QLineEdit("0"); + AlphadBLabel = new QLabel("dB/m"); + hboxImplAlpha->addWidget(AlphaLabel); + hboxImplAlpha->addWidget(AlphalineEdit); + hboxImplAlpha->addWidget(AlphadBLabel); + AlphadBLabel->setVisible(false); + AlphaLabel->setVisible(false); + AlphalineEdit->setVisible(false); + VboxImplementation->addLayout(hboxImplAlpha); + + QHBoxLayout *hboxLengthUnit = new QHBoxLayout(); + UnitsLabel = new QLabel("Length unit"); + UnitsCombo=new QComboBox(); + UnitsCombo->addItem("mm"); + UnitsCombo->addItem("mil"); + UnitsCombo->addItem("um"); + UnitsCombo->addItem("nm"); + UnitsCombo->addItem("inch"); + UnitsCombo->addItem("ft"); + UnitsCombo->addItem("m"); + UnitsCombo->setFixedWidth(60); + hboxLengthUnit->addWidget(UnitsLabel); + hboxLengthUnit->addWidget(UnitsCombo); + VboxImplementation->addLayout(hboxLengthUnit); + + //Add S-param block? + QHBoxLayout *hboxImpl6 = new QHBoxLayout(); + AddSparcheckBox=new QCheckBox("Add S-parameter simulation"); + AddSparcheckBox->setChecked(true); + hboxImpl6->addWidget(AddSparcheckBox); + VboxImplementation->addLayout(hboxImpl6); + + //Implementation. Mutually exclusive radiobuttons for selecting ideal transmission lines, microstrip tech, o CLC pi aproximations of quarter-wave lines + ImplementationgroupBox = new QGroupBox("Implementation"); + QHBoxLayout *hboxImpl7 = new QHBoxLayout();//Layout for the groupbox + QVBoxLayout *vboxImpl = new QVBoxLayout();//Layout for allocating the radiobuttons + IdealTLradioButton=new QRadioButton("Ideal TL");//Synthesize Microstrip + MicrostripradioButton=new QRadioButton("Microstrip");//Synthesize Microstrip + LumpedElementsradioButton=new QRadioButton("Lumped elements");//Lumped elements implementation + //Add widgets to the layout + vboxImpl->addWidget(IdealTLradioButton); + vboxImpl->addWidget(MicrostripradioButton); + vboxImpl->addWidget(LumpedElementsradioButton); + IdealTLradioButton->setChecked(true);//Activate ideal TL by default + ImplementationgroupBox->setLayout(vboxImpl);//Add layout to the groupbox + hboxImpl7->addWidget(ImplementationgroupBox); + VboxImplementation->addLayout(hboxImpl7); + + + //Go! Button + QHBoxLayout *hboxImpl8 = new QHBoxLayout(); + GenerateButton = new QPushButton("Generate schematic"); + hboxImpl8->addWidget(GenerateButton); + VboxImplementation->addLayout(hboxImpl8); + + // Add elements to MicrostripgroupBox + QVBoxLayout *VboxMicrostrip = new QVBoxLayout(); + + //Relative permitivity + QHBoxLayout *hboxMicr1 = new QHBoxLayout(); + RelPermlabel = new QLabel("Relative permitivity"); + RelPermcomboBox = new QComboBox(); + RelPermcomboBox->setEditable(true); + const char **p = List_er; + while(*(++p)) RelPermcomboBox->addItem(*p); + hboxMicr1->addWidget(RelPermlabel); + hboxMicr1->addWidget(RelPermcomboBox); + VboxMicrostrip->addLayout(hboxMicr1); + + //Substrate height + QHBoxLayout *hboxMicr2 = new QHBoxLayout(); + SubstrateHeightlabel = new QLabel("Substrate heigth"); + SubstrateHeightlineEdit = new QLineEdit("1.0"); + SubstrateMMlabel = new QLabel("mm"); + hboxMicr2->addWidget(SubstrateHeightlabel); + hboxMicr2->addWidget(SubstrateHeightlineEdit); + hboxMicr2->addWidget(SubstrateMMlabel); + VboxMicrostrip->addLayout(hboxMicr2); + + //Metal thickness + QHBoxLayout *hboxMicr3 = new QHBoxLayout(); + ThicknessLabel = new QLabel("Metal thickness"); + ThicknesslineEdit = new QLineEdit("12.5"); + ThicknessumLabel = new QLabel("um"); + hboxMicr3->addWidget(ThicknessLabel); + hboxMicr3->addWidget(ThicknesslineEdit); + hboxMicr3->addWidget(ThicknessumLabel); + VboxMicrostrip->addLayout(hboxMicr3); + + //Minimum width + QHBoxLayout *hboxMicr4 = new QHBoxLayout(); + MinWidthLabel= new QLabel("Minimum width"); + MinWidthlineEdit = new QLineEdit("0.4"); + MinWidthmmLabel = new QLabel("mm"); + hboxMicr4->addWidget(MinWidthLabel); + hboxMicr4->addWidget(MinWidthlineEdit); + hboxMicr4->addWidget(MinWidthmmLabel); + VboxMicrostrip->addLayout(hboxMicr4); + + //Maximum width + QHBoxLayout *hboxMicr5 = new QHBoxLayout(); + MaxWidthLabel= new QLabel("Maximum width"); + MaxWidthlineEdit = new QLineEdit("5"); + MaxWidthmmLabel = new QLabel("mm"); + hboxMicr5->addWidget(MaxWidthLabel); + hboxMicr5->addWidget(MaxWidthlineEdit); + hboxMicr5->addWidget(MaxWidthmmLabel); + VboxMicrostrip->addLayout(hboxMicr5); + + //tan(delta) + QHBoxLayout *hboxMicr6 = new QHBoxLayout(); + tanDLabel = new QLabel("tanD"); + tanDlineEdit = new QLineEdit("0.0125"); + hboxMicr6->addWidget(tanDLabel); + hboxMicr6->addWidget(tanDlineEdit); + VboxMicrostrip->addLayout(hboxMicr6); + + //Resistivity + QHBoxLayout *hboxMicr7 = new QHBoxLayout(); + ResistivityLabel = new QLabel("Resistivity"); + ResistivitylineEdit = new QLineEdit("2.43902e-08"); + hboxMicr7->addWidget(ResistivityLabel); + hboxMicr7->addWidget(ResistivitylineEdit); + VboxMicrostrip->addLayout(hboxMicr7); + + //Roughness + QHBoxLayout *hboxMicr8 = new QHBoxLayout(); + RoughnessLabel = new QLabel("Roughness"); + RoughnesslineEdit = new QLineEdit("0.15e-6"); + hboxMicr8->addWidget(RoughnessLabel); + hboxMicr8->addWidget(RoughnesslineEdit); + VboxMicrostrip->addLayout(hboxMicr8); + + QSize sz; + QString s1 = ":/bitmaps/Wilkinson_idealTL.svg"; + QGridLayout * imgLayout = new QGridLayout(); + imgWidget = new QSvgWidget(s1); + sz = imgWidget->size(); + imgWidget->setFixedSize(.6*sz); + imgLayout->addWidget(imgWidget); + ImagegroupBox->setLayout(imgLayout); + imgLayout->setAlignment(imgWidget, Qt::AlignHCenter); + + SpecificationsgroupBox->setLayout(VboxImplementation); + MicrostripgroupBox->setLayout(VboxMicrostrip); + hbox->addWidget(SpecificationsgroupBox); + hbox->addWidget(ImagegroupBox); + hbox->addWidget(MicrostripgroupBox); + centralWidget->setLayout(hbox); + setCentralWidget(centralWidget); + + MicrostripgroupBox->setEnabled(false); + statusBar = new QStatusBar(); + setStatusBar(statusBar); + connect(GenerateButton, SIGNAL(clicked()), SLOT(on_GenerateButton_clicked())); + connect(TopoCombo, SIGNAL(currentIndexChanged(int)), SLOT(on_TopoCombo_currentIndexChanged(int))); + connect(MicrostripradioButton, SIGNAL(clicked()), SLOT(on_MicrostripradioButton_clicked())); + connect(LumpedElementsradioButton, SIGNAL(clicked()), SLOT(on_LCRadioButton_clicked())); + connect(IdealTLradioButton, SIGNAL(clicked()), SLOT(on_IdealTLRadioButton_clicked())); +} + +//------------------------------------------------ +// Destructor +QucsPowerCombiningTool::~QucsPowerCombiningTool() +{ +} + +//--------------------------------------------------------------------------- +// This function scales the frequency according to 'FreqScaleCombo' combobox +double QucsPowerCombiningTool::getScaleFreq() +{ + double exp=1; + switch(FreqScaleCombo->currentIndex()) + { + case 0: + exp=9; + break; + case 1: + exp=6; + break; + case 2: + exp=3; + break; + case 3: + exp=1; + break; + } + return pow(10, exp); +} + +//------------------------------------------------------------------ +// Update image +void QucsPowerCombiningTool::UpdateImage() +{ + int index = TopoCombo->currentIndex(); + bool lumpedImplementation = LumpedElementsradioButton->isChecked(); + bool microstripImplementation = MicrostripradioButton->isChecked(); + + //Update image + switch (index) + { + + case 0: //Wilkinson + if (lumpedImplementation) + { + imgWidget->load(QString(":/bitmaps/WilkinsonLC.svg")); + } + else + { + (microstripImplementation) ? imgWidget->load(QString(":/bitmaps/Wilkinson_microstrip.svg")) + : imgWidget->load(QString(":/bitmaps/Wilkinson_idealTL.svg")); + } + break; + case 1: //Mutistage Wilkinson + if (lumpedImplementation) + { + imgWidget->load(QString(":/bitmaps/MultistageWilkinsonLC.svg")); + } + else + { + (microstripImplementation) ? imgWidget->load(QString(":/bitmaps/MultistageWilkinson_microstrip.svg")) + : imgWidget->load(QString(":/bitmaps/MultistageWilkinson_idealTL.svg")); + } + + break; + case 2: //Tee + (microstripImplementation) ? imgWidget->load(QString(":/bitmaps/Tee_microstrip.svg")) + : imgWidget->load(QString(":/bitmaps/Tee_idealTL.svg")); + break; + case 3: //Branch-line + (microstripImplementation) ? imgWidget->load(QString(":/bitmaps/Branchline_microstrip.svg")) + : imgWidget->load(QString(":/bitmaps/Branchline_idealTL.svg")); + break; + case 4: //Double-box branch-line + (microstripImplementation) ? imgWidget->load(QString(":/bitmaps/DoubleBoxBranchline_microstrip.svg")) + : imgWidget->load(QString(":/bitmaps/DoubleBoxBranchline_idealTL.svg")); + break; + case 5: //Bagley power combiner + (microstripImplementation) ? imgWidget->load(QString(":/bitmaps/Bagley_microstrip.svg")) + : imgWidget->load(QString(":/bitmaps/Bagley_idealTL.svg")); + break; + case 6: //Gysel power combiner + (microstripImplementation) ? imgWidget->load(QString(":/bitmaps/Gysel_microstrip.svg")) + : imgWidget->load(QString(":/bitmaps/Gysel_idealTL.svg")); + break; + // ------------- CORPORATE COMBINERS ----------------- + case 7: //Travelling wave + (microstripImplementation) ? imgWidget->load(QString(":/bitmaps/TravellingWave_microstrip.svg")) + : imgWidget->load(QString(":/bitmaps/TravellingWave_idealTL.svg")); + break; + case 8: //Tree + (microstripImplementation) ? imgWidget->load(QString(":/bitmaps/Tree_microstrip.svg")) + : imgWidget->load(QString(":/bitmaps/Tree_idealTL.svg")); + break; + } +} + + + + +//------------------------------------------------------------------ +// This function changes the window according to the selected topology +void QucsPowerCombiningTool::on_TopoCombo_currentIndexChanged(int index) +{ + bool lumpedImplementation = LumpedElementsradioButton->isChecked(); + + UpdateImage(); + + // Change settings + if ((index==0)|(index==2)||(index==3))//Wilkinson, Tee, Branchline + { + K1lineEdit->setVisible(true); + K1Label->setVisible(true); + K1LabeldB->setVisible(true); + BranchesCombo->clear(); + BranchesCombo->addItem("2"); + BranchesCombo->setEnabled(false); + BranchesCombo->setEditable(false); + } + else + {//The rest of power combiners do not support unequal power ratio + K1lineEdit->setVisible(false); + K1Label->setVisible(false); + K1LabeldB->setVisible(false); + } + + if ((index == 3)||(index == 4)||(index == 6))//Branchline, double-box branchline and Gysel + {//The Gysel power combiner has only two output branches + BranchesCombo->clear(); + BranchesCombo->addItem("2"); + BranchesCombo->setEnabled(false); + BranchesCombo->setEditable(false); + } + + if ((index == 0)||(index==1))//Conventional and multistage Wilkinson + { + LumpedElementsradioButton->setEnabled(true); + } + else//Only the Wilkinson combiners support the CLC equivalent of a quarter-wavelength line + { + if (lumpedImplementation) { + // Lumped elements checkbox selected, reset to Ideal TL + IdealTLradioButton->setChecked(true); + } + LumpedElementsradioButton->setEnabled(false); + } + + if (index == 1)//Multistage Wilkinson. So far, it is not possible to implement more than 7 stages + { + NStagesCombo->clear(); + NStagesCombo->addItem("2"); + NStagesCombo->addItem("3"); + NStagesCombo->addItem("4"); + NStagesCombo->addItem("5"); + NStagesCombo->addItem("6"); + NStagesCombo->addItem("7"); + NStagesLabel->setVisible(true); + NStagesCombo->setVisible(true); + BranchesCombo->clear(); + BranchesCombo->addItem("2");//2 outputs only + BranchesCombo->setEnabled(false); + } + else//There are no more multistage combiners implemented + { + NStagesLabel->setVisible(false); + NStagesCombo->setVisible(false); + } + if(index == 5)//Bagley + { + BranchesCombo->clear(); + BranchesCombo->addItem("3"); + BranchesCombo->addItem("5"); + BranchesCombo->addItem("7"); + BranchesCombo->setEditable(true); + BranchesCombo->setEnabled(true); + } + if(index == 7)// Travelling wave + { + BranchesCombo->clear(); + BranchesCombo->addItem("3"); + BranchesCombo->addItem("4"); + BranchesCombo->addItem("5"); + BranchesCombo->addItem("6"); + BranchesCombo->setEditable(true);//Let the user to specify an arbitrary number of outputs + BranchesCombo->setEnabled(true); + } + if(index == 8)//Tree + { + BranchesCombo->clear(); + BranchesCombo->addItem("4"); + BranchesCombo->addItem("8"); + BranchesCombo->addItem("16"); + BranchesCombo->setEditable(true);//Let the user to specify an arbitrary number of outputs (power of 2) + BranchesCombo->setEnabled(true); + } +} + +//--------------------------------------------------------------- +// This function reads the input values and calls the different methods for generating a schematic +void QucsPowerCombiningTool::on_GenerateButton_clicked() +{ + double err=0; + double Z0 = RefImplineEdit->text().toDouble(); + int N=BranchesCombo->currentText().toInt(); + double K=K1lineEdit->text().toDouble();//Power ratio. Output port 1 + K = pow(10, K/20.);//Conversion to natural units + double Freq = FreqlineEdit->text().toDouble()*getScaleFreq(); + bool SP_block = AddSparcheckBox->isChecked(); + int NStages = NStagesCombo->currentText().toInt(); + tSubstrate Substrate; + double alpha = AlphalineEdit->text().toDouble();//Attenuation coefficient in dB/m + bool microcheck = MicrostripradioButton->isChecked(); + bool LumpedElements = LumpedElementsradioButton->isChecked();//Lumped element implementation? + + if (microcheck)//Substrate + { + Substrate.er=RelPermcomboBox->currentText().section(" ", 0, 0).toDouble(); + Substrate.height=SubstrateHeightlineEdit->text().toDouble()*1e-3; + Substrate.thickness=ThicknesslineEdit->text().toDouble()*1e-6; + Substrate.maxWidth=MaxWidthlineEdit->text().toDouble()*1e-3; + Substrate.minWidth=MinWidthlineEdit->text().toDouble()*1e-3; + Substrate.resistivity=ResistivitylineEdit->text().toDouble(); + Substrate.tand=tanDlineEdit->text().toDouble(); + Substrate.roughness=RoughnesslineEdit->text().toDouble(); + } + + switch (TopoCombo->currentIndex())//Topology selection + { + case 0: // Wilkinson + err = Wilkinson(Z0, Freq, K, SP_block, microcheck, Substrate, alpha, LumpedElements); + break; + case 1: // Multistage Wilkinson + err = MultistageWilkinson(Z0, Freq, NStages, SP_block, microcheck, Substrate, alpha, LumpedElements); + break; + case 2: // Tee + err = Tee(Z0, Freq, K*K, SP_block, microcheck, Substrate, alpha); + break; + case 3: //Branchline + err = Branchline(Z0, Freq, K*K, SP_block, microcheck, Substrate, alpha); + break; + case 4: //Double box branchline + err = DoubleBoxBranchline(Z0, Freq, K*K, SP_block, microcheck, Substrate, alpha); + break; + case 5: //Bagley + err = Bagley(Z0, Freq, N, SP_block, microcheck, Substrate, alpha); + break; + case 6: // Gysel + err = Gysel(Z0, Freq, SP_block, microcheck, Substrate, alpha); + break; + case 7: // Travelling wave + err = TravellingWave(Z0, Freq, N, SP_block, microcheck, Substrate, alpha); + break; + case 8: //Tree + err = Tree(Z0, Freq, N, SP_block, microcheck, Substrate, alpha); + break; + } + if(!err)//Checking errors... + { + statusBar->showMessage(tr("Ready! Use CTRL+V to paste the schematic"), 2000); + } + else + { + statusBar->showMessage(tr("Error! The network could not be generated"), 2000); + } +} + +//-------------------------------------------------------------------------------- +// This function calculates the parameters of the Wilkinson power divider according to the +// specifications. It is written outside of Wilkinson() because it is also used by the corporate power +// combining functions +QString QucsPowerCombiningTool::CalculateWilkinson(double Z0, double K) +{ + // Wilkinson divider design equations + double K2 =K*K; + double Z3 = Z0*sqrt((K2+1)/(K*K*K)); + double Z2 = K2*Z3; + double R=Z0*((K2+1)/K); + double R2 = Z0*K; + double R3 = Z0/K; + return QString("%1;%2;%3;%4;%5").arg(Z2).arg(Z3).arg(R).arg(R2).arg(R3); +} + +//----------------------------------------------------- +// This function calculates a 2Way Wilkinson divider and generates the +// schematic +int QucsPowerCombiningTool::Wilkinson(double Z0, double Freq, double K, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha, bool LumpedElements) +{ + double er, width; + double lambda4=SPEED_OF_LIGHT/(4*Freq); + QString wilkstr = CalculateWilkinson(Z0, K); + double Z2 = wilkstr.section(';', 0, 0).toDouble(); + double Z3 = wilkstr.section(';', 1, 1).toDouble(); + double R = wilkstr.section(';', 2, 2).toDouble(); + double R2 = wilkstr.section(';', 3, 3).toDouble(); + double R3 = wilkstr.section(';', 4, 4).toDouble(); + double C2, C3, CC, C2_, C3_, L2, L3, L2_, L3_, Z4, Z5; + + if (LumpedElements)//Quarter wave transmission line Pi LC equivalent + { + double w = 2*M_PI*Freq; + L2 = Z2/w; + C2 = 1./(L2*w*w); + L3 = Z3/w; + C3 = 1./(L3*w*w); + CC = C2+C3; + if (R2 != R3)//Unequal output power rate => requires matching to Z0 + { + Z4 = Z0*sqrt(K); + Z5 = Z0/sqrt(K); + L2_ = Z4/w; + L3_ = Z5/w; + C2_ = 1./(L2_*w*w); + C3_ = 1./(L3_*w*w); + //Embed the first capacitor of the Pi quarter wave equivalent in the last C of the Wilkinson structure + C2 += C2_; + C3 += C3_; + } + } + + //Qucs schematic + + QString s = "\n"; + s += "\n"; + if (SP_block) + { + //Source + s += QString("\n").arg(Z0); + s += QString("\n"); + //Output port 1 + s += QString("\n").arg(Z0); + s += QString("\n"); + //Output port 2 + s += QString("\n").arg(Z0); + s += QString("\n"); + + //S-parameter analysis component + QString freq_start = QString("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + QString freq_stop = QString("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + s += QString("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg((freq_start)).arg((freq_stop)); + s += QString("\n"); + if (microcheck)s += QString("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); + } + + if (microcheck)//Microstrip implementation + { + er = Substrate.er; + getMicrostrip(Z0, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(Z2, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(Z3, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + } + else + { + if (LumpedElements)// CLC equivalent + { + //First capacitor + s += QString("\n").arg(num2str(CC)); + s += QString("\n"); + + //Upper branch + s += QString("\n").arg(num2str(C2)); + s += QString("\n"); + s += QString("\n").arg(num2str(L2)); + //Lower branch + s += QString("\n").arg(num2str(C3)); + s += QString("\n"); + s += QString("\n").arg(num2str(L3)); + } + else + { + s += QString("\n").arg(RoundVariablePrecision(Z0)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Z0 line + s += QString("\n").arg(RoundVariablePrecision(Z2)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 1 + s += QString("\n").arg(RoundVariablePrecision(Z3)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 2 + } + } + s += QString("\n").arg(R);//Isolation resistor + if (K!=1) + {// An unequal power ratio implies that the load impedance != 50, so it requires matching + if (microcheck)//Microstrip + { + er = Substrate.er; + getMicrostrip(sqrt(Z0*R2), Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(sqrt(Z0*R3), Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + + } + else + { + if (LumpedElements)//CLC equivalent + { + // Upper branch + s += QString("\n").arg(num2str(L2_)); + s += QString("\n").arg(num2str(C2_)); + s += QString("\n"); + + // Lower branch + s += QString("\n").arg(num2str(L3_)); + s += QString("\n").arg(num2str(C3_)); + s += QString("\n"); + } + else + { + s += QString("\n").arg(RoundVariablePrecision(sqrt(Z0*R2))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Quarter wave matching output branch 1 + s += QString("\n").arg(RoundVariablePrecision(sqrt(Z0*R3))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Quarter wave matching output branch 2 + } + } + } + s += "\n"; + + //Wiring + s += "\n"; + s += QString("<0 -30 100 -30 \"\" 0 0 0>\n");//Source to Z0 line + s += QString("<160 -30 190 -30 \"\" 0 0 0 \"\">\n");//Z0 line to branches + s += QString("<190 -90 190 -30 \"\" 0 0 0 \"\">\n");//Z0 line to upper branch + s += QString("<190 -90 230 -90 \"\" 0 0 0 \"\">\n");//Z0 line (corner) to upper branch + s += QString("<290 -90 340 -90 \"\" 0 0 0 \"\">\n");//Upper branch to isolation resistor + s += QString("<340 -90 340 -50 \"\" 0 0 0 \"\">\n");//Upper branch (corner) to isolation resistor + s += QString("<190 -30 190 30 \"\" 0 0 0 \"\">\n");//Z0 line to lower branch + s += QString("<190 30 230 30 \"\" 0 0 0 \"\">\n");//Z0 line (corner) to lower branch + s += QString("<290 30 340 30 \"\" 0 0 0 \"\">\n");//Lower branch to isolation resistor + s += QString("<340 10 340 30 \"\" 0 0 0 \"\">\n");//Lower branch (corner) to isolation resistor + + if (LumpedElements) + { + s += QString("<90 -30 180 -30 \"\" 0 0 0>\n"); + s += QString("<310 -110 310 -90 \"\" 0 0 0>\n");//Upper branch + s += QString("<310 30 310 50 \"\" 0 0 0>\n");//Lower branch + } + + if (K!=1)//Unequal power split ratio => need additional matching + { + if (LumpedElements) + { + s += QString("<340 -90 380 -90 \"\" 0 0 0 \"\">\n");//Upper branch, R to L + s += QString("<440 -90 500 -90 \"\" 0 0 0 \"\">\n");//Upper branch, L to port + s += QString("<450 -110 450 -90 \"\" 0 0 0 \"\">\n");//Upper branch, L to C + s += QString("<340 30 380 30 \"\" 0 0 0 \"\">\n");//Lower branch, R to L + s += QString("<440 30 500 30 \"\" 0 0 0 \"\">\n");//Lower branch, L to port + s += QString("<450 30 450 60 \"\" 0 0 0 \"\">\n");//Lower branch, L to C + } + else//Transmission lines + { + s += QString("<340 -90 380 -90 \"\" 0 0 0 \"\">\n");//Isolation resistor to matching line. Upper branch + s += QString("<440 -90 500 -90 \"\" 0 0 0 \"\">\n");//Matching line to port 2. Upper branch + s += QString("<340 30 380 30 \"\" 0 0 0 \"\">\n");//Isolation resistor to matching line. Lowe branch + s += QString("<440 30 500 30 \"\" 0 0 0 \"\">\n");//Matching line to port 2. Lower branch + } + } + else//Equal power split ratio + { + s += QString("<340 -90 500 -90 \"\" 0 0 0 \"\">\n");//Branch 2 to Port 2 + s += QString("<340 30 500 30 \"\" 0 0 0 \"\">\n");//Branch 2 to Port 3 + } + + s += "\n"; + + QApplication::clipboard()->setText(s, QClipboard::Clipboard);//Copy into clipboard + return 0; + +} + +//----------------------------------------------------------------------------------- +// This function calculates a multistage lambda/4 matching using the Chebyshev weigthing. +// See Microwave Engineering. David Pozar. John Wiley and Sons. 4th Edition. Pg 256-261 +QString QucsPowerCombiningTool::calcChebyLines(double RL, double Z0, double gamma, int N) +{ + if (N > 7)// So far, it is only available Chebyshev weighting up to 7 sections. + // Probably, it makes no sense to use a higher number of sections because of the losses + { + QMessageBox::warning(0, QObject::tr("Error"), + QObject::tr("Chebyshev weighting for N>7 is not available")); + return QString(""); + } + QString s; + double sec_theta_m;// = cosh((1/(1.*N))*acosh((1/gamma)*fabs((RL-Z0)/(Z0+RL))) ); + //double sec_theta_m = cosh((1/(1.*N))*acosh(fabs(log(RL/Z0)/(2*gamma))) ); + (fabs(log(RL/Z0)/(2*gamma)) < 1) ? sec_theta_m = 0 : sec_theta_m = cosh((1/(1.*N))*acosh(fabs(log(RL/Z0)/(2*gamma))) ); + + double w[N]; + + switch(N)//The weights are calculated by equating the reflection coeffient formula to the N-th Chebyshev polinomial + { + case 1: + w[0] = sec_theta_m; + break; + case 2: + w[0] = sec_theta_m*sec_theta_m; + w[1] = 2*(sec_theta_m*sec_theta_m-1); + break; + case 3: + w[0] = pow(sec_theta_m,3); + w[1] = 3*(pow(sec_theta_m, 3) - sec_theta_m); + w[2] = w[1]; + break; + case 4: + w[0] = pow(sec_theta_m, 4); + w[1] = 4*sec_theta_m*sec_theta_m*(sec_theta_m*sec_theta_m-1); + w[2] =2*( 1-4*sec_theta_m*sec_theta_m+3*pow(sec_theta_m, 4)); + w[3]= w[1]; + break; + case 5: + w[0] = pow(sec_theta_m, 5); + w[1] = 5*(pow(sec_theta_m, 5) - pow(sec_theta_m, 3)); + w[2] = 10*pow(sec_theta_m, 5) - 15*pow(sec_theta_m, 3) + 5*sec_theta_m; + w[3] = w[2]; + w[4] = w[1]; + break; + case 6: + w[0] = pow(sec_theta_m, 6); + w[1] = 6*pow(sec_theta_m,4)*(sec_theta_m*sec_theta_m - 1); + w[2] = 15*pow(sec_theta_m, 6) - 24*pow(sec_theta_m, 4) + 9*sec_theta_m*sec_theta_m; + w[3] = 2*(10*pow(sec_theta_m, 6) - 18*pow(sec_theta_m, 4) + 9*sec_theta_m*sec_theta_m - 1); + w[4] = w[2]; + w[5] = w[1]; + break; + case 7: + w[0] = pow(sec_theta_m, 7); + w[1] = 7*pow(sec_theta_m, 5)*(sec_theta_m*sec_theta_m -1); + w[2] = 21*pow(sec_theta_m, 7) - 35*pow(sec_theta_m, 5) + 14*pow(sec_theta_m, 3); + w[3] = 35*pow(sec_theta_m, 7) - 70*pow(sec_theta_m, 5) + 42*pow(sec_theta_m, 3) -7*sec_theta_m ; + w[4] = w[3]; + w[5] = w[2]; + w[6] = w[1]; + break; + } + + double Zaux=Z0, Zi; + for (int i = 0; i < N; i++) + { + (RL gamma, int NStages, double Z0) +{ + double Z_, R, Zaux = Zlines.section(';', NStages-1, NStages-1).toDouble(); + QString s; + std::complex Zi = 0; + for (int i=NStages-1; i>=0;i--) + { + Z_ = abs(Zaux*(Zi + Zaux*tanh(gamma*L))/(Zaux+Zi*tanh(gamma*L))); + Zaux = Zlines.section(';', i-1, i-1).toDouble(); + R = Z0*Z_/(Z_ - Z0); + Zi = Z0; + s +=QString("%1;").arg(2*R); + } + return s; +} + +//------------------------------------------------------------------------------ +// This function synthesizes a multistage Wilkinson power divider. +// References: +// [1] A class of broadband three-port TEM-mode hybrids. Seymour B. Cohn. IEEE +// transactions on microwave theory and techniques. vol MTT-16, No 2, February 1968 +// The approximation of the isolation resistances was taken from +// http://www.mathworks.com/matlabcentral/fileexchange/22996-rf-utilities-v1-2/content/RFutils_M/bwilk.m +int QucsPowerCombiningTool::MultistageWilkinson(double Z0, double Freq, int NStages, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha, bool LumpedElements) +{ + QString Zlines = calcChebyLines(2*Z0, Z0, 0.05, NStages); + double er, width; + microcheck ? er=Substrate.er : 1; + double lambda = SPEED_OF_LIGHT/(Freq); + double lambda4=lambda/4, W; + double alpha; + double C[NStages], L[NStages]; + if (LumpedElements)//CLC pi equivalent calculation + { + double w = 2*M_PI*Freq; + int j = 0; + for (int i = NStages-1; i>= 0;i--, j++) + { + double Zi = Zlines.section(';', i, i).toDouble(); + L[j] = Zi/w; + C[j] = 1./(L[j]*w*w); + } + + } + + + + if (microcheck)//Microstrip implementation + { + double Rs = sqrt((2*M_PI*Freq*4*M_PI*1e-7)/Substrate.resistivity); + getMicrostrip(Z0, Freq, &Substrate, W, er); + alpha = Rs/(Z0*W);//Conductor attenuation coefficient in (Np/m) + } + else + { + alpha = log(pow(0.1*Alpha, 10));//Alpha is given in dB/m, then it is necessary to convert it into Np/m units + } + std::complex gamma(alpha, 2*M_PI/lambda);//It is only considered the attenation of the metal conductor since it tends to be much higher than the dielectric + QString Risol = calcMultistageWilkinsonIsolators(Zlines, lambda4, gamma, NStages, Z0); + + + QString wirestr = "\n", str; + QString s = "\n"; + s += "\n"; + if (SP_block)//Add S-param simulation + { + //Source + s += QString("\n").arg(Z0); + s += QString("\n"); + wirestr +=QString("<0 0 0 -30 \"\" 0 0 0>\n");//Vertical wire + wirestr +=QString("<0 -30 70 -30 \"\" 0 0 0>\n");//Horizontal wire + //S-parameter analysis component + QString freq_start = QString("%1%2").arg((1/NStages)*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + QString freq_stop = QString("%1%2").arg((2+1/NStages)*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + s += QString("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); + str += QString("\"S11_dB=dB(S[1,1])\" 1 "); + str += QString("\"S22_dB=dB(S[2,2])\" 1 "); + str += QString("\"S33_dB=dB(S[3,3])\" 1 "); + str += QString("\"S21_dB=dB(S[2,1])\" 1 "); + str +=QString("\"S31_dB=dB(S[3,1])\" 1 "); + s += QString("\n"); + if (microcheck)s += QString("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); + + } + + int x=100; + int spacing = 150;//Spacing between sections + double Zi, Ri; + if (microcheck)//Microstrip + { + er = Substrate.er; + getMicrostrip(Z0, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x); + } + else + { + if (LumpedElements)//LC elements. Pi CLC equivalent of a lambda/4 line + { + //First capacitor + s += QString("\n").arg(num2str(2*C[0])).arg(x); + s += QString("\n").arg(x); + wirestr +=QString("<%1 -30 %2 -30 \"\" 0 0 0>\n").arg(x-30).arg(x+30); + } + else//Ideal transmission lines + { + s += QString("\n").arg(x).arg(RoundVariablePrecision(Z0)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Z0 line + } + + } + + + x+=50;//Separation between the source transmission line and the beginning of the output branches + wirestr +=QString("<%1 -30 %2 -30 \"\" 0 0 0>\n").arg(x-20).arg(x+30);//Vertical line joining the output branches + wirestr +=QString("<%1 -90 %1 30 \"\" 0 0 0>\n").arg(x+30);//Vertical line joining the output branches + + + + int aux=1; + for (int i = NStages-1; i>= 0;i--, aux++) + { + Zi = Zlines.section(';', i, i).toDouble(); + Ri = Risol.section(';', (NStages-1)-i, (NStages-1)-i).toDouble(); + + wirestr +=QString("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+30).arg(x+70); + wirestr +=QString("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+30).arg(x+70); + + wirestr +=QString("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+130).arg(x+180); + wirestr +=QString("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+130).arg(x+180); + + //Wiring the isolation resistor + wirestr +=QString("<%1 30 %1 10 \"\" 0 0 0>\n").arg(x+160); + wirestr +=QString("<%1 -90 %1 -50 \"\" 0 0 0>\n").arg(x+160); + + if (microcheck) + { + er = Substrate.er; + getMicrostrip(Zi, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x+100); + er = Substrate.er; + getMicrostrip(Zi, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x+100); + } + else + { + if (LumpedElements)//Lumped equivalent + { + if (i == 0)//Last element + { + // Upper branch + s += QString("\n").arg(num2str(L[aux-1])).arg(x+100); + s += QString("\n").arg(num2str(C[aux-1])).arg(x+160); + wirestr +=QString("<%1 -90 %1 -120 \"\" 0 0 0>\n").arg(x+160); + s += QString("\n").arg(x+160); + + // Lower branch + s += QString("\n").arg(num2str(L[aux-1])).arg(x+100); + s += QString("\n").arg(num2str(C[aux-1])).arg(x+160); + wirestr +=QString("<%1 30 %1 60 \"\" 0 0 0>\n").arg(x+160); + s += QString("\n").arg(x+160); + + } + else + { + // Upper branch + s += QString("\n").arg(num2str(L[aux-1])).arg(x+100); + s += QString("\n").arg(num2str(C[aux]+C[aux-1])).arg(x+160); + wirestr +=QString("<%1 -90 %1 -120 \"\" 0 0 0>\n").arg(x+160); + s += QString("\n").arg(x+160); + + // Lower branch + s += QString("\n").arg(num2str(L[aux-1])).arg(x+100); + s += QString("\n").arg(num2str(C[aux]+C[aux-1])).arg(x+160); + wirestr +=QString("<%1 30 %1 60 \"\" 0 0 0>\n").arg(x+160); + s += QString("\n").arg(x+160); + } + } + else//Ideal transmission lines + { + s += QString("\n").arg(x+100).arg(RoundVariablePrecision(Zi)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Upper branch + s += QString("\n").arg(x+100).arg(RoundVariablePrecision(Zi)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Lower branch + } + } + s += QString("\n").arg(x+160).arg(RoundVariablePrecision(Ri));//Isolation resistor + x+=spacing; + + if(SP_block && (i==1))//Add output ports at the last stage + { + s += QString("\n").arg(x+300).arg(Z0); + s += QString("\n").arg(x+300); + wirestr +=QString("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+150).arg(x+300); + s += QString("\n").arg(x+300).arg(Z0); + s += QString("\n").arg(x+300); + wirestr +=QString("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+150).arg(x+300); + } + } + s += "\n"; + wirestr+="\n";; + s += wirestr; + QApplication::clipboard()->setText(s, QClipboard::Clipboard); + return 0; +} + +//------------------------------------------------------------------------------ +// This function generates the schematic of a tee power divider +// Reference: "High Efficiency RF and Microwave Solid State Power Amplifiers". Paolo Colantonio, Franco Giannini and Ernesto Limiti, 2009, Wiley +int QucsPowerCombiningTool::Tee(double Z0, double Freq, double K, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) +{ + double er, width; + double lambda4=SPEED_OF_LIGHT/(4*Freq); + + QString s = "\n"; s += "\n"; + if (SP_block) + { + //Source + s += QString("\n").arg(Z0); + s += QString("\n"); + //Output port 1 + s += QString("\n").arg(2*Z0); + s += QString("\n"); + //Output port 2 + s += QString("\n").arg(2*Z0); + s += QString("\n"); + //S-parameter analysis component + QString freq_start = QString("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + QString freq_stop = QString("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + s += QString("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); + s += QString("\n"); + if (microcheck)s += QString("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); + + } + + if (microcheck) + { + er = Substrate.er; + getMicrostrip(Z0, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(Z0*(K+1), Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(Z0*(K+1)/K, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + } + else + { + s += QString("\n").arg(RoundVariablePrecision(Z0)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Z0 line + s += QString("\n").arg(RoundVariablePrecision(Z0*(K+1))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 1 + s += QString("\n").arg(RoundVariablePrecision(Z0*(K+1)/K)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 2 + } + if (K!=1) + {// An unequal power ratio implies that the load impedance != 50, so it requires matching + if (microcheck) + { + er = Substrate.er; + getMicrostrip(sqrt(2*Z0*Z0*(K+1)), Freq, &Substrate, width, er); + s += QString("\n").arg(width).arg(lambda4/sqrt(er)); + er = Substrate.er; + getMicrostrip(sqrt(Z0*Z0*(K+1)/K), Freq, &Substrate, width, er); + s += QString("\n").arg(width).arg(lambda4/sqrt(er)); + + } + else + { + s += QString("\n").arg(RoundVariablePrecision(sqrt(2*Z0*Z0*(K+1)))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Quarter wave matching output branch 1 + s += QString("\n").arg(RoundVariablePrecision(sqrt(2*Z0*Z0*(K+1)/K))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Quarter wave matching output branch 2 + } + } + s += "\n"; + + //Wiring + s += "\n"; + s += QString("<0 -30 90 -30 \"\" 0 0 0>\n");//Source to Z0 line + s += QString("<150 -30 200 -30 \"\" 0 0 0>\n");//Source to Z0 line + s += QString("<200 30 200 -90 \"\" 0 0 0>\n");//Z0 line to branches + s += QString("<200 30 240 30 \"\" 0 0 0>\n");//Z0 line to branch 1 + s += QString("<200 -90 240 -90 \"\" 0 0 0>\n");//Z0 line to branch 2 + + s += QString("<300 30 340 30 \"\" 0 0 0>\n");//Branch 2to R + s += QString("<300 -90 340 -90 \"\" 0 0 0>\n");//Branch 1 to R + if (K!=1) + { + s += QString("<400 30 450 30 \"\" 0 0 0>\n");//Branch 2 to Port 2 + s += QString("<400 -90 450 -90 \"\" 0 0 0>\n");//Branch 2 to Port 3 + } + else + { + s += QString("<340 30 450 30 \"\" 0 0 0>\n");//Branch 2 to Port 2 + s += QString("<340 -90 450 -90 \"\" 0 0 0>\n");//Branch 2 to Port 3 + } + + s += "\n"; + + QApplication::clipboard()->setText(s, QClipboard::Clipboard); + return 0; +} + +//-------------------------------------------------------------------------------- +// This function generates the schematic of a Branch-line coupler +int QucsPowerCombiningTool::Branchline(double Z0, double Freq, double K, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) +{ + double er, width; + double lambda4=SPEED_OF_LIGHT/(4*Freq); + double ZA = Z0*sqrt(K/(K+1)); + double ZB = Z0*sqrt(K); + + QString s = "\n"; s += "\n"; + if (SP_block) + { + //Source + s += QString("\n").arg(Z0); + s += QString("\n"); + //Output port 1 + s += QString("\n").arg(Z0); + s += QString("\n"); + //Output port 2 + s += QString("\n").arg(Z0); + s += QString("\n"); + //S-parameter analysis component + QString freq_start = QString("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + QString freq_stop = QString("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + s += QString("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); + s += QString("\n"); + if (microcheck)s += QString("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); + + } + + //Branch line coupler + if (microcheck) + { + er = Substrate.er; + getMicrostrip(ZA, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(ZA, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(ZB, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(ZB, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + } + else + { + s += QString("\n").arg(RoundVariablePrecision(ZA)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Z0 line + s += QString("\n").arg(RoundVariablePrecision(ZA)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 1 + s += QString("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 2 + s += QString("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 2 + } + s += QString("\n").arg(Z0);//Isolated port + s += QString("\n"); + s += "\n"; + + //Wiring + s += "\n"; + s += QString("<150 -90 190 -90 \"\" 0 0 0 \"\">\n"); + s += QString("<150 -90 150 -50 \"\" 0 0 0 \"\">\n"); + s += QString("<290 -90 290 -50 \"\" 0 0 0 \"\">\n"); + s += QString("<250 -90 290 -90 \"\" 0 0 0 \"\">\n"); + s += QString("<250 30 290 30 \"\" 0 0 0 \"\">\n"); + s += QString("<290 10 290 30 \"\" 0 0 0 \"\">\n"); + s += QString("<150 30 190 30 \"\" 0 0 0 \"\">\n"); + s += QString("<150 10 150 30 \"\" 0 0 0 \"\">\n"); + s += QString("<290 -150 290 -90 \"\" 0 0 0 \"\">\n"); + s += QString("<290 -150 400 -150 \"\" 0 0 0 \"\">\n"); + s += QString("<150 -150 150 -90 \"\" 0 0 0 \"\">\n"); + s += QString("<50 -150 150 -150 \"\" 0 0 0 \"\">\n"); + s += QString("<150 30 150 60 \"\" 0 0 0 \"\">\n"); + s += QString("<150 30 150 60 \"\" 0 0 0 \"\">\n"); + s += QString("<50 60 150 60 \"\" 0 0 0 \"\">\n"); + s += QString("<290 30 290 60 \"\" 0 0 0 \"\">\n"); + s += QString("<290 60 400 60 \"\" 0 0 0 \"\">\n"); + s += "\n"; + + + QApplication::clipboard()->setText(s, QClipboard::Clipboard); + return 0; + +} + +//-------------------------------------------------------------------------------- +// This function generates the schematic of a double box branchline coupler. +// Reference: Kumar, S.; Danshin, Tom, "A Multisection Broadband Impedance Transforming Branchline Quad Hybrid Suitable for MMIC Realization," +// in Microwave Conference, 1992. 22nd European , vol.2, no., pp.1301-1306, 5-9 Sept. 1992 +int QucsPowerCombiningTool::DoubleBoxBranchline(double Z0, double Freq, double K, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) +{ + double er, width; + double lambda4=SPEED_OF_LIGHT/(4*Freq); + double r=1; + double t = sqrt((1+K)*r); + double ZA = Z0*sqrt(r*(t*t -r))/(t-r); + double ZD = Z0*sqrt(r*(t*t -r))/(t-1); + double ZB = Z0*sqrt(r-(r*r)/(t*t)); + + QString s = "\n"; s += "\n"; + if (SP_block) + { + //Source + s += QString("\n").arg(Z0); + s += QString("\n"); + //Output port 1 + s += QString("\n").arg(Z0); + s += QString("\n"); + //Output port 2 + s += QString("\n").arg(Z0); + s += QString("\n"); + //S-parameter analysis component + QString freq_start = QString("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + QString freq_stop = QString("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + s += QString("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); + s += QString("\n"); + if (microcheck)s += QString("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); + + } + + //Branch line coupler + if (microcheck) + { + er = Substrate.er; + getMicrostrip(ZB, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(ZB, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(ZB, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(ZB, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(ZB, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(ZA, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(ZD, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + + } + else + { + s += QString("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + s += QString("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + + s += QString("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + + + s += QString("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + s += QString("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + + s += QString("\n").arg(RoundVariablePrecision(ZA)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + s += QString("\n").arg(RoundVariablePrecision(ZD)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + } + s += QString("\n").arg(Z0);//Isolated port + s += QString("\n"); + s += "\n"; + + //Wiring + s += "\n"; + s += QString("<290 -90 290 -50 \"\" 0 0 0 \"\">\n"); + s += QString("<290 10 290 30 \"\" 0 0 0 \"\">\n"); + s += QString("<290 -90 320 -90 \"\" 0 0 0 \"\">\n"); + s += QString("<250 -90 290 -90 \"\" 0 0 0 \"\">\n"); + s += QString("<290 30 320 30 \"\" 0 0 0 \"\">\n"); + s += QString("<250 30 290 30 \"\" 0 0 0 \"\">\n"); + s += QString("<380 -90 420 -90 \"\" 0 0 0 \"\">\n"); + s += QString("<420 -90 420 -50 \"\" 0 0 0 \"\">\n"); + s += QString("<380 30 420 30 \"\" 0 0 0 \"\">\n"); + s += QString("<420 10 420 30 \"\" 0 0 0 \"\">\n"); + s += QString("<40 30 40 60 \"\" 0 0 0 \"\">\n"); + s += QString("<40 30 150 30 \"\" 0 0 0 \"\">\n"); + s += QString("<150 30 190 30 \"\" 0 0 0 \"\">\n"); + s += QString("<150 10 150 30 \"\" 0 0 0 \"\">\n"); + s += QString("<150 -90 150 -50 \"\" 0 0 0 \"\">\n"); + s += QString("<150 -90 190 -90 \"\" 0 0 0 \"\">\n"); + s += QString("<420 -160 420 -90 \"\" 0 0 0 \"\">\n"); + s += QString("<420 -160 500 -160 \"\" 0 0 0 \"\">\n"); + s += QString("<500 -160 500 -130 \"\" 0 0 0 \"\">\n"); + s += QString("<500 30 500 60 \"\" 0 0 0 \"\">\n"); + s += QString("<420 30 500 30 \"\" 0 0 0 \"\">\n"); + s += QString("<130 -160 150 -90 \"\" 0 0 0 \"\">\n"); + s += QString("<40 -160 150 -160 \"\" 0 0 0 \"\">\n"); + s += QString("<40 -160 40 -130 \"\" 0 0 0 \"\">\n"); + s += "\n"; + + + QApplication::clipboard()->setText(s, QClipboard::Clipboard); + return 0; + +} + +//----------------------------------------------------------------------------- +// This function generates the schematic of a N-way Bagley combiner (N odd) +// Reference: "High Efficiency RF and Microwave Solid State Power Amplifiers". Paolo Colantonio, Franco Giannini and Ernesto Limiti, 2009, Wiley. Pg. 411 +int QucsPowerCombiningTool::Bagley(double Z0, double Freq, int N, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) +{ + if (N % 2 == 0) + { + N++; + QString str = QString("The number of outputs must be an odd number. N=%1 will be used instead").arg(N); + QMessageBox::warning(this, tr("Bagley"), str, QMessageBox::Close); + } + double er, width; + double lambda4=SPEED_OF_LIGHT/(4*Freq); + double lambda2=lambda4*2; + + double Zbranch = 2*Z0/sqrt(N); + QString s = "\n"; + s += "\n"; + if (SP_block) + { + //Source + s += QString("\n").arg(Z0); + s += QString("\n"); + QString freq_start = QString("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + QString freq_stop = QString("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + s += QString("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); + //Equations + QString str = QString(" \"S11_dB=dB(S[1,1])\" 1 "); + for (int i=2;i<=N+1; i++) + { + str += QString("\"S%1%2_dB=dB(S[%1,1])\" 1 ").arg(i).arg(1); + str += QString("\"S%1%1_dB=dB(S[%1,%1])\" 1 ").arg(i); + } + s += QString("\n"); + if (microcheck)s += QString("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); + } + + //Input section + if (microcheck) + { + er = Substrate.er; + getMicrostrip(Zbranch, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(Zbranch, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + + } + else + { + s += QString("\n").arg(RoundVariablePrecision(Zbranch)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + s += QString("\n").arg(RoundVariablePrecision(Zbranch)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + } + //Output branches + int x = 240; + s += QString("\n").arg(x-100).arg(Z0); + s += QString("\n").arg(x-100); + for (int i=1;i\n").arg(x).arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda2/sqrt(er))); + } + else + { + s += QString("\n").arg(x).arg(RoundVariablePrecision(Zbranch)).arg(ConvertLengthFromM(lambda2)).arg(Alpha); + } + x+=100; + if (SP_block) + { + //i-th output port + s += QString("\n").arg(x).arg(Z0); + s += QString("\n").arg(x); + } + x+=100; + } + + s += "\n"; + + + //Wiring + s += "\n"; + s += QString("<0 -90 100 -90 \"\" 0 0 0>\n");//Source to lambda/4 lines + s += QString("<100 -110 100 -60 \"\" 0 0 0>\n");//lambda/4 lines + s += QString("<100 30 140 30 \"\" 0 0 0>\n");//Lower lambda/4 line to the first lambda/2 section + s += QString("<100 30 100 0 \"\" 0 0 0>\n");//Lower lambda/4 line to the first lambda/2 section + + //Wiring the rest of the lambda/2 sections + x=140; + for (int i=1;i\n").arg(x).arg(x+70); + s += QString("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+130).arg(x+200); + x+=200; + } + + s += QString("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x).arg(x+30); + s += QString("<%1 30 %1 -200 \"\" 0 0 0>\n").arg(x+30);//Final lambda/2 section to the upper lambda/4 section. Vertical line + s += QString("<100 -200 %1 -200 \"\" 0 0 0>\n").arg(x+30);//Final lambda/2 section to the upper lambda/4 section. Horizontal line + s += QString("<100 -170 100 -200 \"\" 0 0 0>\n"); + s += "\n"; + + + QApplication::clipboard()->setText(s, QClipboard::Clipboard); + return 0; + +} + +//--------------------------------------------------------------------------- +// This function generates a Gysel combiner +int QucsPowerCombiningTool::Gysel(double Z0, double Freq, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) +{ + double er, width; + double lambda4=SPEED_OF_LIGHT/(4*Freq); + + QString s = "\n"; + s += "\n"; + if (SP_block) + { + //Source + s += QString("\n").arg(Z0); + s += QString("\n"); + //Output port 1 + s += QString("\n").arg(Z0); + s += QString("\n"); + //Output port 2 + s += QString("\n").arg(Z0); + s += QString("\n"); + + //S-parameter analysis component + QString freq_start = QString("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + QString freq_stop = QString("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + s += QString("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); + s += QString("\n"); + + if (microcheck)s += QString("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); + + + } + + if (microcheck) + { + er = Substrate.er; + getMicrostrip(sqrt(2)*Z0, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(sqrt(2)*Z0, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(Z0, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(Z0, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); + er = Substrate.er; + getMicrostrip(Z0/sqrt(2), Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(2*lambda4/sqrt(er))); + + } + else + { + s += QString("\n").arg(RoundVariablePrecision(sqrt(2)*Z0)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + s += QString("\n").arg(RoundVariablePrecision(sqrt(2)*Z0)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + s += QString("\n").arg(Z0).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + s += QString("\n").arg(Z0).arg(ConvertLengthFromM(lambda4)).arg(Alpha); + s += QString("\n").arg(RoundVariablePrecision(Z0/sqrt(2))).arg(ConvertLengthFromM(2*lambda4)).arg(Alpha); + } + //Resistors + s += QString("\n").arg(Z0);//Isolation resistor + s += QString("\n"); + s += QString("\n").arg(Z0);//Isolation resistor + s += QString("\n"); + + s += "\n"; + + s += "\n"; + s += QString("<120 70 120 100 \"\" 0 0 0 \"\">\n");//Source to the lines at the input + s += QString("<120 -130 120 -100 \"\" 0 0 0 \"\">\n");//Line between the two lines at the input + s += QString("<120 -130 190 -130 \"\" 0 0 0 \"\">\n"); + s += QString("<120 -200 120 -130 \"\" 0 0 0 \"\">\n"); + s += QString("<250 -130 320 -130 \"\" 0 0 0 \"\">\n"); + s += QString("<320 -130 320 -50 \"\" 0 0 0 \"\">\n"); + s += QString("<120 100 190 100 \"\" 0 0 0 \"\">\n");//Line between the line on the top to the upper resistor + s += QString("<250 100 320 100 \"\" 0 0 0 \"\">\n");//Line between the line on the top to the lower resistor + s += QString("<320 10 320 100 \"\" 0 0 0 \"\">\n"); + s += QString("<30 -200 120 -200 \"\" 0 0 0 \"\">\n"); + s += QString("<30 -200 30 -190 \"\" 0 0 0 \"\">\n"); + s += QString("<400 -200 400 -190 \"\" 0 0 0 \"\">\n"); + s += QString("<320 -200 320 -130 \"\" 0 0 0 \"\">\n"); + s += QString("<320 -200 400 -200 \"\" 0 0 0 \"\">\n"); + s += QString("<320 100 400 100 \"\" 0 0 0 \"\">\n"); + s += QString("<30 100 120 100 \"\" 0 0 0 \"\">\n"); + s += QString("<30 100 30 110 \"\" 0 0 0 \"\">\n"); + s += QString("<120 -40 120 -30 \"\" 0 0 0 \"\">\n"); + s += QString("<120 -30 120 10 \"\" 0 0 0 \"\">\n"); + s += QString("<0 -30 120 -30 \"\" 0 0 0 \"\">\n"); + s += "\n"; + + QApplication::clipboard()->setText(s, QClipboard::Clipboard); + return 0; +} + +int QucsPowerCombiningTool::TravellingWave(double Z0, double Freq, int N, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) +{ + double er, width; + double lambda4=SPEED_OF_LIGHT/(4*Freq); + + QString wirestr = "\n", str; + QString s = "\n"; + s += "\n"; + if (SP_block) + { + //Source + s += QString("\n").arg(Z0); + s += QString("\n"); + wirestr +=QString("<0 0 0 -30 \"\" 0 0 0>\n");//Vertical wire + wirestr +=QString("<0 -30 40 -30 \"\" 0 0 0>\n");//Horizontal wire + //S-parameter analysis component + QString freq_start = QString("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + QString freq_stop = QString("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + s += QString("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); + // Equations + str = QString("\"S11_dB=dB(S[1,1])\" 1 "); + for (int i=2;i<=N+1; i++) + { + str += QString("\"S%1%2_dB=dB(S[%1,1])\" 1 ").arg(i).arg(1); + str += QString("\"S%1%1_dB=dB(S[%1,%1])\" 1 ").arg(i); + } + s += QString("\n"); + if (microcheck)s += QString("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); + + } + + QString wilkstr, aux_str, aux_str_2; + double Z2, Z3, R, R2, R3; + int x=100; + int spacing = 350; + for (int n = N-1; n>0;n--) + { + wilkstr = CalculateWilkinson(Z0, sqrt(n)); + Z2 = wilkstr.section(';', 0, 0).toDouble(); + Z3 = wilkstr.section(';', 1, 1).toDouble(); + R = wilkstr.section(';', 2, 2).toDouble(); + R2 = wilkstr.section(';', 3, 3).toDouble(); + R3 = wilkstr.section(';', 4, 4).toDouble(); + + wirestr +=QString("<%1 -30 %2 -30 \"\" 0 0 0>\n").arg(x-60).arg(x-30); + wirestr +=QString("<%1 -30 %2 -30 \"\" 0 0 0>\n").arg(x+30).arg(x+60); + + wirestr +=QString("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+60).arg(x+100); + wirestr +=QString("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+60).arg(x+100); + + wirestr +=QString("<%1 -90 %1 30 \"\" 0 0 0>\n").arg(x+60); + + wirestr +=QString("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+160).arg(x+200); + wirestr +=QString("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+160).arg(x+200); + + wirestr +=QString("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+260).arg(x+350); + wirestr +=QString("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+260).arg(x+290); + wirestr +=QString("<%1 30 %1 60 \"\" 0 0 0>\n").arg(x+350); + + //Wiring the isolation resistor + wirestr +=QString("<%1 30 %1 10 \"\" 0 0 0>\n").arg(x+190); + wirestr +=QString("<%1 -90 %1 -50 \"\" 0 0 0>\n").arg(x+190); + + if(n>1)wirestr +=QString("<%1 -90 %1 -30 \"\" 0 0 0>\n").arg(x+290); + if(n==1)wirestr +=QString("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+290).arg(x+350);//Last power combiner + + if (microcheck) + { + er = Substrate.er; + getMicrostrip(Z0, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x); + er = Substrate.er; + getMicrostrip(Z3, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x+130); + er = Substrate.er; + getMicrostrip(Z2, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x+130); + } + else + { + aux_str = ConvertLengthFromM(lambda4); + s += QString("\n").arg(x).arg(Z0).arg(aux_str).arg(Alpha);//Z0 line + s += QString("\n").arg(x+130).arg(RoundVariablePrecision(Z3)).arg(aux_str).arg(Alpha);//Output branch 1 + s += QString("\n").arg(x+130).arg(RoundVariablePrecision(Z2)).arg(aux_str).arg(Alpha);//Output branch 2 + } + s += QString("\n").arg(x+190).arg(RoundVariablePrecision(R));//Isolation resistor + + if ((R2!=50)||(R3 != 50)) + {// An unequal power ratio implies that the load impedance != 50, so it requires matching. + if (microcheck) + { + er = Substrate.er; + getMicrostrip(sqrt(Z0*R3), Freq, &Substrate, width, er); + aux_str = ConvertLengthFromM(lambda4/sqrt(er)); + aux_str_2 = ConvertLengthFromM(width); + s += QString("\n").arg(aux_str_2).arg(aux_str).arg(x+230); + + er = Substrate.er; + getMicrostrip(sqrt(Z0*R2), Freq, &Substrate, width, er); + aux_str = ConvertLengthFromM(lambda4/sqrt(er)); + aux_str_2 = ConvertLengthFromM(width); + s += QString("\n").arg(aux_str_2).arg(aux_str).arg(x+230); + + } + else + { + s += QString("\n").arg(x+230).arg(RoundVariablePrecision(sqrt(Z0*R3))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Quarter wave matching output branch 1 + s += QString("\n").arg(x+230).arg(RoundVariablePrecision(sqrt(Z0*R2))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Quarter wave matching output branch 2 + } + } + else + { + wirestr +=QString("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+200).arg(x+260); + wirestr +=QString("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+200).arg(x+260); + } + x+=spacing; + + if(SP_block)//Add output terms + { + s += QString("\n").arg(x).arg(Z0); + s += QString("\n").arg(x); + + if(n==1)//The last Wilkinson divider + { + s += QString("\n").arg(x).arg(Z0); + s += QString("\n").arg(x); + } + } + + } + s += "\n"; + wirestr+="\n";; + s += wirestr; + + QApplication::clipboard()->setText(s, QClipboard::Clipboard); + return 0; +} + + +int QucsPowerCombiningTool::Tree(double Z0, double Freq, int N, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) +{ + if ((N & (N - 1)) != 0)//Checking if the number of outputs is power of 2 + { + N = pow(2, ceil(log(N)/log(2)));//Rounding to the next power of 2 + QString str = QString("The number of outputs must be a power of 2. A %1-way combiner will be designed").arg(N); + QMessageBox::warning(this, tr("Tree combiner"), str, QMessageBox::Close); + } + double er, width; + double lambda4=SPEED_OF_LIGHT/(4*Freq); + double Zbranch = sqrt(2)*Z0; + QString wirestr = "\n", str; + QString s = "\n"; + s += "\n"; + + int x=0; + int y=60, yaux;//Separation between the output branches of a single Wilkinson divider + int sp=60, spaux;//Vertical spacing between Wilkinson splitters + int offset=0,offsetaux=offset;//Vertical coordinate of the dividers + for (int n=N/2;n>=1;n=pow(2, floor(log(n-1)/log(2)))) + {//It starts drawing the last sections so as to avoid overlapping + + if (n!=N/2) + { + offset = offsetaux + yaux+0.5*spaux; + offsetaux=offset; + } + for (int i=1; i<= n; i++) + { + if (microcheck) + { + er = Substrate.er; + getMicrostrip(Z0, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x).arg(offset); + + er = Substrate.er; + getMicrostrip(Zbranch, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x+100).arg(y+offset); + + er = Substrate.er; + getMicrostrip(Zbranch, Freq, &Substrate, width, er); + s += QString("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x+100).arg(-y+offset); + + } + else + { + s += QString("\n").arg(x).arg(offset).arg(Z0).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Z0 line + s += QString("\n").arg(x+100).arg(y+offset).arg(Zbranch).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 1 + s += QString("\n").arg(x+100).arg(-y+offset).arg(Zbranch).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 2 + } + s += QString("\n").arg(x+160).arg(offset).arg(2*Z0);//Isolation resistor + + wirestr +=QString("<%1 %3 %2 %3 \"\" 0 0 0>\n").arg(x+30).arg(x+70).arg(y+offset); + wirestr +=QString("<%1 %3 %2 %3 \"\" 0 0 0>\n").arg(x+30).arg(x+70).arg(-y+offset); + + wirestr +=QString("<%1 %2 %1 %3 \"\" 0 0 0>\n").arg(x+30).arg(y+offset).arg(-y+offset); + + wirestr +=QString("<%1 %3 %2 %3 \"\" 0 0 0>\n").arg(x+130).arg(x+170).arg(y+offset); + wirestr +=QString("<%1 %3 %2 %3 \"\" 0 0 0>\n").arg(x+130).arg(x+170).arg(-y+offset); + + //Wiring the isolation resistor + wirestr +=QString("<%1 %2 %1 %3 \"\" 0 0 0>\n").arg(x+160).arg(y+offset).arg(offset+30); + wirestr +=QString("<%1 %2 %1 %3 \"\" 0 0 0>\n").arg(x+160).arg(-y+offset).arg(offset-30); + + if (SP_block) + { + if (n==1)//Source + { + int sindex=s.indexOf("\n"); + s.insert(sindex+13, QString("\n").arg(Z0).arg(x-100).arg(offset+50)); + s += QString("\n").arg(x-100).arg(offset+80); + wirestr +=QString("<%1 %2 %1 %3 \"\" 0 0 0>\n").arg(x-100).arg(offset+20).arg(offset);//Vertical wire + wirestr +=QString("<%2 %1 %3 %1 \"\" 0 0 0>\n").arg(offset).arg(x-100).arg(x-30);//Horizontal wire + + //S-parameter analysis component + QString freq_start = QString("%1%2").arg((1/N)*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + QString freq_stop = QString("%1%2").arg((2+1/N)*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); + s += QString("<.SP SP1 1 %3 %4 0 67 0 0 \"lin\" 1 \"%1\" 1 \"%2\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop).arg(x-200).arg(offset+200); + + // Equations + str = QString("\"S11_dB=dB(S[1,1])\" 1 "); + for (int i=2;i<=N+1; i++) + { + str += QString("\"S%1%2_dB=dB(S[%1,1])\" 1 ").arg(i).arg(1); + str += QString("\"S%1%1_dB=dB(S[%1,%1])\" 1 ").arg(i); + } + s += QString("\n"); + if (microcheck)s += QString("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); + + } + if(n==N/2)//Loads + { + s += QString("\n").arg(Z0).arg(x+200).arg(offset+y); + s += QString("\n").arg(x+230).arg(offset+y); + s += QString("\n").arg(Z0).arg(x+200).arg(offset-y); + s += QString("\n").arg(x+230).arg(offset-y); + } + } + + offset+=sp+2*y; + } + yaux=y; + spaux=sp; + y=y+0.5*sp; + sp=2*yaux+sp; + x-=200; + + } + s += "\n"; + wirestr+="\n";; + s += wirestr; + QApplication::clipboard()->setText(s, QClipboard::Clipboard); + return 0; +} + + + +// FUNCTIONS FOR THE MICROSTRIP LINE SYNTHESIS. JUST COPIED FROM THE QUCS-FILTER TOOL +///////////////////////////////////////////////////////////////////////////////////////////////// +#define MAX_ERROR 1e-7 +void calcMicrostrip(tSubstrate *substrate, + double width, double freq, double& er_eff, double& zl) +{ + double a, b; + double h = substrate->height; + double t = substrate->thickness; + double er = substrate->er; + double Wh = width / h; + t /= h; + + // quasi-static models by Hammerstad + + double w1 = Wh; + if(t > 1e-100) { // width correction due to metal thickness? + a = coth(sqrt(6.517*Wh)); + b = t / M_PI * log(1.0 + 10.873127 / t / a / a); + w1 += b; + Wh += 0.5 * b * (1.0 + sech(sqrt(er - 1.0))); + } + + // relative effective permittivity + a = Wh * Wh; + b = a * a; + er_eff = -0.564 * pow((er-0.9) / (er+3.0), 0.053); + er_eff *= 1.0 + log((b + a/2704.0) / (b + 0.432)) / 49.0 + log(1.0 + a*Wh/5929.741) / 18.7; + er_eff = (er+1.0) / 2.0 + (er-1.0) / 2.0 * pow(1.0 + 10.0/Wh, er_eff); + + // characteristic impedance + zl = 6.0 + 0.2831853 * exp(-pow(30.666/Wh, 0.7528)); + zl = Z_FIELD / 2.0/M_PI * log(zl/Wh + sqrt(1.0 + 4.0/Wh/Wh)); + + // characteristic impedance (same again for "w1") + a = 6.0 + 0.2831853 * exp(-pow(30.666/w1, 0.7528)); + a = Z_FIELD / 2.0/M_PI * log(a/w1 + sqrt(1.0 + 4.0/w1/w1)); + + a /= zl; + zl /= sqrt(er_eff); + er_eff *= a * a; + + + // dispersion models by Kirschning + + freq *= h / 1e6; // normalize frequency into GHz*mm + + // relative effective permittivity + a = 0.0363 * exp(-4.6*Wh) * (1.0 - exp(-pow(freq/38.7, 4.97))); + a *= 1.0 + 2.751 * (1.0 - exp(-pow(er/15.916, 8.0))); + a = pow((0.1844 + a) * freq, 1.5763); + a *= 0.27488 + Wh*(0.6315 + 0.525 / pow(1.0+0.0157*freq, 20.0)) - 0.065683*exp(-8.7513*Wh); + a *= 0.33622 * (1.0 - exp(-0.03442*er)); + double er_freq = er - (er - er_eff) / (1.0 + a); + + // characteristic impedance + a = -0.03891 * pow(er, 1.4); + b = -0.267 * pow (Wh, 7.0); + double R7 = 1.206 - 0.3144*exp(a) * (1.0 - exp(b)); + + a = 0.016 + pow(0.0514*er, 4.524); + b = pow(freq/28.843, 12.0); + a = 5.086 * a * b / (0.3838 + 0.386*a) / (1.0 + 1.2992*b); + b = -22.2 * pow(Wh, 1.92); + a *= exp(b); + b = pow(er - 1.0, 6.0); + double R9 = a*b / (1.0 + 10.0*b); + + a = 4.766 * exp(-3.228 * pow(Wh, 0.641)); // = R3 + a = 1.0 + 1.275 + * (1.0 - exp(-0.004625*a*pow(er, 1.674) * pow(freq/18.365, 2.745))); // = R8 + + b = 0.9408 * pow(er_freq, a) - 0.9603; // = R13 + b /= (0.9408 - R9) * pow(er_eff, a) - 0.9603; + R9 = b; // = R13 / R14 + + a = 0.00044 * pow(er, 2.136) + 0.0184; // = R10 + a *= 0.707 * pow(freq/12.3, 1.097); // = R15 + a = exp(-0.026*pow(freq, 1.15656) - a); + b = pow(freq/19.47, 6.0); + b /= 1.0 + 0.0962 * b; // = R11 + b = 1.0 + 0.0503 *er*er* b * (1.0 - exp(-pow(Wh/15, 6.0))); // = R16 + R7 *= (1.0 - 1.1241 * a / b / (1.0 + 0.00245*Wh*Wh)); // = R17 + + zl *= pow(R9, R7); + er_eff = er_freq; +} + +// ------------------------------------------------------------------- +// Calculates the width 'width' and the relative effective permittivity 'er_eff' +// of a microstrip line. It uses an iterative search algorithm because +// synthesis equations doesn't exist. +void QucsPowerCombiningTool::getMicrostrip(double Z0, double freq, tSubstrate *substrate, + double &width, double &er_eff) +{ + int iteration = 0; // iteration counter + double Z0_current, Z0_result, increment; + + width = 1e-3; // start with 1mm + + do { + // compute line parameters + calcMicrostrip(substrate, width, freq, er_eff, Z0_current); + + if(fabs(Z0 - Z0_current) < MAX_ERROR) + break; // wanted value was found + + increment = width / 100.0; + width += increment; + + // compute line parameters + calcMicrostrip(substrate, width, freq, er_eff, Z0_result); + + // Newton iteration: w(n+1) = w(n) - f(w(n))/f'(w(n)) + // with f(w(n)) = Z0_current - Z0 + // and f'(w(n)) = (Z0_result - Z0_current) / increment + width -= (Z0_current - Z0) / (Z0_result - Z0_current) * increment; + if(width <= 0.0) + width = increment; + + iteration++; + } while(iteration < 150); +} +//////////////////////////////////////////////////////////////////////////////////////////////////// + + + +void QucsPowerCombiningTool::on_MicrostripradioButton_clicked() +{ + UpdateImage(); + MicrostripgroupBox->setEnabled(true); + AlphaLabel->setVisible(true); + AlphalineEdit->setVisible(true); + AlphadBLabel->setVisible(true); +} + +void QucsPowerCombiningTool::on_IdealTLRadioButton_clicked() +{ + UpdateImage(); + MicrostripgroupBox->setEnabled(false); + AlphaLabel->setVisible(false); + AlphalineEdit->setVisible(false); + AlphadBLabel->setVisible(false); +} + +void QucsPowerCombiningTool::on_LCRadioButton_clicked() +{ + UpdateImage(); + MicrostripgroupBox->setEnabled(false); + + //Hide the length unit combo + UnitsCombo->setVisible(false); + UnitsLabel->setVisible(false); + +} + +//Rounds a double number using the minimum number of decimal places +QString QucsPowerCombiningTool::RoundVariablePrecision(double val) +{ + int precision = 0;//By default, it takes 2 decimal places + while (val*pow(10, precision) < 100) precision++;//Adds another decimal place if the conversion is less than 0.1, 0.01, etc + return QString::number(val, 'F', precision);// Round to 'precision' decimals. +} + + +//This function creates a string for the transmission line length and automatically changes the unit length if the value lies outside [1,999.99] +QString QucsPowerCombiningTool::ConvertLengthFromM(double len) +{ + int index = UnitsCombo->currentIndex(); + double conv; + + do{ + conv=len; + switch (index) + { + case 1: //mils + conv *= 39370.1; + if (conv > 999.99) + { + index = 4;//inches + break; + } + if(conv < 1) + { + index = 2;//microns + break; + } + return QString("%1 mil").arg(RoundVariablePrecision(conv)); + case 2: //microns + conv *= 1e6; + if (conv > 999.99) + { + index = 0;//milimeters + break; + } + if(conv < 1) + { + index = 3;//nanometers + break; + } + return QString("%1 um").arg(RoundVariablePrecision(conv)); + case 3: //nanometers + conv *= 1e9; + if (conv > 999.99) + { + index = 2;//microns + break; + } + return QString("%1 nm").arg(RoundVariablePrecision(conv)); + case 4: //inch + conv *= 39.3701; + if (conv > 999.99) + { + index = 5;//feets + break; + } + if(conv < 1) + { + index = 1;//mils + break; + } + return QString("%1 in").arg(RoundVariablePrecision(conv)); + case 5: //ft + conv *= 3.280841666667; + if (conv > 999.99) + { + index = 6;//meters + break; + } + if(conv < 1) + { + index = 4;//inches + break; + } + return QString("%1 ft").arg(RoundVariablePrecision(conv)); + case 6: //m + if(conv < 1) + { + index = 0;//mm + break; + } + return QString("%1").arg(RoundVariablePrecision(len)); + default: //milimeters + conv *=1e3; + if (conv > 999.99) + { + index = 6;//meters + break; + } + if(conv < 1) + { + index = 2;//microns + break; + } + return QString("%1 mm").arg(RoundVariablePrecision(conv)); + } + }while(true); + return QString(""); +} + +// Copied from Qucs misc class +// Converts a double number into string adding the corresponding prefix +QString QucsPowerCombiningTool::num2str(double Num) +{ + char c = 0; + double cal = fabs(Num); + if(cal > 1e-20) { + cal = log10(cal) / 3.0; + if(cal < -0.2) cal -= 0.98; + int Expo = int(cal); + + if(Expo >= -5) if(Expo <= 4) + switch(Expo) { + case -5: c = 'f'; break; + case -4: c = 'p'; break; + case -3: c = 'n'; break; + case -2: c = 'u'; break; + case -1: c = 'm'; break; + case 1: c = 'k'; break; + case 2: c = 'M'; break; + case 3: c = 'G'; break; + case 4: c = 'T'; break; + } + + if(c) Num /= pow(10.0, double(3*Expo)); + } + + QString Str = RoundVariablePrecision(Num); + if(c) Str += c; + + return Str; +} + // Image diff --git a/qucs-powercombining/qucspowercombiningtool.h b/qucs-powercombining/qucspowercombiningtool.h new file mode 100644 index 00000000..9be38e29 --- /dev/null +++ b/qucs-powercombining/qucspowercombiningtool.h @@ -0,0 +1,113 @@ +/* + * qucspowercombiningtool.h - Power combining tool definition + * + * copyright (C) 2017 Andres Martinez-Mera + * + * 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. + * + * + */ +#ifdef HAVE_CONFIG_H +# include +#endif +#include +#include +#include +#include +struct tSubstrate { + double er; + double height; + double thickness; + double tand; + double resistivity; + double roughness; + double minWidth, maxWidth; +}; + + +static const double Z_FIELD = 376.73031346958504364963; +static const double SPEED_OF_LIGHT = 299792458.0; + +/*! coth function */ +static inline double coth(const double x) { + return (1.0 + 2.0 / (exp(2.0*(x)) - 1.0)); +} + +/*! sech function */ +static inline double sech(const double x) { + return (2.0 / (exp(x) + exp(-(x)))); +} + + +/* + References: + [1] "High Efficiency RF and Microwave Solid State Power Amplifiers". Paolo Colantonio, + Franco Giannini and Ernesto Limiti. 2009. John Wiley and Sons Inc. + [2] "RF and Microwave Transmitter Design, First Edition". Andrei Grebennikov. 2011. John Wiley and Sons Inc. +*/ + +class QucsPowerCombiningTool : public QMainWindow +{ + Q_OBJECT +public: + QucsPowerCombiningTool(); + ~QucsPowerCombiningTool(); + QLabel *NLabel,*RefImp,*FreqLabel,*K1Label, *K1LabeldB, *RelPermlabel, *SubstrateHeightlabel, *SubstrateMMlabel, *ThicknessLabel, + *ThicknessumLabel, *MinWidthLabel, *MinWidthmmLabel, *MaxWidthLabel, *MaxWidthmmLabel, *tanDLabel, + *ResistivityLabel, *RoughnessLabel,*TopoLabel, *OhmLabel, *NStagesLabel, *AlphaLabel, *AlphadBLabel, *UnitsLabel; + + QComboBox *TopoCombo, *BranchesCombo, *FreqScaleCombo, *RelPermcomboBox, *NStagesCombo, *UnitsCombo; + + QLineEdit *RefImplineEdit, *FreqlineEdit, *K1lineEdit, *SubstrateHeightlineEdit, *ThicknesslineEdit, *MinWidthlineEdit, + *MaxWidthlineEdit, *tanDlineEdit, *ResistivitylineEdit, *RoughnesslineEdit, *AlphalineEdit; + + QCheckBox *AddSparcheckBox; + QRadioButton *IdealTLradioButton, *MicrostripradioButton, *LumpedElementsradioButton; + QPushButton *GenerateButton; + QGroupBox *SpecificationsgroupBox,*MicrostripgroupBox, *ImagegroupBox, *ImplementationgroupBox; + QWidget *centralWidget; + QStatusBar *statusBar; + QGridLayout *gboxImage; + QSvgWidget *imgWidget; + +private slots: + void on_TopoCombo_currentIndexChanged(int index); + void on_GenerateButton_clicked(); + void on_MicrostripradioButton_clicked(); + void on_LCRadioButton_clicked(); + void on_IdealTLRadioButton_clicked(); + +private: + double getScaleFreq(); + void getMicrostrip(double Z0, double freq, tSubstrate *substrate, double &width, double &er_eff); + QString ConvertLengthFromM(double); + QString RoundVariablePrecision(double); + QString num2str(double); + void UpdateImage(); + QString CalculateWilkinson(double Z0, double K); + int Wilkinson(double Z0, double Freq, double K, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha, bool LumpedElements); + int MultistageWilkinson(double Z0, double Freq, int NStages, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha, bool LumpedElements); + int Tee(double Z0, double Freq, double K, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha); + int Branchline(double Z0, double Freq, double K, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha); + int DoubleBoxBranchline(double Z0, double Freq, double K, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha); + int Bagley(double Z0, double Freq, int N, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha); + int Gysel(double Z0, double Freq, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha); + int TravellingWave(double Z0, double Freq, int N, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha); + int Tree(double Z0, double Freq, int N, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha); + QString calcChebyLines(double RL, double Z0, double gamma, int NStages); + QString calcMultistageWilkinsonIsolators(QString Zlines, double L, std::complex gamma, int NStages, double Z0); + +};